1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-07-15 01:34:26 +02:00

making a start on unidirectional data binding to fix these UI bugs

This commit is contained in:
Jesse Duffield
2018-12-04 19:50:11 +11:00
parent ccc771d8b1
commit 99a8b1ae8b
12 changed files with 333 additions and 160 deletions

View File

@ -52,6 +52,8 @@ func newLogger(config config.AppConfigurer) *logrus.Entry {
} else { } else {
log = newProductionLogger(config) log = newProductionLogger(config)
} }
log.Formatter = &logrus.JSONFormatter{}
if config.GetUserConfig().GetString("reporting") == "on" { if config.GetUserConfig().GetString("reporting") == "on" {
// this isn't really a secret token: it only has permission to push new rollbar items // this isn't really a secret token: it only has permission to push new rollbar items
hook := rollrus.NewHook("23432119147a4367abf7c0de2aa99a2d", environment) hook := rollrus.NewHook("23432119147a4367abf7c0de2aa99a2d", environment)

View File

@ -130,6 +130,17 @@ func (c *GitCommand) GetStashEntryDiff(index int) (string, error) {
// GetStatusFiles git status files // GetStatusFiles git status files
func (c *GitCommand) GetStatusFiles() []*File { func (c *GitCommand) GetStatusFiles() []*File {
// files := []*File{}
// for i := 0; i < 100; i++ {
// files = append(files, &File{
// Name: strconv.Itoa(i),
// DisplayString: strconv.Itoa(i),
// Type: "file",
// })
// }
// return files
statusOutput, _ := c.GitStatus() statusOutput, _ := c.GitStatus()
statusStrings := utils.SplitLines(statusOutput) statusStrings := utils.SplitLines(statusOutput)
files := []*File{} files := []*File{}

View File

@ -131,30 +131,30 @@ func (gui *Gui) handleMerge(g *gocui.Gui, v *gocui.View) error {
} }
func (gui *Gui) getSelectedBranch(v *gocui.View) *commands.Branch { func (gui *Gui) getSelectedBranch(v *gocui.View) *commands.Branch {
lineNumber := gui.getItemPosition(v) selectedLine := gui.State.Panels.Branches.SelectedLine
return gui.State.Branches[lineNumber] if selectedLine == -1 {
return nil
} }
func (gui *Gui) renderBranchesOptions(g *gocui.Gui) error { return gui.State.Branches[selectedLine]
return gui.renderGlobalOptions(g)
} }
// may want to standardise how these select methods work // may want to standardise how these select methods work
func (gui *Gui) handleBranchSelect(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleBranchSelect(g *gocui.Gui, v *gocui.View) error {
if err := gui.renderBranchesOptions(g); err != nil {
return err
}
// This really shouldn't happen: there should always be a master branch // This really shouldn't happen: there should always be a master branch
if len(gui.State.Branches) == 0 { if len(gui.State.Branches) == 0 {
return gui.renderString(g, "main", gui.Tr.SLocalize("NoBranchesThisRepo")) return gui.renderString(g, "main", gui.Tr.SLocalize("NoBranchesThisRepo"))
} }
go func() {
branch := gui.getSelectedBranch(v) branch := gui.getSelectedBranch(v)
diff, err := gui.GitCommand.GetBranchGraph(branch.Name) if err := gui.focusPoint(0, gui.State.Panels.Branches.SelectedLine, v); err != nil {
if err != nil && strings.HasPrefix(diff, "fatal: ambiguous argument") { return err
diff = gui.Tr.SLocalize("NoTrackingThisBranch")
} }
gui.renderString(g, "main", diff) go func() {
graph, err := gui.GitCommand.GetBranchGraph(branch.Name)
if err != nil && strings.HasPrefix(graph, "fatal: ambiguous argument") {
graph = gui.Tr.SLocalize("NoTrackingThisBranch")
}
_ = gui.renderString(g, "main", graph)
}() }()
return nil return nil
} }
@ -173,6 +173,8 @@ func (gui *Gui) refreshBranches(g *gocui.Gui) error {
} }
gui.State.Branches = builder.Build() gui.State.Branches = builder.Build()
gui.refreshSelectedLine(&gui.State.Panels.Branches.SelectedLine, len(gui.State.Branches))
v.Clear() v.Clear()
list, err := utils.RenderList(gui.State.Branches) list, err := utils.RenderList(gui.State.Branches)
if err != nil { if err != nil {
@ -186,3 +188,17 @@ func (gui *Gui) refreshBranches(g *gocui.Gui) error {
}) })
return nil return nil
} }
func (gui *Gui) handleBranchesNextLine(g *gocui.Gui, v *gocui.View) error {
panelState := gui.State.Panels.Branches
gui.changeSelectedLine(&panelState.SelectedLine, len(gui.State.Branches), false)
return gui.handleBranchSelect(gui.g, v)
}
func (gui *Gui) handleBranchesPrevLine(g *gocui.Gui, v *gocui.View) error {
panelState := gui.State.Panels.Branches
gui.changeSelectedLine(&panelState.SelectedLine, len(gui.State.Branches), true)
return gui.handleBranchSelect(gui.g, v)
}

View File

@ -22,6 +22,8 @@ func (gui *Gui) refreshCommits(g *gocui.Gui) error {
return err return err
} }
gui.refreshSelectedLine(&gui.State.Panels.Commits.SelectedLine, len(gui.State.Commits))
v.Clear() v.Clear()
list, err := utils.RenderList(gui.State.Commits) list, err := utils.RenderList(gui.State.Commits)
@ -41,9 +43,9 @@ func (gui *Gui) refreshCommits(g *gocui.Gui) error {
func (gui *Gui) handleResetToCommit(g *gocui.Gui, commitView *gocui.View) error { func (gui *Gui) handleResetToCommit(g *gocui.Gui, commitView *gocui.View) error {
return gui.createConfirmationPanel(g, commitView, gui.Tr.SLocalize("ResetToCommit"), gui.Tr.SLocalize("SureResetThisCommit"), func(g *gocui.Gui, v *gocui.View) error { return gui.createConfirmationPanel(g, commitView, gui.Tr.SLocalize("ResetToCommit"), gui.Tr.SLocalize("SureResetThisCommit"), func(g *gocui.Gui, v *gocui.View) error {
commit, err := gui.getSelectedCommit(g) commit := gui.getSelectedCommit(g)
if err != nil { if commit == nil {
panic(err) panic(errors.New(gui.Tr.SLocalize("NoCommitsThisBranch")))
} }
if err := gui.GitCommand.ResetToCommit(commit.Sha); err != nil { if err := gui.GitCommand.ResetToCommit(commit.Sha); err != nil {
return gui.createErrorPanel(g, err.Error()) return gui.createErrorPanel(g, err.Error())
@ -59,21 +61,15 @@ func (gui *Gui) handleResetToCommit(g *gocui.Gui, commitView *gocui.View) error
}, nil) }, nil)
} }
func (gui *Gui) renderCommitsOptions(g *gocui.Gui) error { func (gui *Gui) handleCommitSelect(g *gocui.Gui, v *gocui.View) error {
return gui.renderGlobalOptions(g) commit := gui.getSelectedCommit(g)
if commit == nil {
return gui.renderString(g, "main", gui.Tr.SLocalize("NoCommitsThisBranch"))
} }
func (gui *Gui) handleCommitSelect(g *gocui.Gui, v *gocui.View) error { if err := gui.focusPoint(0, gui.State.Panels.Commits.SelectedLine, v); err != nil {
if err := gui.renderCommitsOptions(g); err != nil {
return err return err
} }
commit, err := gui.getSelectedCommit(g)
if err != nil {
if err.Error() != gui.Tr.SLocalize("NoCommitsThisBranch") {
return err
}
return gui.renderString(g, "main", gui.Tr.SLocalize("NoCommitsThisBranch"))
}
commitText, err := gui.GitCommand.Show(commit.Sha) commitText, err := gui.GitCommand.Show(commit.Sha)
if err != nil { if err != nil {
return err return err
@ -85,12 +81,12 @@ func (gui *Gui) handleCommitSquashDown(g *gocui.Gui, v *gocui.View) error {
if gui.getItemPosition(v) != 0 { if gui.getItemPosition(v) != 0 {
return gui.createErrorPanel(g, gui.Tr.SLocalize("OnlySquashTopmostCommit")) return gui.createErrorPanel(g, gui.Tr.SLocalize("OnlySquashTopmostCommit"))
} }
if len(gui.State.Commits) == 1 { if len(gui.State.Commits) <= 1 {
return gui.createErrorPanel(g, gui.Tr.SLocalize("YouNoCommitsToSquash")) return gui.createErrorPanel(g, gui.Tr.SLocalize("YouNoCommitsToSquash"))
} }
commit, err := gui.getSelectedCommit(g) commit := gui.getSelectedCommit(g)
if err != nil { if commit == nil {
return err return errors.New(gui.Tr.SLocalize("NoCommitsThisBranch"))
} }
if err := gui.GitCommand.SquashPreviousTwoCommits(commit.Name); err != nil { if err := gui.GitCommand.SquashPreviousTwoCommits(commit.Name); err != nil {
return gui.createErrorPanel(g, err.Error()) return gui.createErrorPanel(g, err.Error())
@ -113,16 +109,16 @@ func (gui *Gui) anyUnStagedChanges(files []*commands.File) bool {
} }
func (gui *Gui) handleCommitFixup(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleCommitFixup(g *gocui.Gui, v *gocui.View) error {
if len(gui.State.Commits) == 1 { if len(gui.State.Commits) <= 1 {
return gui.createErrorPanel(g, gui.Tr.SLocalize("YouNoCommitsToSquash")) return gui.createErrorPanel(g, gui.Tr.SLocalize("YouNoCommitsToSquash"))
} }
if gui.anyUnStagedChanges(gui.State.Files) { if gui.anyUnStagedChanges(gui.State.Files) {
return gui.createErrorPanel(g, gui.Tr.SLocalize("CantFixupWhileUnstagedChanges")) return gui.createErrorPanel(g, gui.Tr.SLocalize("CantFixupWhileUnstagedChanges"))
} }
branch := gui.State.Branches[0] branch := gui.State.Branches[0]
commit, err := gui.getSelectedCommit(g) commit := gui.getSelectedCommit(g)
if err != nil { if commit == nil {
return err return gui.createErrorPanel(g, gui.Tr.SLocalize("NoCommitsThisBranch"))
} }
message := gui.Tr.SLocalize("SureFixupThisCommit") message := gui.Tr.SLocalize("SureFixupThisCommit")
gui.createConfirmationPanel(g, v, gui.Tr.SLocalize("Fixup"), message, func(g *gocui.Gui, v *gocui.View) error { gui.createConfirmationPanel(g, v, gui.Tr.SLocalize("Fixup"), message, func(g *gocui.Gui, v *gocui.View) error {
@ -165,18 +161,27 @@ func (gui *Gui) handleRenameCommitEditor(g *gocui.Gui, v *gocui.View) error {
return nil return nil
} }
func (gui *Gui) getSelectedCommit(g *gocui.Gui) (*commands.Commit, error) { func (gui *Gui) getSelectedCommit(g *gocui.Gui) *commands.Commit {
v, err := g.View("commits") selectedLine := gui.State.Panels.Commits.SelectedLine
if err != nil { if selectedLine == -1 {
panic(err) return nil
} }
if len(gui.State.Commits) == 0 {
return &commands.Commit{}, errors.New(gui.Tr.SLocalize("NoCommitsThisBranch")) return gui.State.Commits[selectedLine]
} }
lineNumber := gui.getItemPosition(v)
if lineNumber > len(gui.State.Commits)-1 { func (gui *Gui) handleCommitsNextLine(g *gocui.Gui, v *gocui.View) error {
gui.Log.Info(gui.Tr.SLocalize("PotentialErrInGetselectedCommit"), gui.State.Commits, lineNumber) gui.Log.Info(utils.AsJson(gui.State.Panels))
return gui.State.Commits[len(gui.State.Commits)-1], nil panelState := gui.State.Panels.Commits
gui.changeSelectedLine(&panelState.SelectedLine, len(gui.State.Commits), false)
return gui.handleCommitSelect(gui.g, v)
} }
return gui.State.Commits[lineNumber], nil
func (gui *Gui) handleCommitsPrevLine(g *gocui.Gui, v *gocui.View) error {
gui.Log.Info(utils.AsJson(gui.State.Panels))
panelState := gui.State.Panels.Commits
gui.changeSelectedLine(&panelState.SelectedLine, len(gui.State.Commits), true)
return gui.handleCommitSelect(gui.g, v)
} }

View File

@ -140,15 +140,12 @@ func (gui *Gui) handleAddPatch(g *gocui.Gui, v *gocui.View) error {
} }
func (gui *Gui) getSelectedFile(g *gocui.Gui) (*commands.File, error) { func (gui *Gui) getSelectedFile(g *gocui.Gui) (*commands.File, error) {
if len(gui.State.Files) == 0 { selectedLine := gui.State.Panels.Files.SelectedLine
if selectedLine == -1 {
return &commands.File{}, gui.Errors.ErrNoFiles return &commands.File{}, gui.Errors.ErrNoFiles
} }
filesView, err := g.View("files")
if err != nil { return gui.State.Files[selectedLine], nil
panic(err)
}
lineNumber := gui.getItemPosition(filesView)
return gui.State.Files[lineNumber], nil
} }
func (gui *Gui) handleFileRemove(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleFileRemove(g *gocui.Gui, v *gocui.View) error {
@ -194,26 +191,23 @@ func (gui *Gui) handleIgnoreFile(g *gocui.Gui, v *gocui.View) error {
return gui.refreshFiles(g) return gui.refreshFiles(g)
} }
func (gui *Gui) renderfilesOptions(g *gocui.Gui, file *commands.File) error {
return gui.renderGlobalOptions(g)
}
func (gui *Gui) handleFileSelect(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleFileSelect(g *gocui.Gui, v *gocui.View) error {
file, err := gui.getSelectedFile(g) file, err := gui.getSelectedFile(g)
if err != nil { if err != nil {
if err != gui.Errors.ErrNoFiles { if err != gui.Errors.ErrNoFiles {
return err return err
} }
gui.renderString(g, "main", gui.Tr.SLocalize("NoChangedFiles")) return gui.renderString(g, "main", gui.Tr.SLocalize("NoChangedFiles"))
return gui.renderfilesOptions(g, nil)
}
if err := gui.renderfilesOptions(g, file); err != nil {
return err
} }
if file.HasMergeConflicts { if file.HasMergeConflicts {
return gui.refreshMergePanel(g) return gui.refreshMergePanel(g)
} }
if err := gui.focusPoint(0, gui.State.Panels.Files.SelectedLine, v); err != nil {
return err
}
content := gui.GitCommand.Diff(file, false) content := gui.GitCommand.Diff(file, false)
return gui.renderString(g, "main", content) return gui.renderString(g, "main", content)
} }
@ -309,6 +303,7 @@ func (gui *Gui) refreshStateFiles() {
// get files to stage // get files to stage
files := gui.GitCommand.GetStatusFiles() files := gui.GitCommand.GetStatusFiles()
gui.State.Files = gui.GitCommand.MergeStatusFiles(gui.State.Files, files) gui.State.Files = gui.GitCommand.MergeStatusFiles(gui.State.Files, files)
gui.refreshSelectedLine(&gui.State.Panels.Files.SelectedLine, len(gui.State.Files))
gui.updateHasMergeConflictStatus() gui.updateHasMergeConflictStatus()
} }
@ -340,6 +335,20 @@ func (gui *Gui) catSelectedFile(g *gocui.Gui) (string, error) {
return cat, nil return cat, nil
} }
func (gui *Gui) handleFilesNextLine(g *gocui.Gui, v *gocui.View) error {
panelState := gui.State.Panels.Files
gui.changeSelectedLine(&panelState.SelectedLine, len(gui.State.Files), false)
return gui.handleFileSelect(gui.g, v)
}
func (gui *Gui) handleFilesPrevLine(g *gocui.Gui, v *gocui.View) error {
panelState := gui.State.Panels.Files
gui.changeSelectedLine(&panelState.SelectedLine, len(gui.State.Files), true)
return gui.handleFileSelect(gui.g, v)
}
func (gui *Gui) refreshFiles(g *gocui.Gui) error { func (gui *Gui) refreshFiles(g *gocui.Gui) error {
filesView, err := g.View("files") filesView, err := g.View("files")
if err != nil { if err != nil {
@ -347,6 +356,8 @@ func (gui *Gui) refreshFiles(g *gocui.Gui) error {
} }
gui.refreshStateFiles() gui.refreshStateFiles()
gui.g.Update(func(g *gocui.Gui) error {
filesView.Clear() filesView.Clear()
list, err := utils.RenderList(gui.State.Files) list, err := utils.RenderList(gui.State.Files)
if err != nil { if err != nil {
@ -354,10 +365,12 @@ func (gui *Gui) refreshFiles(g *gocui.Gui) error {
} }
fmt.Fprint(filesView, list) fmt.Fprint(filesView, list)
gui.correctCursor(filesView)
if filesView == g.CurrentView() { if filesView == g.CurrentView() {
gui.handleFileSelect(g, filesView) gui.handleFileSelect(g, filesView)
} }
return nil
})
return nil return nil
} }

View File

@ -72,13 +72,37 @@ type Gui struct {
statusManager *statusManager statusManager *statusManager
} }
type stagingState struct { type stagingPanelState struct {
SelectedLine int
StageableLines []int StageableLines []int
HunkStarts []int HunkStarts []int
CurrentLineIndex int
Diff string Diff string
} }
type filePanelState struct {
SelectedLine int
}
type branchPanelState struct {
SelectedLine int
}
type commitPanelState struct {
SelectedLine int
}
type stashPanelState struct {
SelectedLine int
}
type panelStates struct {
Files *filePanelState
Staging *stagingPanelState
Branches *branchPanelState
Commits *commitPanelState
Stash *stashPanelState
}
type guiState struct { type guiState struct {
Files []*commands.File Files []*commands.File
Branches []*commands.Branch Branches []*commands.Branch
@ -92,7 +116,7 @@ type guiState struct {
EditHistory *stack.Stack EditHistory *stack.Stack
Platform commands.Platform Platform commands.Platform
Updating bool Updating bool
StagingState *stagingState Panels *panelStates
} }
// NewGui builds a new gui handler // NewGui builds a new gui handler
@ -108,6 +132,12 @@ func NewGui(log *logrus.Entry, gitCommand *commands.GitCommand, oSCommand *comma
Conflicts: make([]commands.Conflict, 0), Conflicts: make([]commands.Conflict, 0),
EditHistory: stack.New(), EditHistory: stack.New(),
Platform: *oSCommand.Platform, Platform: *oSCommand.Platform,
Panels: &panelStates{
Files: &filePanelState{SelectedLine: -1},
Branches: &branchPanelState{SelectedLine: 0},
Commits: &commitPanelState{SelectedLine: -1},
Stash: &stashPanelState{SelectedLine: -1},
},
} }
gui := &Gui{ gui := &Gui{
@ -193,9 +223,11 @@ func (gui *Gui) layout(g *gocui.Gui) error {
} }
v.Title = gui.Tr.SLocalize("NotEnoughSpace") v.Title = gui.Tr.SLocalize("NotEnoughSpace")
v.Wrap = true v.Wrap = true
g.SetCurrentView(v.Name()) g.SetViewOnTop("limit")
} }
return nil return nil
} else {
g.SetViewOnBottom("limit")
} }
g.DeleteView("limit") g.DeleteView("limit")
@ -247,12 +279,13 @@ func (gui *Gui) layout(g *gocui.Gui) error {
v.FgColor = gocui.ColorWhite v.FgColor = gocui.ColorWhite
} }
if v, err := g.SetView("branches", 0, filesBranchesBoundary+panelSpacing, leftSideWidth, commitsBranchesBoundary, gocui.TOP|gocui.BOTTOM); err != nil { branchesView, err := g.SetView("branches", 0, filesBranchesBoundary+panelSpacing, leftSideWidth, commitsBranchesBoundary, gocui.TOP|gocui.BOTTOM)
if err != nil {
if err != gocui.ErrUnknownView { if err != gocui.ErrUnknownView {
return err return err
} }
v.Title = gui.Tr.SLocalize("BranchesTitle") branchesView.Title = gui.Tr.SLocalize("BranchesTitle")
v.FgColor = gocui.ColorWhite branchesView.FgColor = gocui.ColorWhite
} }
if v, err := g.SetView("commits", 0, commitsBranchesBoundary+panelSpacing, leftSideWidth, commitsStashBoundary, gocui.TOP|gocui.BOTTOM); err != nil { if v, err := g.SetView("commits", 0, commitsBranchesBoundary+panelSpacing, leftSideWidth, commitsStashBoundary, gocui.TOP|gocui.BOTTOM); err != nil {
@ -325,11 +358,14 @@ func (gui *Gui) layout(g *gocui.Gui) error {
return err return err
} }
gui.handleFileSelect(g, filesView) gui.g.SetCurrentView(filesView.Name())
gui.refreshFiles(g) gui.refreshFiles(g)
gui.refreshBranches(g) gui.refreshBranches(g)
gui.refreshCommits(g) gui.refreshCommits(g)
gui.refreshStashEntries(g) gui.refreshStashEntries(g)
if err := gui.renderGlobalOptions(g); err != nil {
return err
}
if err := gui.switchFocus(g, nil, filesView); err != nil { if err := gui.switchFocus(g, nil, filesView); err != nil {
return err return err
} }
@ -341,6 +377,17 @@ func (gui *Gui) layout(g *gocui.Gui) error {
} }
} }
listViews := map[*gocui.View]int{
filesView: gui.State.Panels.Files.SelectedLine,
branchesView: gui.State.Panels.Branches.SelectedLine,
}
for view, selectedLine := range listViews {
// check if the selected line is now out of view and if so refocus it
if err := gui.focusPoint(0, selectedLine, view); err != nil {
return err
}
}
return gui.resizeCurrentPopupPanel(g) return gui.resizeCurrentPopupPanel(g)
} }
@ -411,7 +458,7 @@ func (gui *Gui) Run() error {
} }
gui.goEvery(g, time.Second*60, gui.fetch) gui.goEvery(g, time.Second*60, gui.fetch)
gui.goEvery(g, time.Second*10, gui.refreshFiles) // gui.goEvery(g, time.Second*2, gui.refreshFiles) // TODO: comment back in
gui.goEvery(g, time.Millisecond*50, gui.updateLoader) gui.goEvery(g, time.Millisecond*50, gui.updateLoader)
gui.goEvery(g, time.Millisecond*50, gui.renderAppStatus) gui.goEvery(g, time.Millisecond*50, gui.renderAppStatus)

View File

@ -219,7 +219,12 @@ func (gui *Gui) GetKeybindings() []*Binding {
Handler: gui.handleSwitchToStagingPanel, Handler: gui.handleSwitchToStagingPanel,
Description: gui.Tr.SLocalize("StageLines"), Description: gui.Tr.SLocalize("StageLines"),
KeyReadable: "enter", KeyReadable: "enter",
}, { },
{ViewName: "files", Key: 'k', Modifier: gocui.ModNone, Handler: gui.handleFilesPrevLine},
{ViewName: "files", Key: gocui.KeyArrowUp, Modifier: gocui.ModNone, Handler: gui.handleFilesPrevLine},
{ViewName: "files", Key: 'j', Modifier: gocui.ModNone, Handler: gui.handleFilesNextLine},
{ViewName: "files", Key: gocui.KeyArrowDown, Modifier: gocui.ModNone, Handler: gui.handleFilesNextLine},
{
ViewName: "main", ViewName: "main",
Key: gocui.KeyEsc, Key: gocui.KeyEsc,
Modifier: gocui.ModNone, Modifier: gocui.ModNone,
@ -322,7 +327,12 @@ func (gui *Gui) GetKeybindings() []*Binding {
Modifier: gocui.ModNone, Modifier: gocui.ModNone,
Handler: gui.handleMerge, Handler: gui.handleMerge,
Description: gui.Tr.SLocalize("mergeIntoCurrentBranch"), Description: gui.Tr.SLocalize("mergeIntoCurrentBranch"),
}, { },
{ViewName: "branches", Key: 'k', Modifier: gocui.ModNone, Handler: gui.handleBranchesPrevLine},
{ViewName: "branches", Key: gocui.KeyArrowUp, Modifier: gocui.ModNone, Handler: gui.handleBranchesPrevLine},
{ViewName: "branches", Key: 'j', Modifier: gocui.ModNone, Handler: gui.handleBranchesNextLine},
{ViewName: "branches", Key: gocui.KeyArrowDown, Modifier: gocui.ModNone, Handler: gui.handleBranchesNextLine},
{
ViewName: "commits", ViewName: "commits",
Key: 's', Key: 's',
Modifier: gocui.ModNone, Modifier: gocui.ModNone,
@ -352,7 +362,12 @@ func (gui *Gui) GetKeybindings() []*Binding {
Modifier: gocui.ModNone, Modifier: gocui.ModNone,
Handler: gui.handleCommitFixup, Handler: gui.handleCommitFixup,
Description: gui.Tr.SLocalize("fixupCommit"), Description: gui.Tr.SLocalize("fixupCommit"),
}, { },
{ViewName: "commits", Key: 'k', Modifier: gocui.ModNone, Handler: gui.handleCommitsPrevLine},
{ViewName: "commits", Key: gocui.KeyArrowUp, Modifier: gocui.ModNone, Handler: gui.handleCommitsPrevLine},
{ViewName: "commits", Key: 'j', Modifier: gocui.ModNone, Handler: gui.handleCommitsNextLine},
{ViewName: "commits", Key: gocui.KeyArrowDown, Modifier: gocui.ModNone, Handler: gui.handleCommitsNextLine},
{
ViewName: "stash", ViewName: "stash",
Key: gocui.KeySpace, Key: gocui.KeySpace,
Modifier: gocui.ModNone, Modifier: gocui.ModNone,
@ -455,17 +470,22 @@ func (gui *Gui) GetKeybindings() []*Binding {
// Would make these keybindings global but that interferes with editing // Would make these keybindings global but that interferes with editing
// input in the confirmation panel // input in the confirmation panel
for _, viewName := range []string{"status", "files", "branches", "commits", "stash", "menu"} { for _, viewName := range []string{"status", "commits", "stash", "menu"} {
bindings = append(bindings, []*Binding{
{ViewName: viewName, Key: gocui.KeyArrowUp, Modifier: gocui.ModNone, Handler: gui.cursorUp},
{ViewName: viewName, Key: gocui.KeyArrowDown, Modifier: gocui.ModNone, Handler: gui.cursorDown},
{ViewName: viewName, Key: 'k', Modifier: gocui.ModNone, Handler: gui.cursorUp},
{ViewName: viewName, Key: 'j', Modifier: gocui.ModNone, Handler: gui.cursorDown},
}...)
}
for _, viewName := range []string{"status", "branches", "files", "commits", "stash", "menu"} {
bindings = append(bindings, []*Binding{ bindings = append(bindings, []*Binding{
{ViewName: viewName, Key: gocui.KeyTab, Modifier: gocui.ModNone, Handler: gui.nextView}, {ViewName: viewName, Key: gocui.KeyTab, Modifier: gocui.ModNone, Handler: gui.nextView},
{ViewName: viewName, Key: gocui.KeyArrowLeft, Modifier: gocui.ModNone, Handler: gui.previousView}, {ViewName: viewName, Key: gocui.KeyArrowLeft, Modifier: gocui.ModNone, Handler: gui.previousView},
{ViewName: viewName, Key: gocui.KeyArrowRight, Modifier: gocui.ModNone, Handler: gui.nextView}, {ViewName: viewName, Key: gocui.KeyArrowRight, Modifier: gocui.ModNone, Handler: gui.nextView},
{ViewName: viewName, Key: gocui.KeyArrowUp, Modifier: gocui.ModNone, Handler: gui.cursorUp},
{ViewName: viewName, Key: gocui.KeyArrowDown, Modifier: gocui.ModNone, Handler: gui.cursorDown},
{ViewName: viewName, Key: 'h', Modifier: gocui.ModNone, Handler: gui.previousView}, {ViewName: viewName, Key: 'h', Modifier: gocui.ModNone, Handler: gui.previousView},
{ViewName: viewName, Key: 'l', Modifier: gocui.ModNone, Handler: gui.nextView}, {ViewName: viewName, Key: 'l', Modifier: gocui.ModNone, Handler: gui.nextView},
{ViewName: viewName, Key: 'k', Modifier: gocui.ModNone, Handler: gui.cursorUp},
{ViewName: viewName, Key: 'j', Modifier: gocui.ModNone, Handler: gui.cursorDown},
}...) }...)
} }

View File

@ -40,22 +40,22 @@ func (gui *Gui) refreshStagingPanel() error {
return nil return nil
} }
var currentLineIndex int var selectedLine int
if gui.State.StagingState != nil { if gui.State.Panels.Staging != nil {
end := len(stageableLines) - 1 end := len(stageableLines) - 1
if end < gui.State.StagingState.CurrentLineIndex { if end < gui.State.Panels.Staging.SelectedLine {
currentLineIndex = end selectedLine = end
} else { } else {
currentLineIndex = gui.State.StagingState.CurrentLineIndex selectedLine = gui.State.Panels.Staging.SelectedLine
} }
} else { } else {
currentLineIndex = 0 selectedLine = 0
} }
gui.State.StagingState = &stagingState{ gui.State.Panels.Staging = &stagingPanelState{
StageableLines: stageableLines, StageableLines: stageableLines,
HunkStarts: hunkStarts, HunkStarts: hunkStarts,
CurrentLineIndex: currentLineIndex, SelectedLine: selectedLine,
Diff: diff, Diff: diff,
} }
@ -74,7 +74,7 @@ func (gui *Gui) handleStagingEscape(g *gocui.Gui, v *gocui.View) error {
return err return err
} }
gui.State.StagingState = nil gui.State.Panels.Staging = nil
return gui.switchFocus(gui.g, nil, gui.getFilesView(gui.g)) return gui.switchFocus(gui.g, nil, gui.getFilesView(gui.g))
} }
@ -96,9 +96,9 @@ func (gui *Gui) handleStagingNextHunk(g *gocui.Gui, v *gocui.View) error {
} }
func (gui *Gui) handleCycleHunk(prev bool) error { func (gui *Gui) handleCycleHunk(prev bool) error {
state := gui.State.StagingState state := gui.State.Panels.Staging
lineNumbers := state.StageableLines lineNumbers := state.StageableLines
currentLine := lineNumbers[state.CurrentLineIndex] currentLine := lineNumbers[state.SelectedLine]
currentHunkIndex := utils.PrevIndex(state.HunkStarts, currentLine) currentHunkIndex := utils.PrevIndex(state.HunkStarts, currentLine)
var newHunkIndex int var newHunkIndex int
if prev { if prev {
@ -115,22 +115,22 @@ func (gui *Gui) handleCycleHunk(prev bool) error {
} }
} }
state.CurrentLineIndex = utils.NextIndex(lineNumbers, state.HunkStarts[newHunkIndex]) state.SelectedLine = utils.NextIndex(lineNumbers, state.HunkStarts[newHunkIndex])
return gui.focusLineAndHunk() return gui.focusLineAndHunk()
} }
func (gui *Gui) handleCycleLine(prev bool) error { func (gui *Gui) handleCycleLine(prev bool) error {
state := gui.State.StagingState state := gui.State.Panels.Staging
lineNumbers := state.StageableLines lineNumbers := state.StageableLines
currentLine := lineNumbers[state.CurrentLineIndex] currentLine := lineNumbers[state.SelectedLine]
var newIndex int var newIndex int
if prev { if prev {
newIndex = utils.PrevIndex(lineNumbers, currentLine) newIndex = utils.PrevIndex(lineNumbers, currentLine)
} else { } else {
newIndex = utils.NextIndex(lineNumbers, currentLine) newIndex = utils.NextIndex(lineNumbers, currentLine)
} }
state.CurrentLineIndex = newIndex state.SelectedLine = newIndex
return gui.focusLineAndHunk() return gui.focusLineAndHunk()
} }
@ -139,9 +139,9 @@ func (gui *Gui) handleCycleLine(prev bool) error {
// selected line and size of the hunk // selected line and size of the hunk
func (gui *Gui) focusLineAndHunk() error { func (gui *Gui) focusLineAndHunk() error {
stagingView := gui.getStagingView(gui.g) stagingView := gui.getStagingView(gui.g)
state := gui.State.StagingState state := gui.State.Panels.Staging
lineNumber := state.StageableLines[state.CurrentLineIndex] lineNumber := state.StageableLines[state.SelectedLine]
// we want the bottom line of the view buffer to ideally be the bottom line // we want the bottom line of the view buffer to ideally be the bottom line
// of the hunk, but if the hunk is too big we'll just go three lines beyond // of the hunk, but if the hunk is too big we'll just go three lines beyond
@ -170,23 +170,7 @@ func (gui *Gui) focusLineAndHunk() error {
bottomLine = lineNumber + 3 bottomLine = lineNumber + 3
} }
return gui.focusLine(lineNumber, bottomLine, stagingView) return gui.generalFocusLine(lineNumber, bottomLine, stagingView)
}
// focusLine takes a lineNumber to focus, and a bottomLine to ensure we can see
func (gui *Gui) focusLine(lineNumber int, bottomLine int, v *gocui.View) error {
_, height := v.Size()
overScroll := bottomLine - height + 1
if overScroll < 0 {
overScroll = 0
}
if err := v.SetOrigin(0, overScroll); err != nil {
return err
}
if err := v.SetCursor(0, lineNumber-overScroll); err != nil {
return err
}
return nil
} }
func (gui *Gui) handleStageHunk(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleStageHunk(g *gocui.Gui, v *gocui.View) error {
@ -198,13 +182,13 @@ func (gui *Gui) handleStageLine(g *gocui.Gui, v *gocui.View) error {
} }
func (gui *Gui) handleStageLineOrHunk(hunk bool) error { func (gui *Gui) handleStageLineOrHunk(hunk bool) error {
state := gui.State.StagingState state := gui.State.Panels.Staging
p, err := git.NewPatchModifier(gui.Log) p, err := git.NewPatchModifier(gui.Log)
if err != nil { if err != nil {
return err return err
} }
currentLine := state.StageableLines[state.CurrentLineIndex] currentLine := state.StageableLines[state.SelectedLine]
var patch string var patch string
if hunk { if hunk {
patch, err = p.ModifyPatchForHunk(state.Diff, state.HunkStarts, currentLine) patch, err = p.ModifyPatchForHunk(state.Diff, state.HunkStarts, currentLine)

View File

@ -37,14 +37,7 @@ func (gui *Gui) getSelectedStashEntry(v *gocui.View) *commands.StashEntry {
return gui.State.StashEntries[lineNumber] return gui.State.StashEntries[lineNumber]
} }
func (gui *Gui) renderStashOptions(g *gocui.Gui) error {
return gui.renderGlobalOptions(g)
}
func (gui *Gui) handleStashEntrySelect(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleStashEntrySelect(g *gocui.Gui, v *gocui.View) error {
if err := gui.renderStashOptions(g); err != nil {
return err
}
go func() { go func() {
stashEntry := gui.getSelectedStashEntry(v) stashEntry := gui.getSelectedStashEntry(v)
if stashEntry == nil { if stashEntry == nil {

View File

@ -42,10 +42,6 @@ func (gui *Gui) refreshStatus(g *gocui.Gui) error {
return nil return nil
} }
func (gui *Gui) renderStatusOptions(g *gocui.Gui) error {
return gui.renderGlobalOptions(g)
}
func (gui *Gui) handleCheckForUpdate(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleCheckForUpdate(g *gocui.Gui, v *gocui.View) error {
gui.Updater.CheckForNewUpdate(gui.onUserUpdateCheckFinish, true) gui.Updater.CheckForNewUpdate(gui.onUserUpdateCheckFinish, true)
return gui.createMessagePanel(gui.g, v, "", gui.Tr.SLocalize("CheckingForUpdates")) return gui.createMessagePanel(gui.g, v, "", gui.Tr.SLocalize("CheckingForUpdates"))
@ -63,10 +59,7 @@ func (gui *Gui) handleStatusSelect(g *gocui.Gui, v *gocui.View) error {
"Buy Jesse a coffee: https://donorbox.org/lazygit", "Buy Jesse a coffee: https://donorbox.org/lazygit",
}, "\n\n") }, "\n\n")
if err := gui.renderString(g, "main", dashboardString); err != nil { return gui.renderString(g, "main", dashboardString)
return err
}
return gui.renderStatusOptions(g)
} }
func (gui *Gui) handleOpenConfig(g *gocui.Gui, v *gocui.View) error { func (gui *Gui) handleOpenConfig(g *gocui.Gui, v *gocui.View) error {

View File

@ -230,20 +230,47 @@ func (gui *Gui) resetOrigin(v *gocui.View) error {
// if the cursor down past the last item, move it to the last line // if the cursor down past the last item, move it to the last line
func (gui *Gui) correctCursor(v *gocui.View) error { func (gui *Gui) correctCursor(v *gocui.View) error {
cx, cy := v.Cursor() cx, cy := v.Cursor()
ox, oy := v.Origin() return gui.focusPoint(cx, cy, v)
_, height := v.Size() }
maxY := height - 1
ly := v.LinesHeight() - 1 // if the cursor down past the last item, move it to the last line
if oy+cy <= ly { func (gui *Gui) focusPoint(cx int, cy int, v *gocui.View) error {
if cy < 0 {
return nil return nil
} }
newCy := utils.Min(ly, maxY) ox, oy := v.Origin()
if err := v.SetCursor(cx, newCy); err != nil { _, height := v.Size()
ly := height - 1
// if line is above origin, move origin and set cursor to zero
// if line is below origin + height, move origin and set cursor to max
// otherwise set cursor to value - origin
if ly > v.LinesHeight() {
if err := v.SetCursor(cx, cy); err != nil {
return err return err
} }
if err := v.SetOrigin(ox, ly-newCy); err != nil { if err := v.SetOrigin(ox, 0); err != nil {
return err return err
} }
} else if cy < oy {
if err := v.SetCursor(cx, 0); err != nil {
return err
}
if err := v.SetOrigin(ox, cy); err != nil {
return err
}
} else if cy > oy+ly {
if err := v.SetCursor(cx, ly); err != nil {
return err
}
if err := v.SetOrigin(ox, cy-ly); err != nil {
return err
}
} else {
if err := v.SetCursor(cx, cy-oy); err != nil {
return err
}
}
return nil return nil
} }
@ -334,3 +361,59 @@ func (gui *Gui) resizePopupPanel(g *gocui.Gui, v *gocui.View) error {
_, err := g.SetView(v.Name(), x0, y0, x1, y1, 0) _, err := g.SetView(v.Name(), x0, y0, x1, y1, 0)
return err return err
} }
// focusLine focuses and selects the given line
func (gui *Gui) focusLine(lineNumber int, v *gocui.View) error {
_, height := v.Size()
overScroll := lineNumber - height + 1
if overScroll < 0 {
overScroll = 0
}
if err := v.SetOrigin(0, overScroll); err != nil {
return err
}
if err := v.SetCursor(0, lineNumber-overScroll); err != nil {
return err
}
return nil
}
// generalFocusLine takes a lineNumber to focus, and a bottomLine to ensure we can see
func (gui *Gui) generalFocusLine(lineNumber int, bottomLine int, v *gocui.View) error {
_, height := v.Size()
overScroll := bottomLine - height + 1
if overScroll < 0 {
overScroll = 0
}
if err := v.SetOrigin(0, overScroll); err != nil {
return err
}
if err := v.SetCursor(0, lineNumber-overScroll); err != nil {
return err
}
return nil
}
func (gui *Gui) changeSelectedLine(line *int, total int, up bool) {
if up {
if *line == -1 || *line == 0 {
return
}
*line -= 1
} else {
if *line == -1 || *line == total-1 {
return
}
*line += 1
}
}
func (gui *Gui) refreshSelectedLine(line *int, total int) {
if *line == -1 && total > 0 {
*line = 0
} else if total-1 < *line {
*line = total - 1
}
}

View File

@ -1,6 +1,7 @@
package utils package utils
import ( import (
"encoding/json"
"errors" "errors"
"fmt" "fmt"
"log" "log"
@ -235,3 +236,8 @@ func PrevIndex(numbers []int, currentNumber int) int {
} }
return end return end
} }
func AsJson(i interface{}) string {
bytes, _ := json.MarshalIndent(i, "", " ")
return string(bytes)
}