Files
crush-code-agent-ide/internal/tui/tui.go
Ayman Bagabas cc5ef2912d feat(tui): completions: add select and insert keybinds
This adds keybinds to select next/previous completion item using ctrl+p
and ctrl+n similar to Vim's completion behavior.
2025-07-23 13:20:03 -04:00

503 lines
15 KiB
Go

package tui
import (
"context"
"fmt"
"github.com/charmbracelet/bubbles/v2/key"
tea "github.com/charmbracelet/bubbletea/v2"
"github.com/charmbracelet/crush/internal/app"
"github.com/charmbracelet/crush/internal/config"
"github.com/charmbracelet/crush/internal/llm/agent"
"github.com/charmbracelet/crush/internal/permission"
"github.com/charmbracelet/crush/internal/pubsub"
cmpChat "github.com/charmbracelet/crush/internal/tui/components/chat"
"github.com/charmbracelet/crush/internal/tui/components/chat/splash"
"github.com/charmbracelet/crush/internal/tui/components/completions"
"github.com/charmbracelet/crush/internal/tui/components/core"
"github.com/charmbracelet/crush/internal/tui/components/core/layout"
"github.com/charmbracelet/crush/internal/tui/components/core/status"
"github.com/charmbracelet/crush/internal/tui/components/dialogs"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/commands"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/compact"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/filepicker"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/models"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/permissions"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/quit"
"github.com/charmbracelet/crush/internal/tui/components/dialogs/sessions"
"github.com/charmbracelet/crush/internal/tui/page"
"github.com/charmbracelet/crush/internal/tui/page/chat"
"github.com/charmbracelet/crush/internal/tui/styles"
"github.com/charmbracelet/crush/internal/tui/util"
"github.com/charmbracelet/lipgloss/v2"
)
// MouseEventFilter filters mouse events based on the current focus state
// This is used with tea.WithFilter to prevent mouse scroll events from
// interfering with typing performance in the editor
func MouseEventFilter(m tea.Model, msg tea.Msg) tea.Msg {
// Only filter mouse events
switch msg.(type) {
case tea.MouseWheelMsg, tea.MouseMotionMsg:
// Check if we have an appModel and if editor is focused
if appModel, ok := m.(*appModel); ok {
if appModel.currentPage == chat.ChatPageID {
if chatPage, ok := appModel.pages[appModel.currentPage].(chat.ChatPage); ok {
// If editor is focused (not chatFocused), filter out mouse wheel/motion events
if !chatPage.IsChatFocused() {
return nil // Filter out the event
}
}
}
}
}
// Allow all other events to pass through
return msg
}
// appModel represents the main application model that manages pages, dialogs, and UI state.
type appModel struct {
wWidth, wHeight int // Window dimensions
width, height int
keyMap KeyMap
currentPage page.PageID
previousPage page.PageID
pages map[page.PageID]util.Model
loadedPages map[page.PageID]bool
// Status
status status.StatusCmp
showingFullHelp bool
app *app.App
dialog dialogs.DialogCmp
completions completions.Completions
isConfigured bool
// Chat Page Specific
selectedSessionID string // The ID of the currently selected session
}
// Init initializes the application model and returns initial commands.
func (a appModel) Init() tea.Cmd {
var cmds []tea.Cmd
cmd := a.pages[a.currentPage].Init()
cmds = append(cmds, cmd)
a.loadedPages[a.currentPage] = true
cmd = a.status.Init()
cmds = append(cmds, cmd)
cmds = append(cmds, tea.EnableMouseAllMotion)
return tea.Batch(cmds...)
}
// Update handles incoming messages and updates the application state.
func (a *appModel) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
var cmds []tea.Cmd
var cmd tea.Cmd
a.isConfigured = config.HasInitialDataConfig()
switch msg := msg.(type) {
case tea.KeyboardEnhancementsMsg:
for id, page := range a.pages {
m, pageCmd := page.Update(msg)
a.pages[id] = m.(util.Model)
if pageCmd != nil {
cmds = append(cmds, pageCmd)
}
}
return a, tea.Batch(cmds...)
case tea.WindowSizeMsg:
a.completions.Update(msg)
return a, a.handleWindowResize(msg.Width, msg.Height)
// Completions messages
case completions.OpenCompletionsMsg, completions.FilterCompletionsMsg, completions.CloseCompletionsMsg:
u, completionCmd := a.completions.Update(msg)
a.completions = u.(completions.Completions)
switch msg := msg.(type) {
case completions.OpenCompletionsMsg:
x, _ := a.completions.Position()
if a.completions.Width()+x >= a.wWidth {
// Adjust X position to fit in the window.
msg.X = a.wWidth - a.completions.Width() - 1
u, completionCmd = a.completions.Update(msg)
a.completions = u.(completions.Completions)
}
}
return a, completionCmd
// Dialog messages
case dialogs.OpenDialogMsg, dialogs.CloseDialogMsg:
u, completionCmd := a.completions.Update(completions.CloseCompletionsMsg{})
a.completions = u.(completions.Completions)
u, dialogCmd := a.dialog.Update(msg)
a.dialog = u.(dialogs.DialogCmp)
return a, tea.Batch(completionCmd, dialogCmd)
case commands.ShowArgumentsDialogMsg:
return a, util.CmdHandler(
dialogs.OpenDialogMsg{
Model: commands.NewCommandArgumentsDialog(
msg.CommandID,
msg.Content,
msg.ArgNames,
),
},
)
// Page change messages
case page.PageChangeMsg:
return a, a.moveToPage(msg.ID)
// Status Messages
case util.InfoMsg, util.ClearStatusMsg:
s, statusCmd := a.status.Update(msg)
a.status = s.(status.StatusCmp)
cmds = append(cmds, statusCmd)
return a, tea.Batch(cmds...)
// Session
case cmpChat.SessionSelectedMsg:
a.selectedSessionID = msg.ID
case cmpChat.SessionClearedMsg:
a.selectedSessionID = ""
// Commands
case commands.SwitchSessionsMsg:
return a, func() tea.Msg {
allSessions, _ := a.app.Sessions.List(context.Background())
return dialogs.OpenDialogMsg{
Model: sessions.NewSessionDialogCmp(allSessions, a.selectedSessionID),
}
}
case commands.SwitchModelMsg:
return a, util.CmdHandler(
dialogs.OpenDialogMsg{
Model: models.NewModelDialogCmp(),
},
)
// Compact
case commands.CompactMsg:
return a, util.CmdHandler(dialogs.OpenDialogMsg{
Model: compact.NewCompactDialogCmp(a.app.CoderAgent, msg.SessionID, true),
})
// Model Switch
case models.ModelSelectedMsg:
config.Get().UpdatePreferredModel(msg.ModelType, msg.Model)
// Update the agent with the new model/provider configuration
if err := a.app.UpdateAgentModel(); err != nil {
return a, util.ReportError(fmt.Errorf("model changed to %s but failed to update agent: %v", msg.Model.Model, err))
}
modelTypeName := "large"
if msg.ModelType == config.SelectedModelTypeSmall {
modelTypeName = "small"
}
return a, util.ReportInfo(fmt.Sprintf("%s model changed to %s", modelTypeName, msg.Model.Model))
// File Picker
case chat.OpenFilePickerMsg:
if a.dialog.ActiveDialogID() == filepicker.FilePickerID {
// If the commands dialog is already open, close it
return a, util.CmdHandler(dialogs.CloseDialogMsg{})
}
return a, util.CmdHandler(dialogs.OpenDialogMsg{
Model: filepicker.NewFilePickerCmp(a.app.Config().WorkingDir()),
})
// Permissions
case pubsub.Event[permission.PermissionRequest]:
return a, util.CmdHandler(dialogs.OpenDialogMsg{
Model: permissions.NewPermissionDialogCmp(msg.Payload),
})
case permissions.PermissionResponseMsg:
switch msg.Action {
case permissions.PermissionAllow:
a.app.Permissions.Grant(msg.Permission)
case permissions.PermissionAllowForSession:
a.app.Permissions.GrantPersistent(msg.Permission)
case permissions.PermissionDeny:
a.app.Permissions.Deny(msg.Permission)
}
return a, nil
// Agent Events
case pubsub.Event[agent.AgentEvent]:
payload := msg.Payload
// Forward agent events to dialogs
if a.dialog.HasDialogs() && a.dialog.ActiveDialogID() == compact.CompactDialogID {
u, dialogCmd := a.dialog.Update(payload)
a.dialog = u.(dialogs.DialogCmp)
cmds = append(cmds, dialogCmd)
}
// Handle auto-compact logic
if payload.Done && payload.Type == agent.AgentEventTypeResponse && a.selectedSessionID != "" {
// Get current session to check token usage
session, err := a.app.Sessions.Get(context.Background(), a.selectedSessionID)
if err == nil {
model := a.app.CoderAgent.Model()
contextWindow := model.ContextWindow
tokens := session.CompletionTokens + session.PromptTokens
if (tokens >= int64(float64(contextWindow)*0.95)) && !config.Get().Options.DisableAutoSummarize { // Show compact confirmation dialog
cmds = append(cmds, util.CmdHandler(dialogs.OpenDialogMsg{
Model: compact.NewCompactDialogCmp(a.app.CoderAgent, a.selectedSessionID, false),
}))
}
}
}
return a, tea.Batch(cmds...)
case splash.OnboardingCompleteMsg:
a.isConfigured = config.HasInitialDataConfig()
updated, pageCmd := a.pages[a.currentPage].Update(msg)
a.pages[a.currentPage] = updated.(util.Model)
cmds = append(cmds, pageCmd)
return a, tea.Batch(cmds...)
// Key Press Messages
case tea.KeyPressMsg:
return a, a.handleKeyPressMsg(msg)
case tea.PasteMsg:
if a.dialog.HasDialogs() {
u, dialogCmd := a.dialog.Update(msg)
a.dialog = u.(dialogs.DialogCmp)
cmds = append(cmds, dialogCmd)
} else {
updated, pageCmd := a.pages[a.currentPage].Update(msg)
a.pages[a.currentPage] = updated.(util.Model)
cmds = append(cmds, pageCmd)
}
return a, tea.Batch(cmds...)
}
s, _ := a.status.Update(msg)
a.status = s.(status.StatusCmp)
updated, cmd := a.pages[a.currentPage].Update(msg)
a.pages[a.currentPage] = updated.(util.Model)
if a.dialog.HasDialogs() {
u, dialogCmd := a.dialog.Update(msg)
a.dialog = u.(dialogs.DialogCmp)
cmds = append(cmds, dialogCmd)
}
cmds = append(cmds, cmd)
return a, tea.Batch(cmds...)
}
// handleWindowResize processes window resize events and updates all components.
func (a *appModel) handleWindowResize(width, height int) tea.Cmd {
var cmds []tea.Cmd
a.wWidth, a.wHeight = width, height
if a.showingFullHelp {
height -= 5
} else {
height -= 2
}
a.width, a.height = width, height
// Update status bar
s, cmd := a.status.Update(tea.WindowSizeMsg{Width: width, Height: height})
a.status = s.(status.StatusCmp)
cmds = append(cmds, cmd)
// Update the current page
for p, page := range a.pages {
updated, pageCmd := page.Update(tea.WindowSizeMsg{Width: width, Height: height})
a.pages[p] = updated.(util.Model)
cmds = append(cmds, pageCmd)
}
// Update the dialogs
dialog, cmd := a.dialog.Update(tea.WindowSizeMsg{Width: width, Height: height})
a.dialog = dialog.(dialogs.DialogCmp)
cmds = append(cmds, cmd)
return tea.Batch(cmds...)
}
// handleKeyPressMsg processes keyboard input and routes to appropriate handlers.
func (a *appModel) handleKeyPressMsg(msg tea.KeyPressMsg) tea.Cmd {
if a.completions.Open() {
// completions
keyMap := a.completions.KeyMap()
switch {
case key.Matches(msg, keyMap.Up), key.Matches(msg, keyMap.Down),
key.Matches(msg, keyMap.Select), key.Matches(msg, keyMap.Cancel),
key.Matches(msg, keyMap.UpInsert), key.Matches(msg, keyMap.DownInsert):
u, cmd := a.completions.Update(msg)
a.completions = u.(completions.Completions)
return cmd
}
}
switch {
// help
case key.Matches(msg, a.keyMap.Help):
a.status.ToggleFullHelp()
a.showingFullHelp = !a.showingFullHelp
return a.handleWindowResize(a.wWidth, a.wHeight)
// dialogs
case key.Matches(msg, a.keyMap.Quit):
if a.dialog.ActiveDialogID() == quit.QuitDialogID {
return tea.Quit
}
return util.CmdHandler(dialogs.OpenDialogMsg{
Model: quit.NewQuitDialog(),
})
case key.Matches(msg, a.keyMap.Commands):
// if the app is not configured show no commands
if !a.isConfigured {
return nil
}
if a.dialog.ActiveDialogID() == commands.CommandsDialogID {
return util.CmdHandler(dialogs.CloseDialogMsg{})
}
if a.dialog.HasDialogs() {
return nil
}
return util.CmdHandler(dialogs.OpenDialogMsg{
Model: commands.NewCommandDialog(a.selectedSessionID),
})
case key.Matches(msg, a.keyMap.Sessions):
// if the app is not configured show no sessions
if !a.isConfigured {
return nil
}
if a.dialog.ActiveDialogID() == sessions.SessionsDialogID {
return util.CmdHandler(dialogs.CloseDialogMsg{})
}
if a.dialog.HasDialogs() && a.dialog.ActiveDialogID() != commands.CommandsDialogID {
return nil
}
var cmds []tea.Cmd
if a.dialog.ActiveDialogID() == commands.CommandsDialogID {
// If the commands dialog is open, close it first
cmds = append(cmds, util.CmdHandler(dialogs.CloseDialogMsg{}))
}
cmds = append(cmds,
func() tea.Msg {
allSessions, _ := a.app.Sessions.List(context.Background())
return dialogs.OpenDialogMsg{
Model: sessions.NewSessionDialogCmp(allSessions, a.selectedSessionID),
}
},
)
return tea.Sequence(cmds...)
default:
if a.dialog.HasDialogs() {
u, dialogCmd := a.dialog.Update(msg)
a.dialog = u.(dialogs.DialogCmp)
return dialogCmd
} else {
updated, cmd := a.pages[a.currentPage].Update(msg)
a.pages[a.currentPage] = updated.(util.Model)
return cmd
}
}
}
// moveToPage handles navigation between different pages in the application.
func (a *appModel) moveToPage(pageID page.PageID) tea.Cmd {
if a.app.CoderAgent.IsBusy() {
// TODO: maybe remove this : For now we don't move to any page if the agent is busy
return util.ReportWarn("Agent is busy, please wait...")
}
var cmds []tea.Cmd
if _, ok := a.loadedPages[pageID]; !ok {
cmd := a.pages[pageID].Init()
cmds = append(cmds, cmd)
a.loadedPages[pageID] = true
}
a.previousPage = a.currentPage
a.currentPage = pageID
if sizable, ok := a.pages[a.currentPage].(layout.Sizeable); ok {
cmd := sizable.SetSize(a.width, a.height)
cmds = append(cmds, cmd)
}
return tea.Batch(cmds...)
}
// View renders the complete application interface including pages, dialogs, and overlays.
func (a *appModel) View() tea.View {
page := a.pages[a.currentPage]
if withHelp, ok := page.(core.KeyMapHelp); ok {
a.status.SetKeyMap(withHelp.Help())
}
pageView := page.View()
components := []string{
pageView,
}
components = append(components, a.status.View())
appView := lipgloss.JoinVertical(lipgloss.Top, components...)
layers := []*lipgloss.Layer{
lipgloss.NewLayer(appView),
}
if a.dialog.HasDialogs() {
layers = append(
layers,
a.dialog.GetLayers()...,
)
}
var cursor *tea.Cursor
if v, ok := page.(util.Cursor); ok {
cursor = v.Cursor()
}
activeView := a.dialog.ActiveModel()
if activeView != nil {
cursor = nil // Reset cursor if a dialog is active unless it implements util.Cursor
if v, ok := activeView.(util.Cursor); ok {
cursor = v.Cursor()
}
}
if a.completions.Open() && cursor != nil {
cmp := a.completions.View()
x, y := a.completions.Position()
layers = append(
layers,
lipgloss.NewLayer(cmp).X(x).Y(y),
)
}
canvas := lipgloss.NewCanvas(
layers...,
)
var view tea.View
t := styles.CurrentTheme()
view.Layer = canvas
view.BackgroundColor = t.BgBase
view.Cursor = cursor
return view
}
// New creates and initializes a new TUI application model.
func New(app *app.App) tea.Model {
chatPage := chat.New(app)
keyMap := DefaultKeyMap()
keyMap.pageBindings = chatPage.Bindings()
model := &appModel{
currentPage: chat.ChatPageID,
app: app,
status: status.NewStatusCmp(),
loadedPages: make(map[page.PageID]bool),
keyMap: keyMap,
pages: map[page.PageID]util.Model{
chat.ChatPageID: chatPage,
},
dialog: dialogs.NewDialogCmp(),
completions: completions.New(),
}
return model
}