1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-12 04:23:03 +02:00
lazygit/pkg/gui/view_helpers.go

575 lines
14 KiB
Go
Raw Normal View History

2018-08-14 11:05:26 +02:00
package gui
2018-05-26 05:23:39 +02:00
import (
"fmt"
"sort"
"strings"
2020-03-26 14:48:11 +02:00
"sync"
2018-05-26 05:23:39 +02:00
2019-11-16 07:38:38 +02:00
"github.com/go-errors/errors"
"github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/utils"
"github.com/spkg/bom"
2018-05-26 05:23:39 +02:00
)
2020-05-17 13:54:51 +02:00
func (gui *Gui) getCyclableViews() []string {
return []string{"status", "files", "branches", "commits", "stash"}
}
2018-06-06 04:17:49 +02:00
2020-03-28 02:32:31 +02:00
// models/views that we can refresh
const (
COMMITS = iota
BRANCHES
FILES
STASH
REFLOG
TAGS
REMOTES
2020-03-28 00:57:36 +02:00
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()
2020-03-28 02:32:31 +02:00
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 {
2020-03-26 14:48:11 +02:00
wg := sync.WaitGroup{}
f := func() {
var scopeMap map[int]bool
if len(options.scope) == 0 {
2020-03-28 00:57:36 +02:00
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 {
2020-03-27 12:23:42 +02:00
go gui.refreshCommits()
} else {
2020-03-27 12:23:42 +02:00
gui.refreshCommits()
}
wg.Done()
}()
}
2020-03-26 14:48:11 +02:00
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()
}()
}
2020-03-26 14:48:11 +02:00
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()
2020-03-26 14:48:11 +02:00
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()
}
2020-03-26 14:48:11 +02:00
return nil
2018-06-06 04:17:49 +02:00
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) nextView(g *gocui.Gui, v *gocui.View) error {
var focusedViewName string
2020-05-17 13:54:51 +02:00
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 {
2018-08-15 11:49:43 +02:00
message := gui.Tr.TemplateLocalize(
"IssntListOfViews",
Teml{
"name": viewName,
2018-08-15 11:49:43 +02:00
},
)
gui.Log.Info(message)
return nil
}
}
}
focusedView, err := g.View(focusedViewName)
if err != nil {
panic(err)
}
2020-03-29 04:33:54 +02:00
if err := gui.resetOrigin(gui.getMainView()); err != nil {
return err
}
2020-08-15 08:36:39 +02:00
return gui.switchFocus(v, focusedView)
2018-06-06 04:17:49 +02:00
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) previousView(g *gocui.Gui, v *gocui.View) error {
2020-05-17 13:54:51 +02:00
cyclableViews := gui.getCyclableViews()
var focusedViewName string
2018-07-22 04:58:39 +02:00
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 {
2018-08-15 11:49:43 +02:00
message := gui.Tr.TemplateLocalize(
"IssntListOfViews",
Teml{
"name": viewName,
2018-08-15 11:49:43 +02:00
},
)
gui.Log.Info(message)
return nil
}
}
}
focusedView, err := g.View(focusedViewName)
if err != nil {
panic(err)
}
2020-03-29 04:33:54 +02:00
if err := gui.resetOrigin(gui.getMainView()); err != nil {
return err
}
2020-08-15 08:36:39 +02:00
return gui.switchFocus(v, focusedView)
}
2020-08-15 09:23:16 +02:00
func (gui *Gui) newLineFocused(v *gocui.View) error {
switch v.Name() {
2018-09-05 11:12:11 +02:00
case "menu":
2020-08-15 08:53:12 +02:00
return gui.handleMenuSelect()
case "status":
2020-08-15 08:54:02 +02:00
return gui.handleStatusSelect()
case "files":
2020-08-15 08:54:48 +02:00
return gui.focusAndSelectFile()
case "branches":
2019-11-16 07:38:38 +02:00
branchesView := gui.getBranchesView()
switch branchesView.Context {
case "local-branches":
2020-08-15 08:48:35 +02:00
return gui.handleBranchSelect()
2019-11-16 07:38:38 +02:00
case "remotes":
2020-08-15 08:50:30 +02:00
return gui.handleRemoteSelect()
2019-11-16 08:35:59 +02:00
case "remote-branches":
2020-08-15 08:52:04 +02:00
return gui.handleRemoteBranchSelect()
2019-11-18 00:38:36 +02:00
case "tags":
2020-08-15 09:01:43 +02:00
return gui.handleTagSelect()
2019-11-16 07:38:38 +02:00
default:
return errors.New("unknown branches panel context: " + branchesView.Context)
}
case "commits":
2020-08-15 09:01:43 +02:00
return gui.handleCommitSelect()
2019-03-11 00:28:47 +02:00
case "commitFiles":
2020-08-15 09:01:43 +02:00
return gui.handleCommitFileSelect()
case "stash":
2020-08-15 09:01:43 +02:00
return gui.handleStashEntrySelect()
case "confirmation":
return nil
2018-08-11 07:09:37 +02:00
case "commitMessage":
2020-08-15 09:01:43 +02:00
return gui.handleCommitFocused()
case "credentials":
2020-08-15 09:01:43 +02:00
return gui.handleCredentialsViewFocused()
case "main":
2019-11-16 03:41:04 +02:00
if gui.State.MainContext == "merging" {
return gui.refreshMergePanel()
}
v.Highlight = false
return nil
case "search":
return nil
default:
panic(gui.Tr.SLocalize("NoViewMachingNewLineFocusedSwitchStatement"))
}
2018-06-06 04:17:49 +02:00
}
2020-08-15 09:23:16 +02:00
func (gui *Gui) returnFocus(v *gocui.View) error {
previousView, err := gui.g.View(gui.State.PreviousView)
if err != nil {
2018-08-23 10:43:16 +02:00
// always fall back to files view if there's no 'previous' view stored
2020-08-15 09:23:16 +02:00
previousView, err = gui.g.View("files")
2018-08-23 10:43:16 +02:00
if err != nil {
gui.Log.Error(err)
}
}
2020-08-15 08:36:39 +02:00
return gui.switchFocus(v, previousView)
2018-05-26 05:23:39 +02:00
}
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
}
2020-08-15 08:36:39 +02:00
return gui.switchFocus(nil, view)
}
}
func (gui *Gui) closePopupPanels() error {
gui.onNewPopupPanel()
2020-08-15 09:23:16 +02:00
err := gui.closeConfirmationPrompt(true)
if err != nil {
gui.Log.Error(err)
return err
}
return nil
}
2018-06-09 11:06:33 +02:00
// pass in oldView = nil if you don't want to be able to return to your old view
2019-02-16 12:01:17 +02:00
// TODO: move some of this logic into our onFocusLost and onFocus hooks
2020-08-15 08:36:39 +02:00
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()
}
2019-02-16 12:01:17 +02:00
gui.Log.Info("setting highlight to true for view" + newView.Name())
2018-08-15 11:49:43 +02:00
message := gui.Tr.TemplateLocalize(
"newFocusedViewIs",
Teml{
2018-08-15 11:49:43 +02:00
"newFocusedView": newView.Name(),
},
)
gui.Log.Info(message)
2020-08-15 08:36:39 +02:00
if _, err := gui.g.SetCurrentView(newView.Name()); err != nil {
return err
}
2020-08-15 08:36:39 +02:00
if _, err := gui.g.SetViewOnTop(newView.Name()); err != nil {
return err
}
2020-08-15 08:36:39 +02:00
gui.g.Cursor = newView.Editable
2018-08-28 20:13:01 +02:00
2018-12-07 09:52:31 +02:00
if err := gui.renderPanelOptions(); err != nil {
return err
}
2020-08-15 09:23:16 +02:00
return gui.newLineFocused(newView)
2018-05-26 05:23:39 +02:00
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) resetOrigin(v *gocui.View) error {
2019-04-25 21:37:19 +02:00
_ = v.SetCursor(0, 0)
return v.SetOrigin(0, 0)
2018-06-09 11:06:33 +02:00
}
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
2020-08-15 08:36:39 +02:00
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)
2020-03-09 02:34:10 +02:00
return nil
})
2018-05-26 05:23:39 +02:00
}
2018-08-14 11:05:26 +02:00
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, ", ")
2018-06-09 11:06:33 +02:00
}
2018-12-07 09:52:31 +02:00
func (gui *Gui) renderOptionsMap(optionsMap map[string]string) error {
2020-08-15 08:36:39 +02:00
gui.renderString("options", gui.optionsMapToString(optionsMap))
2020-03-09 02:34:10 +02:00
return nil
2018-06-09 11:06:33 +02:00
}
2018-08-11 07:09:37 +02:00
// TODO: refactor properly
2018-09-03 18:45:52 +02:00
// i'm so sorry but had to add this getBranchesView
2018-12-08 07:54:54 +02:00
func (gui *Gui) getFilesView() *gocui.View {
v, _ := gui.g.View("files")
return v
}
func (gui *Gui) getCommitsView() *gocui.View {
v, _ := gui.g.View("commits")
2018-08-11 07:09:37 +02:00
return v
}
2018-12-08 07:54:54 +02:00
func (gui *Gui) getCommitMessageView() *gocui.View {
v, _ := gui.g.View("commitMessage")
2018-08-11 07:09:37 +02:00
return v
}
2018-12-08 07:54:54 +02:00
func (gui *Gui) getBranchesView() *gocui.View {
v, _ := gui.g.View("branches")
2018-08-11 07:09:37 +02:00
return v
}
2018-08-14 11:05:26 +02:00
2018-12-08 07:54:54 +02:00
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
}
2018-12-08 07:54:54 +02:00
func (gui *Gui) getStashView() *gocui.View {
v, _ := gui.g.View("stash")
return v
}
2019-03-11 00:28:47 +02:00
func (gui *Gui) getCommitFilesView() *gocui.View {
v, _ := gui.g.View("commitFiles")
return v
}
2019-11-16 05:00:27 +02:00
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
}
2020-03-26 14:20:12 +02:00
func (gui *Gui) getStatusView() *gocui.View {
v, _ := gui.g.View("status")
return v
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) trimmedContent(v *gocui.View) string {
return strings.TrimSpace(v.Buffer())
}
2019-02-25 13:11:35 +02:00
func (gui *Gui) currentViewName() string {
currentView := gui.g.CurrentView()
2020-05-16 04:35:19 +02:00
if currentView == nil {
return ""
}
2018-08-14 11:05:26 +02:00
return currentView.Name()
}
2018-09-05 11:07:46 +02:00
2020-08-15 09:23:16 +02:00
func (gui *Gui) resizeCurrentPopupPanel() error {
v := gui.g.CurrentView()
if gui.isPopupPanel(v.Name()) {
2020-08-15 09:23:16 +02:00
return gui.resizePopupPanel(v)
2018-09-05 11:07:46 +02:00
}
return nil
}
2020-08-15 09:23:16 +02:00
func (gui *Gui) resizePopupPanel(v *gocui.View) error {
2018-09-05 11:07:46 +02:00
// If the confirmation panel is already displayed, just resize the width,
// otherwise continue
content := v.Buffer()
2020-08-15 09:23:16 +02:00
x0, y0, x1, y1 := gui.getConfirmationPanelDimensions(v.Wrap, content)
2018-09-05 11:07:46 +02:00
vx0, vy0, vx1, vy1 := v.Dimensions()
if vx0 == x0 && vy0 == y0 && vx1 == x1 && vy1 == y1 {
return nil
}
gui.Log.Info(gui.Tr.SLocalize("resizingPopupPanel"))
2020-08-15 09:23:16 +02:00
_, err := gui.g.SetView(v.Name(), x0, y0, x1, y1, 0)
2018-09-05 11:07:46 +02:00
return err
}
2019-11-16 05:00:27 +02:00
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 {
2019-11-16 05:00:27 +02:00
*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
}
}
2018-12-07 09:52:31 +02:00
2020-02-25 11:11:07 +02:00
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
})
}
2018-12-07 09:52:31 +02:00
func (gui *Gui) renderPanelOptions() error {
currentView := gui.g.CurrentView()
switch currentView.Name() {
case "menu":
return gui.renderMenuOptions()
case "main":
2019-11-16 03:41:04 +02:00
if gui.State.MainContext == "merging" {
return gui.renderMergeOptions()
}
2018-12-07 09:52:31 +02:00
}
return gui.renderGlobalOptions()
2018-12-07 09:52:31 +02:00
}
2019-02-25 13:11:35 +02:00
2020-03-29 01:31:34 +02:00
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"),
2020-03-29 01:31:34 +02:00
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"
}
2019-02-25 13:11:35 +02:00
func (gui *Gui) popupPanelFocused() bool {
return gui.isPopupPanel(gui.currentViewName())
2019-02-25 13:11:35 +02:00
}
2019-11-10 07:20:35 +02:00
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()
}
}
2020-05-18 14:00:07 +02:00
// 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
}