1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2024-11-30 09:16:47 +02:00
lazygit/pkg/gui/gui.go

368 lines
10 KiB
Go
Raw Normal View History

package gui
2018-05-19 03:16:34 +02:00
import (
2018-05-26 05:23:39 +02:00
// "io"
// "io/ioutil"
2018-05-26 05:23:39 +02:00
2018-08-12 13:04:47 +02:00
"errors"
2018-08-14 00:33:40 +02:00
"io/ioutil"
2018-08-12 13:04:47 +02:00
"log"
2018-08-14 00:33:40 +02:00
"os"
2018-08-12 13:04:47 +02:00
"os/exec"
"strings"
"time"
// "strings"
2018-08-06 08:11:29 +02:00
2018-08-23 14:22:03 +02:00
"github.com/sirupsen/logrus"
"github.com/golang-collections/collections/stack"
"github.com/jesseduffield/gocui"
2018-08-13 12:26:02 +02:00
"github.com/jesseduffield/lazygit/pkg/commands"
2018-08-18 05:22:05 +02:00
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/i18n"
2018-05-19 03:16:34 +02:00
)
// OverlappingEdges determines if panel edges overlap
var OverlappingEdges = false
// SentinelErrors are the errors that have special meaning and need to be checked
// by calling functions. The less of these, the better
type SentinelErrors struct {
ErrSubProcess error
ErrNoFiles error
}
// GenerateSentinelErrors makes the sentinel errors for the gui. We're defining it here
// because we can't do package-scoped errors with localization, and also because
// it seems like package-scoped variables are bad in general
// https://dave.cheney.net/2017/06/11/go-without-package-scoped-variables
// In the future it would be good to implement some of the recommendations of
// that article. For now, if we don't need an error to be a sentinel, we will just
// define it inline. This has implications for error messages that pop up everywhere
// in that we'll be duplicating the default values. We may need to look at
// having a default localisation bundle defined, and just using keys-only when
// localising things in the code.
func (gui *Gui) GenerateSentinelErrors() {
gui.Errors = SentinelErrors{
ErrSubProcess: errors.New(gui.Tr.SLocalize("RunningSubprocess")),
ErrNoFiles: errors.New(gui.Tr.SLocalize("NoChangedFiles")),
}
}
2018-08-14 11:05:26 +02:00
// Teml is short for template used to make the required map[string]interface{} shorter when using gui.Tr.SLocalize and gui.Tr.TemplateLocalize
2018-08-16 13:35:04 +02:00
type Teml i18n.Teml
2018-08-12 13:04:47 +02:00
2018-08-13 12:26:02 +02:00
// Gui wraps the gocui Gui object which handles rendering and events
type Gui struct {
2018-08-14 00:33:40 +02:00
g *gocui.Gui
2018-08-13 12:26:02 +02:00
Log *logrus.Logger
GitCommand *commands.GitCommand
OSCommand *commands.OSCommand
SubProcess *exec.Cmd
2018-08-13 13:16:21 +02:00
State guiState
2018-08-18 05:22:05 +02:00
Config config.AppConfigurer
Tr *i18n.Localizer
Errors SentinelErrors
2018-08-13 13:16:21 +02:00
}
type guiState struct {
Files []commands.File
Branches []commands.Branch
Commits []commands.Commit
StashEntries []commands.StashEntry
PreviousView string
HasMergeConflicts bool
ConflictIndex int
ConflictTop bool
Conflicts []commands.Conflict
EditHistory *stack.Stack
2018-08-14 04:24:32 +02:00
Platform commands.Platform
2018-08-13 12:26:02 +02:00
}
// NewGui builds a new gui handler
2018-08-18 05:22:05 +02:00
func NewGui(log *logrus.Logger, gitCommand *commands.GitCommand, oSCommand *commands.OSCommand, tr *i18n.Localizer, config config.AppConfigurer) (*Gui, error) {
2018-08-13 13:16:21 +02:00
initialState := guiState{
2018-08-13 12:26:02 +02:00
Files: make([]commands.File, 0),
PreviousView: "files",
Commits: make([]commands.Commit, 0),
StashEntries: make([]commands.StashEntry, 0),
ConflictIndex: 0,
ConflictTop: true,
Conflicts: make([]commands.Conflict, 0),
EditHistory: stack.New(),
2018-08-14 04:24:32 +02:00
Platform: *oSCommand.Platform,
2018-08-13 12:26:02 +02:00
}
gui := &Gui{
2018-08-13 12:26:02 +02:00
Log: log,
GitCommand: gitCommand,
OSCommand: oSCommand,
State: initialState,
2018-08-18 05:22:05 +02:00
Config: config,
Tr: tr,
}
gui.GenerateSentinelErrors()
return gui, nil
2018-08-13 12:26:02 +02:00
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) scrollUpMain(g *gocui.Gui, v *gocui.View) error {
mainView, _ := g.View("main")
ox, oy := mainView.Origin()
if oy >= 1 {
2018-08-18 05:22:05 +02:00
return mainView.SetOrigin(ox, oy-gui.Config.GetUserConfig().GetInt("gui.scrollHeight"))
}
return nil
2018-05-21 12:52:48 +02:00
}
2018-05-19 03:16:34 +02:00
2018-08-13 13:16:21 +02:00
func (gui *Gui) scrollDownMain(g *gocui.Gui, v *gocui.View) error {
mainView, _ := g.View("main")
ox, oy := mainView.Origin()
if oy < len(mainView.BufferLines()) {
2018-08-18 05:22:05 +02:00
return mainView.SetOrigin(ox, oy+gui.Config.GetUserConfig().GetInt("gui.scrollHeight"))
}
return nil
2018-05-19 09:04:33 +02:00
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) handleRefresh(g *gocui.Gui, v *gocui.View) error {
return gui.refreshSidePanels(g)
2018-06-09 11:06:33 +02:00
}
func max(a, b int) int {
if a > b {
return a
}
return b
}
// layout is called for every screen re-render e.g. when the screen is resized
2018-08-13 12:26:02 +02:00
func (gui *Gui) layout(g *gocui.Gui) error {
g.Highlight = true
width, height := g.Size()
2018-08-20 12:52:32 +02:00
version := gui.Config.GetVersion()
leftSideWidth := width / 3
statusFilesBoundary := 2
filesBranchesBoundary := 2 * height / 5 // height - 20
commitsBranchesBoundary := 3 * height / 5 // height - 10
commitsStashBoundary := height - 5 // height - 5
2018-08-20 12:52:32 +02:00
optionsVersionBoundary := width - max(len(version), 1)
2018-08-05 14:00:02 +02:00
minimumHeight := 16
minimumWidth := 10
2018-08-05 14:00:02 +02:00
panelSpacing := 1
if OverlappingEdges {
panelSpacing = 0
}
if height < minimumHeight || width < minimumWidth {
v, err := g.SetView("limit", 0, 0, max(width-1, 2), max(height-1, 2), 0)
2018-08-05 14:00:02 +02:00
if err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Title = gui.Tr.SLocalize("NotEnoughSpace")
2018-08-05 14:00:02 +02:00
v.Wrap = true
}
return nil
}
2018-05-21 12:52:48 +02:00
2018-08-06 07:41:59 +02:00
g.DeleteView("limit")
optionsTop := height - 2
// hiding options if there's not enough space
if height < 30 {
optionsTop = height - 1
}
2018-05-21 12:52:48 +02:00
v, err := g.SetView("main", leftSideWidth+panelSpacing, 0, width-1, optionsTop, gocui.LEFT)
if err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Title = gui.Tr.SLocalize("DiffTitle")
2018-08-05 14:00:02 +02:00
v.Wrap = true
2018-08-06 08:11:29 +02:00
v.FgColor = gocui.ColorWhite
}
2018-05-19 03:16:34 +02:00
2018-08-05 14:00:02 +02:00
if v, err := g.SetView("status", 0, 0, leftSideWidth, statusFilesBoundary, gocui.BOTTOM|gocui.RIGHT); err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Title = gui.Tr.SLocalize("StatusTitle")
2018-08-06 08:11:29 +02:00
v.FgColor = gocui.ColorWhite
}
2018-06-01 15:23:31 +02:00
filesView, err := g.SetView("files", 0, statusFilesBoundary+panelSpacing, leftSideWidth, filesBranchesBoundary, gocui.TOP|gocui.BOTTOM)
if err != nil {
if err != gocui.ErrUnknownView {
return err
}
2018-08-05 14:00:02 +02:00
filesView.Highlight = true
filesView.Title = gui.Tr.SLocalize("FilesTitle")
2018-08-06 08:11:29 +02:00
v.FgColor = gocui.ColorWhite
}
2018-05-26 05:23:39 +02:00
if v, err := g.SetView("branches", 0, filesBranchesBoundary+panelSpacing, leftSideWidth, commitsBranchesBoundary, gocui.TOP|gocui.BOTTOM); err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Title = gui.Tr.SLocalize("BranchesTitle")
2018-08-06 08:11:29 +02:00
v.FgColor = gocui.ColorWhite
}
2018-05-21 12:52:48 +02:00
if v, err := g.SetView("commits", 0, commitsBranchesBoundary+panelSpacing, leftSideWidth, commitsStashBoundary, gocui.TOP|gocui.BOTTOM); err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Title = gui.Tr.SLocalize("CommitsTitle")
2018-08-06 08:11:29 +02:00
v.FgColor = gocui.ColorWhite
}
2018-06-05 10:48:46 +02:00
if v, err := g.SetView("stash", 0, commitsStashBoundary+panelSpacing, leftSideWidth, optionsTop, gocui.TOP|gocui.RIGHT); err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Title = gui.Tr.SLocalize("StashTitle")
2018-08-06 08:11:29 +02:00
v.FgColor = gocui.ColorWhite
}
2018-05-21 12:52:48 +02:00
2018-08-20 12:52:32 +02:00
if v, err := g.SetView("options", -1, optionsTop, optionsVersionBoundary-1, optionsTop+2, 0); err != nil {
if err != gocui.ErrUnknownView {
return err
}
v.Frame = false
2018-08-18 05:54:39 +02:00
if v.FgColor, err = gui.GetOptionsPanelTextColor(); err != nil {
return err
}
2018-08-08 00:32:52 +02:00
}
2018-08-13 12:26:02 +02:00
if gui.getCommitMessageView(g) == nil {
2018-08-11 07:09:37 +02:00
// doesn't matter where this view starts because it will be hidden
if commitMessageView, err := g.SetView("commitMessage", 0, 0, width, height, 0); err != nil {
if err != gocui.ErrUnknownView {
return err
}
g.SetViewOnBottom("commitMessage")
commitMessageView.Title = gui.Tr.SLocalize("CommitMessage")
2018-08-11 07:09:37 +02:00
commitMessageView.FgColor = gocui.ColorWhite
commitMessageView.Editable = true
}
}
2018-08-20 12:52:32 +02:00
if v, err := g.SetView("version", optionsVersionBoundary-1, optionsTop, width, optionsTop+2, 0); err != nil {
2018-08-08 00:32:52 +02:00
if err != gocui.ErrUnknownView {
return err
}
v.BgColor = gocui.ColorDefault
v.FgColor = gocui.ColorGreen
v.Frame = false
if err := gui.renderString(g, "version", version); err != nil {
return err
}
2018-06-05 10:48:46 +02:00
// these are only called once
2018-08-13 12:26:02 +02:00
gui.handleFileSelect(g, filesView)
gui.refreshFiles(g)
2018-08-13 13:16:21 +02:00
gui.refreshBranches(g)
gui.refreshCommits(g)
gui.refreshStashEntries(g)
if err := gui.switchFocus(g, nil, filesView); err != nil {
return err
}
}
2018-05-19 03:16:34 +02:00
2018-08-13 13:16:21 +02:00
gui.resizePopupPanels(g)
return nil
2018-05-19 03:16:34 +02:00
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) fetch(g *gocui.Gui) error {
gui.GitCommand.Fetch()
gui.refreshStatus(g)
return nil
2018-06-02 05:51:03 +02:00
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) updateLoader(g *gocui.Gui) error {
if confirmationView, _ := g.View("confirmation"); confirmationView != nil {
2018-08-13 12:26:02 +02:00
content := gui.trimmedContent(confirmationView)
if strings.Contains(content, "...") {
staticContent := strings.Split(content, "...")[0] + "..."
2018-08-13 13:16:21 +02:00
gui.renderString(g, "confirmation", staticContent+" "+gui.loader())
}
}
return nil
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) goEvery(g *gocui.Gui, interval time.Duration, function func(*gocui.Gui) error) {
go func() {
for range time.Tick(interval) {
function(g)
}
}()
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) resizePopupPanels(g *gocui.Gui) error {
v := g.CurrentView()
if v.Name() == "commitMessage" || v.Name() == "confirmation" {
2018-08-13 13:16:21 +02:00
return gui.resizePopupPanel(g, v)
}
return nil
}
2018-08-13 12:26:02 +02:00
// Run setup the gui with keybindings and start the mainloop
2018-08-13 13:16:21 +02:00
func (gui *Gui) Run() error {
g, err := gocui.NewGui(gocui.OutputNormal, OverlappingEdges)
if err != nil {
2018-08-13 13:16:21 +02:00
return err
}
defer g.Close()
2018-05-19 09:04:33 +02:00
2018-08-14 00:33:40 +02:00
gui.g = g // TODO: always use gui.g rather than passing g around everywhere
2018-08-18 05:53:58 +02:00
if err := gui.SetColorScheme(); err != nil {
return err
}
2018-08-11 07:09:37 +02:00
2018-08-13 13:16:21 +02:00
gui.goEvery(g, time.Second*60, gui.fetch)
gui.goEvery(g, time.Second*10, gui.refreshFiles)
gui.goEvery(g, time.Millisecond*10, gui.updateLoader)
2018-08-13 12:26:02 +02:00
g.SetManagerFunc(gui.layout)
2018-05-19 09:04:33 +02:00
2018-08-13 12:26:02 +02:00
if err = gui.keybindings(g); err != nil {
2018-08-13 13:16:21 +02:00
return err
}
2018-05-19 09:04:33 +02:00
2018-08-07 10:05:43 +02:00
err = g.MainLoop()
2018-08-13 13:16:21 +02:00
return err
2018-08-13 12:26:02 +02:00
}
// RunWithSubprocesses loops, instantiating a new gocui.Gui with each iteration
// if the error returned from a run is a ErrSubProcess, it runs the subprocess
// otherwise it handles the error, possibly by quitting the application
func (gui *Gui) RunWithSubprocesses() {
for {
if err := gui.Run(); err != nil {
if err == gocui.ErrQuit {
break
} else if err == gui.Errors.ErrSubProcess {
2018-08-14 00:33:40 +02:00
gui.SubProcess.Stdin = os.Stdin
gui.SubProcess.Stdout = os.Stdout
gui.SubProcess.Stderr = os.Stderr
2018-08-13 12:26:02 +02:00
gui.SubProcess.Run()
2018-08-14 00:33:40 +02:00
gui.SubProcess.Stdout = ioutil.Discard
gui.SubProcess.Stderr = ioutil.Discard
gui.SubProcess.Stdin = nil
2018-08-13 13:35:54 +02:00
gui.SubProcess = nil
2018-08-13 12:26:02 +02:00
} else {
log.Panicln(err)
}
}
}
2018-05-19 09:04:33 +02:00
}
2018-08-13 13:16:21 +02:00
func (gui *Gui) quit(g *gocui.Gui, v *gocui.View) error {
return gocui.ErrQuit
2018-05-26 05:23:39 +02:00
}