package gui import ( "fmt" "sort" "strings" "sync" "github.com/go-errors/errors" "github.com/jesseduffield/gocui" "github.com/jesseduffield/lazygit/pkg/utils" "github.com/spkg/bom" ) func (gui *Gui) getCyclableViews() []string { return []string{"status", "files", "branches", "commits", "stash"} } // models/views that we can refresh const ( COMMITS = iota BRANCHES FILES STASH REFLOG TAGS REMOTES STATUS ) const ( SYNC = iota // wait until everything is done before returning ASYNC // return immediately, allowing each independent thing to update itself BLOCK_UI // wrap code in an update call to ensure UI updates all at once and keybindings aren't executed till complete ) type refreshOptions struct { then func() scope []int // e.g. []int{COMMITS, BRANCHES}. Leave empty to refresh everything mode int // one of SYNC (default), ASYNC, and BLOCK_UI } func intArrToMap(arr []int) map[int]bool { output := map[int]bool{} for _, el := range arr { output[el] = true } return output } func (gui *Gui) refreshSidePanels(options refreshOptions) error { wg := sync.WaitGroup{} f := func() { var scopeMap map[int]bool if len(options.scope) == 0 { scopeMap = intArrToMap([]int{COMMITS, BRANCHES, FILES, STASH, REFLOG, TAGS, REMOTES, STATUS}) } else { scopeMap = intArrToMap(options.scope) } if scopeMap[COMMITS] || scopeMap[BRANCHES] || scopeMap[REFLOG] { wg.Add(1) func() { if options.mode == ASYNC { go gui.refreshCommits() } else { gui.refreshCommits() } wg.Done() }() } if scopeMap[FILES] { wg.Add(1) func() { if options.mode == ASYNC { go gui.refreshFiles() } else { gui.refreshFiles() } wg.Done() }() } if scopeMap[STASH] { wg.Add(1) func() { if options.mode == ASYNC { go gui.refreshStashEntries(gui.g) } else { gui.refreshStashEntries(gui.g) } wg.Done() }() } if scopeMap[TAGS] { wg.Add(1) func() { if options.mode == ASYNC { go gui.refreshTags() } else { gui.refreshTags() } wg.Done() }() } if scopeMap[REMOTES] { wg.Add(1) func() { if options.mode == ASYNC { go gui.refreshRemotes() } else { gui.refreshRemotes() } wg.Done() }() } wg.Wait() gui.refreshStatus() if options.then != nil { options.then() } } if options.mode == BLOCK_UI { gui.g.Update(func(g *gocui.Gui) error { f() return nil }) } else { f() } return nil } func (gui *Gui) nextView(g *gocui.Gui, v *gocui.View) error { var focusedViewName string cyclableViews := gui.getCyclableViews() if v == nil || v.Name() == cyclableViews[len(cyclableViews)-1] { focusedViewName = cyclableViews[0] } else { // if we're in the commitFiles view we'll act like we're in the commits view viewName := v.Name() if viewName == "commitFiles" { viewName = "commits" } for i := range cyclableViews { if viewName == cyclableViews[i] { focusedViewName = cyclableViews[i+1] break } if i == len(cyclableViews)-1 { message := gui.Tr.TemplateLocalize( "IssntListOfViews", Teml{ "name": viewName, }, ) gui.Log.Info(message) return nil } } } focusedView, err := g.View(focusedViewName) if err != nil { panic(err) } if err := gui.resetOrigin(gui.getMainView()); err != nil { return err } return gui.switchFocus(v, focusedView) } func (gui *Gui) previousView(g *gocui.Gui, v *gocui.View) error { cyclableViews := gui.getCyclableViews() var focusedViewName string if v == nil || v.Name() == cyclableViews[0] { focusedViewName = cyclableViews[len(cyclableViews)-1] } else { // if we're in the commitFiles view we'll act like we're in the commits view viewName := v.Name() if viewName == "commitFiles" { viewName = "commits" } for i := range cyclableViews { if viewName == cyclableViews[i] { focusedViewName = cyclableViews[i-1] // TODO: make this work properly break } if i == len(cyclableViews)-1 { message := gui.Tr.TemplateLocalize( "IssntListOfViews", Teml{ "name": viewName, }, ) gui.Log.Info(message) return nil } } } focusedView, err := g.View(focusedViewName) if err != nil { panic(err) } if err := gui.resetOrigin(gui.getMainView()); err != nil { return err } return gui.switchFocus(v, focusedView) } func (gui *Gui) newLineFocused(g *gocui.Gui, v *gocui.View) error { switch v.Name() { case "menu": return gui.handleMenuSelect() case "status": return gui.handleStatusSelect() case "files": return gui.focusAndSelectFile(g, v) case "branches": branchesView := gui.getBranchesView() switch branchesView.Context { case "local-branches": return gui.handleBranchSelect() case "remotes": return gui.handleRemoteSelect() case "remote-branches": return gui.handleRemoteBranchSelect() case "tags": return gui.handleTagSelect(g, v) default: return errors.New("unknown branches panel context: " + branchesView.Context) } case "commits": return gui.handleCommitSelect(g, v) case "commitFiles": return gui.handleCommitFileSelect(g, v) case "stash": return gui.handleStashEntrySelect(g, v) case "confirmation": return nil case "commitMessage": return gui.handleCommitFocused(g, v) case "credentials": return gui.handleCredentialsViewFocused(g, v) case "main": if gui.State.MainContext == "merging" { return gui.refreshMergePanel() } v.Highlight = false return nil case "search": return nil default: panic(gui.Tr.SLocalize("NoViewMachingNewLineFocusedSwitchStatement")) } } func (gui *Gui) returnFocus(g *gocui.Gui, v *gocui.View) error { previousView, err := g.View(gui.State.PreviousView) if err != nil { // always fall back to files view if there's no 'previous' view stored previousView, err = g.View("files") if err != nil { gui.Log.Error(err) } } return gui.switchFocus(v, previousView) } func (gui *Gui) goToSideView(sideViewName string) func(g *gocui.Gui, v *gocui.View) error { return func(g *gocui.Gui, v *gocui.View) error { view, err := g.View(sideViewName) if err != nil { gui.Log.Error(err) return nil } err = gui.closePopupPanels() if err != nil { gui.Log.Error(err) return nil } return gui.switchFocus(nil, view) } } func (gui *Gui) closePopupPanels() error { gui.onNewPopupPanel() err := gui.closeConfirmationPrompt(gui.g, true) if err != nil { gui.Log.Error(err) return err } return nil } // pass in oldView = nil if you don't want to be able to return to your old view // TODO: move some of this logic into our onFocusLost and onFocus hooks func (gui *Gui) switchFocus(oldView, newView *gocui.View) error { // we assume we'll never want to return focus to a popup panel i.e. // we should never stack popup panels if oldView != nil && !gui.isPopupPanel(oldView.Name()) { gui.State.PreviousView = oldView.Name() } gui.Log.Info("setting highlight to true for view" + newView.Name()) message := gui.Tr.TemplateLocalize( "newFocusedViewIs", Teml{ "newFocusedView": newView.Name(), }, ) gui.Log.Info(message) if _, err := gui.g.SetCurrentView(newView.Name()); err != nil { return err } if _, err := gui.g.SetViewOnTop(newView.Name()); err != nil { return err } gui.g.Cursor = newView.Editable if err := gui.renderPanelOptions(); err != nil { return err } return gui.newLineFocused(gui.g, newView) } func (gui *Gui) resetOrigin(v *gocui.View) error { _ = v.SetCursor(0, 0) return v.SetOrigin(0, 0) } func (gui *Gui) cleanString(s string) string { output := string(bom.Clean([]byte(s))) return utils.NormalizeLinefeeds(output) } func (gui *Gui) setViewContent(v *gocui.View, s string) { v.Clear() fmt.Fprint(v, gui.cleanString(s)) } // renderString resets the origin of a view and sets its content func (gui *Gui) renderString(viewName, s string) { gui.g.Update(func(*gocui.Gui) error { v, err := gui.g.View(viewName) if err != nil { return nil // return gracefully if view has been deleted } if err := v.SetOrigin(0, 0); err != nil { return err } if err := v.SetCursor(0, 0); err != nil { return err } gui.setViewContent(v, s) return nil }) } func (gui *Gui) optionsMapToString(optionsMap map[string]string) string { optionsArray := make([]string, 0) for key, description := range optionsMap { optionsArray = append(optionsArray, key+": "+description) } sort.Strings(optionsArray) return strings.Join(optionsArray, ", ") } func (gui *Gui) renderOptionsMap(optionsMap map[string]string) error { gui.renderString("options", gui.optionsMapToString(optionsMap)) return nil } // TODO: refactor properly // i'm so sorry but had to add this getBranchesView func (gui *Gui) getFilesView() *gocui.View { v, _ := gui.g.View("files") return v } func (gui *Gui) getCommitsView() *gocui.View { v, _ := gui.g.View("commits") return v } func (gui *Gui) getCommitMessageView() *gocui.View { v, _ := gui.g.View("commitMessage") return v } func (gui *Gui) getBranchesView() *gocui.View { v, _ := gui.g.View("branches") return v } func (gui *Gui) getMainView() *gocui.View { v, _ := gui.g.View("main") return v } func (gui *Gui) getSecondaryView() *gocui.View { v, _ := gui.g.View("secondary") return v } func (gui *Gui) getStashView() *gocui.View { v, _ := gui.g.View("stash") return v } func (gui *Gui) getCommitFilesView() *gocui.View { v, _ := gui.g.View("commitFiles") return v } func (gui *Gui) getMenuView() *gocui.View { v, _ := gui.g.View("menu") return v } func (gui *Gui) getSearchView() *gocui.View { v, _ := gui.g.View("search") return v } func (gui *Gui) getStatusView() *gocui.View { v, _ := gui.g.View("status") return v } func (gui *Gui) trimmedContent(v *gocui.View) string { return strings.TrimSpace(v.Buffer()) } func (gui *Gui) currentViewName() string { currentView := gui.g.CurrentView() if currentView == nil { return "" } return currentView.Name() } func (gui *Gui) resizeCurrentPopupPanel(g *gocui.Gui) error { v := g.CurrentView() if gui.isPopupPanel(v.Name()) { return gui.resizePopupPanel(g, v) } return nil } func (gui *Gui) resizePopupPanel(g *gocui.Gui, v *gocui.View) error { // If the confirmation panel is already displayed, just resize the width, // otherwise continue content := v.Buffer() x0, y0, x1, y1 := gui.getConfirmationPanelDimensions(g, v.Wrap, content) vx0, vy0, vx1, vy1 := v.Dimensions() if vx0 == x0 && vy0 == y0 && vx1 == x1 && vy1 == y1 { return nil } gui.Log.Info(gui.Tr.SLocalize("resizingPopupPanel")) _, err := g.SetView(v.Name(), x0, y0, x1, y1, 0) return err } func (gui *Gui) changeSelectedLine(line *int, total int, change int) { // TODO: find out why we're doing this if *line == -1 { return } if *line+change < 0 { *line = 0 } else if *line+change >= total { *line = total - 1 } else { *line += change } } func (gui *Gui) refreshSelectedLine(line *int, total int) { if *line == -1 && total > 0 { *line = 0 } else if total-1 < *line { *line = total - 1 } } func (gui *Gui) renderDisplayStrings(v *gocui.View, displayStrings [][]string) { gui.g.Update(func(g *gocui.Gui) error { list := utils.RenderDisplayStrings(displayStrings) v.Clear() fmt.Fprint(v, list) return nil }) } func (gui *Gui) renderPanelOptions() error { currentView := gui.g.CurrentView() switch currentView.Name() { case "menu": return gui.renderMenuOptions() case "main": if gui.State.MainContext == "merging" { return gui.renderMergeOptions() } } return gui.renderGlobalOptions() } func (gui *Gui) renderGlobalOptions() error { return gui.renderOptionsMap(map[string]string{ fmt.Sprintf("%s/%s", gui.getKeyDisplay("universal.scrollUpMain"), gui.getKeyDisplay("universal.scrollDownMain")): gui.Tr.SLocalize("scroll"), fmt.Sprintf("%s %s %s %s", gui.getKeyDisplay("universal.prevBlock"), gui.getKeyDisplay("universal.nextBlock"), gui.getKeyDisplay("universal.prevItem"), gui.getKeyDisplay("universal.nextItem")): gui.Tr.SLocalize("navigate"), gui.getKeyDisplay("universal.return"): gui.Tr.SLocalize("cancel"), gui.getKeyDisplay("universal.quit"): gui.Tr.SLocalize("quit"), gui.getKeyDisplay("universal.optionMenu"): gui.Tr.SLocalize("menu"), "1-5": gui.Tr.SLocalize("jump"), }) } func (gui *Gui) isPopupPanel(viewName string) bool { return viewName == "commitMessage" || viewName == "credentials" || viewName == "confirmation" || viewName == "menu" } func (gui *Gui) popupPanelFocused() bool { return gui.isPopupPanel(gui.currentViewName()) } func (gui *Gui) handleClick(v *gocui.View, itemCount int, selectedLine *int, handleSelect func(*gocui.Gui, *gocui.View) error) error { if gui.popupPanelFocused() && v != nil && !gui.isPopupPanel(v.Name()) { return nil } if _, err := gui.g.SetCurrentView(v.Name()); err != nil { return err } newSelectedLine := v.SelectedLineIdx() if newSelectedLine < 0 { newSelectedLine = 0 } if newSelectedLine > itemCount-1 { newSelectedLine = itemCount - 1 } *selectedLine = newSelectedLine return handleSelect(gui.g, v) } // often gocui wants functions in the form `func(g *gocui.Gui, v *gocui.View) error` // but sometimes we just have a function that returns an error, so this is a // convenience wrapper to give gocui what it wants. func (gui *Gui) wrappedHandler(f func() error) func(g *gocui.Gui, v *gocui.View) error { return func(g *gocui.Gui, v *gocui.View) error { return f() } } // secondaryViewFocused tells us whether it appears that the secondary view is focused. The view is actually never focused for real: we just swap the main and secondary views and then you're still focused on the main view so that we can give you access to all its keybindings for free. I will probably regret this design decision soon enough. func (gui *Gui) secondaryViewFocused() bool { return gui.State.Panels.LineByLine != nil && gui.State.Panels.LineByLine.SecondaryFocused }