mirror of
https://github.com/charmbracelet/crush.git
synced 2025-08-02 05:20:46 +03:00
This adds keybinds to select next/previous completion item using ctrl+p and ctrl+n similar to Vim's completion behavior.
503 lines
15 KiB
Go
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
|
|
}
|