1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-26 05:37:18 +02:00
lazygit/pkg/gui/gui.go

930 lines
26 KiB
Go
Raw Normal View History

package gui
2018-05-19 11:16:34 +10:00
import (
"fmt"
"io/ioutil"
2019-03-03 23:08:07 +11:00
"math"
"os"
2018-12-07 19:22:22 +01:00
"sync"
2018-05-26 13:23:39 +10:00
// "io"
// "io/ioutil"
2018-05-26 13:23:39 +10:00
2018-08-12 21:04:47 +10:00
"os/exec"
"strings"
"time"
"github.com/fsnotify/fsnotify"
"github.com/go-errors/errors"
// "strings"
2018-08-06 16:11:29 +10:00
2018-11-25 13:15:36 +01:00
"github.com/fatih/color"
"github.com/golang-collections/collections/stack"
"github.com/jesseduffield/gocui"
2018-08-13 20:26:02 +10:00
"github.com/jesseduffield/lazygit/pkg/commands"
2018-08-18 13:22:05 +10:00
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/i18n"
"github.com/jesseduffield/lazygit/pkg/theme"
2018-08-19 23:28:29 +10:00
"github.com/jesseduffield/lazygit/pkg/updates"
2019-02-25 22:11:35 +11:00
"github.com/jesseduffield/lazygit/pkg/utils"
"github.com/sirupsen/logrus"
2018-05-19 11:16:34 +10:00
)
2019-11-10 22:07:45 +11:00
const StartupPopupVersion = 1
// 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
2018-09-07 09:41:15 +10:00
ErrSwitchRepo 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-09-07 09:41:15 +10:00
ErrSwitchRepo: errors.New("switching repo"),
}
}
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 21:04:47 +10:00
2018-08-13 20:26:02 +10:00
// Gui wraps the gocui Gui object which handles rendering and events
type Gui struct {
2019-04-07 16:45:55 +10:00
g *gocui.Gui
Log *logrus.Entry
GitCommand *commands.GitCommand
OSCommand *commands.OSCommand
SubProcess *exec.Cmd
State guiState
Config config.AppConfigurer
Tr *i18n.Localizer
Errors SentinelErrors
Updater *updates.Updater
statusManager *statusManager
credentials credentials
waitForIntro sync.WaitGroup
fileWatcher *fsnotify.Watcher
2018-08-13 21:16:21 +10:00
}
2018-12-07 18:52:31 +11:00
// for now the staging panel state, unlike the other panel states, is going to be
// non-mutative, so that we don't accidentally end up
// with mismatches of data. We might change this in the future
type lineByLinePanelState struct {
SelectedLineIdx int
FirstLineIdx int
LastLineIdx int
Diff string
PatchParser *commands.PatchParser
SelectMode int // one of LINE, HUNK, or RANGE
SecondaryFocused bool // this is for if we show the left or right panel
}
2018-12-08 16:54:54 +11:00
type mergingPanelState struct {
ConflictIndex int
ConflictTop bool
Conflicts []commands.Conflict
EditHistory *stack.Stack
}
type filePanelState struct {
SelectedLine int
}
2019-11-13 23:18:31 +11:00
// TODO: consider splitting this out into the window and the branches view
type branchPanelState struct {
SelectedLine int
2019-11-13 23:18:31 +11:00
}
type remotePanelState struct {
SelectedLine int
}
2019-11-16 17:35:59 +11:00
type remoteBranchesState struct {
SelectedLine int
}
2019-11-18 09:38:36 +11:00
type tagsPanelState struct {
SelectedLine int
}
type commitPanelState struct {
SelectedLine int
SpecificDiffMode bool
}
type stashPanelState struct {
SelectedLine int
}
type menuPanelState struct {
SelectedLine int
2019-11-10 16:20:35 +11:00
OnPress func(g *gocui.Gui, v *gocui.View) error
}
type commitFilesPanelState struct {
SelectedLine int
}
2019-11-10 16:20:35 +11:00
type statusPanelState struct {
pushables string
pullables string
}
type panelStates struct {
2019-11-16 17:35:59 +11:00
Files *filePanelState
Branches *branchPanelState
Remotes *remotePanelState
RemoteBranches *remoteBranchesState
2019-11-18 09:38:36 +11:00
Tags *tagsPanelState
2019-11-16 17:35:59 +11:00
Commits *commitPanelState
Stash *stashPanelState
Menu *menuPanelState
LineByLine *lineByLinePanelState
Merging *mergingPanelState
CommitFiles *commitFilesPanelState
Status *statusPanelState
}
2018-08-13 21:16:21 +10:00
type guiState struct {
Files []*commands.File
Branches []*commands.Branch
Commits []*commands.Commit
StashEntries []*commands.StashEntry
CommitFiles []*commands.CommitFile
DiffEntries []*commands.Commit
2019-11-13 23:18:31 +11:00
Remotes []*commands.Remote
2019-11-17 10:23:06 +11:00
RemoteBranches []*commands.RemoteBranch
2019-11-18 09:38:36 +11:00
Tags []*commands.Tag
2019-11-17 10:23:06 +11:00
MenuItemCount int // can't store the actual list because it's of interface{} type
PreviousView string
Platform commands.Platform
Updating bool
Panels *panelStates
WorkingTreeState string // one of "merging", "rebasing", "normal"
2019-11-16 12:41:04 +11:00
MainContext string // used to keep the main and secondary views' contexts in sync
CherryPickedCommits []*commands.Commit
SplitMainPanel bool
RetainOriginalDir bool
IsRefreshingFiles bool
RefreshingFilesMutex sync.Mutex
2018-08-13 20:26:02 +10:00
}
// for now the split view will always be on
2018-08-13 20:26:02 +10:00
// NewGui builds a new gui handler
func NewGui(log *logrus.Entry, gitCommand *commands.GitCommand, oSCommand *commands.OSCommand, tr *i18n.Localizer, config config.AppConfigurer, updater *updates.Updater) (*Gui, error) {
2018-08-13 21:16:21 +10:00
initialState := guiState{
Files: make([]*commands.File, 0),
PreviousView: "files",
Commits: make([]*commands.Commit, 0),
CherryPickedCommits: make([]*commands.Commit, 0),
StashEntries: make([]*commands.StashEntry, 0),
2019-03-28 18:58:34 +09:00
DiffEntries: make([]*commands.Commit, 0),
Platform: *oSCommand.Platform,
Panels: &panelStates{
2019-11-16 17:35:59 +11:00
Files: &filePanelState{SelectedLine: -1},
Branches: &branchPanelState{SelectedLine: 0},
Remotes: &remotePanelState{SelectedLine: 0},
RemoteBranches: &remoteBranchesState{SelectedLine: -1},
2019-11-18 09:38:36 +11:00
Tags: &tagsPanelState{SelectedLine: -1},
2019-11-16 17:35:59 +11:00
Commits: &commitPanelState{SelectedLine: -1},
CommitFiles: &commitFilesPanelState{SelectedLine: -1},
Stash: &stashPanelState{SelectedLine: -1},
Menu: &menuPanelState{SelectedLine: 0},
2018-12-08 16:54:54 +11:00
Merging: &mergingPanelState{
ConflictIndex: 0,
ConflictTop: true,
Conflicts: []commands.Conflict{},
EditHistory: stack.New(),
},
2019-11-10 16:20:35 +11:00
Status: &statusPanelState{},
},
2018-08-13 20:26:02 +10:00
}
gui := &Gui{
2018-08-25 15:55:49 +10:00
Log: log,
GitCommand: gitCommand,
OSCommand: oSCommand,
State: initialState,
Config: config,
Tr: tr,
Updater: updater,
statusManager: &statusManager{},
}
gui.watchFilesForChanges()
gui.GenerateSentinelErrors()
return gui, nil
2018-08-13 20:26:02 +10:00
}
2019-11-10 16:50:36 +11:00
func (gui *Gui) scrollUpView(viewName string) error {
mainView, _ := gui.g.View(viewName)
ox, oy := mainView.Origin()
2019-03-03 23:08:07 +11:00
newOy := int(math.Max(0, float64(oy-gui.Config.GetUserConfig().GetInt("gui.scrollHeight"))))
return mainView.SetOrigin(ox, newOy)
2018-05-21 20:52:48 +10:00
}
2018-05-19 11:16:34 +10:00
2019-11-10 16:50:36 +11:00
func (gui *Gui) scrollDownView(viewName string) error {
mainView, _ := gui.g.View(viewName)
ox, oy := mainView.Origin()
y := oy
if !gui.Config.GetUserConfig().GetBool("gui.scrollPastBottom") {
_, sy := mainView.Size()
y += sy
}
if y < len(mainView.BufferLines()) {
2018-08-18 13:22:05 +10:00
return mainView.SetOrigin(ox, oy+gui.Config.GetUserConfig().GetInt("gui.scrollHeight"))
}
return nil
2018-05-19 17:04:33 +10:00
}
2019-11-10 16:50:36 +11:00
func (gui *Gui) scrollUpMain(g *gocui.Gui, v *gocui.View) error {
return gui.scrollUpView("main")
}
func (gui *Gui) scrollDownMain(g *gocui.Gui, v *gocui.View) error {
return gui.scrollDownView("main")
}
func (gui *Gui) scrollUpSecondary(g *gocui.Gui, v *gocui.View) error {
return gui.scrollUpView("secondary")
}
func (gui *Gui) scrollDownSecondary(g *gocui.Gui, v *gocui.View) error {
return gui.scrollDownView("secondary")
}
2018-08-13 21:16:21 +10:00
func (gui *Gui) handleRefresh(g *gocui.Gui, v *gocui.View) error {
return gui.refreshSidePanels(g)
2018-06-09 19:06:33 +10:00
}
func max(a, b int) int {
if a > b {
return a
}
return b
}
// getFocusLayout returns a manager function for when view gain and lose focus
func (gui *Gui) getFocusLayout() func(g *gocui.Gui) error {
var previousView *gocui.View
return func(g *gocui.Gui) error {
newView := gui.g.CurrentView()
if err := gui.onFocusChange(); err != nil {
return err
}
// for now we don't consider losing focus to a popup panel as actually losing focus
if newView != previousView && !gui.isPopupPanel(newView.Name()) {
if err := gui.onFocusLost(previousView, newView); err != nil {
return err
}
if err := gui.onFocus(newView); err != nil {
return err
}
previousView = newView
}
return nil
}
}
2019-02-16 21:01:17 +11:00
func (gui *Gui) onFocusChange() error {
currentView := gui.g.CurrentView()
for _, view := range gui.g.Views() {
view.Highlight = view == currentView
}
2019-11-10 16:20:35 +11:00
return nil
2019-02-16 21:01:17 +11:00
}
func (gui *Gui) onFocusLost(v *gocui.View, newView *gocui.View) error {
if v == nil {
return nil
}
switch v.Name() {
case "branches":
if v.Context == "local-branches" {
// This stops the branches panel from showing the upstream/downstream changes to the selected branch, when it loses focus
// inside renderListPanel it checks to see if the panel has focus
if err := gui.renderListPanel(gui.getBranchesView(), gui.State.Branches); err != nil {
return err
}
2019-02-16 21:30:29 +11:00
}
case "main":
2019-03-11 09:28:47 +11:00
// if we have lost focus to a first-class panel, we need to do some cleanup
2019-11-16 12:41:04 +11:00
if err := gui.changeMainViewsContext("normal"); err != nil {
2019-03-02 13:22:02 +11:00
return err
}
case "commitFiles":
2019-11-16 12:41:04 +11:00
if gui.State.MainContext != "patch-building" {
if _, err := gui.g.SetViewOnBottom(v.Name()); err != nil {
return err
}
2019-03-12 19:20:19 +11:00
}
}
gui.Log.Info(v.Name() + " focus lost")
return nil
}
func (gui *Gui) onFocus(v *gocui.View) error {
if v == nil {
return nil
}
gui.Log.Info(v.Name() + " focus gained")
return nil
}
// layout is called for every screen re-render e.g. when the screen is resized
2018-08-13 20:26:02 +10:00
func (gui *Gui) layout(g *gocui.Gui) error {
g.Highlight = true
width, height := g.Size()
2019-04-20 15:56:23 +02:00
information := gui.Config.GetVersion()
if gui.g.Mouse {
donate := color.New(color.FgMagenta, color.Underline).Sprint(gui.Tr.SLocalize("Donate"))
information = donate + " " + information
}
2019-04-20 15:56:23 +02:00
2019-04-26 08:24:14 +02:00
minimumHeight := 9
2019-04-20 15:56:23 +02:00
minimumWidth := 10
if height < minimumHeight || width < minimumWidth {
v, err := g.SetView("limit", 0, 0, width-1, height-1, 0)
2019-04-20 15:56:23 +02:00
if err != nil {
if err.Error() != "unknown view" {
return err
}
v.Title = gui.Tr.SLocalize("NotEnoughSpace")
v.Wrap = true
2019-04-20 16:51:50 +02:00
_, _ = g.SetViewOnTop("limit")
2019-04-20 15:56:23 +02:00
}
return nil
}
currView := gui.g.CurrentView()
currentCyclebleView := gui.State.PreviousView
if currView != nil {
viewName := currView.Name()
usePreviouseView := true
for _, view := range cyclableViews {
if view == viewName {
currentCyclebleView = viewName
usePreviouseView = false
break
2019-04-20 15:56:23 +02:00
}
}
if usePreviouseView {
currentCyclebleView = gui.State.PreviousView
}
}
2019-04-20 15:56:23 +02:00
usableSpace := height - 7
extraSpace := usableSpace - (usableSpace/3)*3
vHeights := map[string]int{
"status": 3,
"files": (usableSpace / 3) + extraSpace,
"branches": usableSpace / 3,
"commits": usableSpace / 3,
"stash": 3,
"options": 1,
}
if height < 28 {
defaultHeight := 3
2019-04-25 21:37:19 +02:00
if height < 21 {
defaultHeight = 1
2019-04-20 15:56:23 +02:00
}
vHeights = map[string]int{
"status": defaultHeight,
"files": defaultHeight,
"branches": defaultHeight,
"commits": defaultHeight,
"stash": defaultHeight,
"options": defaultHeight,
}
vHeights[currentCyclebleView] = height - defaultHeight*4 - 1
2019-04-20 15:56:23 +02:00
}
optionsVersionBoundary := width - max(len(utils.Decolorise(information)), 1)
2018-08-05 22:00:02 +10:00
2018-08-25 15:55:49 +10:00
appStatus := gui.statusManager.getStatusString()
appStatusOptionsBoundary := 0
if appStatus != "" {
appStatusOptionsBoundary = len(appStatus) + 2
2018-08-23 18:43:16 +10:00
}
panelSpacing := 1
if OverlappingEdges {
panelSpacing = 0
}
_, _ = g.SetViewOnBottom("limit")
2018-08-06 07:41:59 +02:00
g.DeleteView("limit")
textColor := theme.GocuiDefaultTextColor
leftSideWidth := width / 3
panelSplitX := width - 1
if gui.State.SplitMainPanel {
units := 7
leftSideWidth = width / units
panelSplitX = (1 + ((units - 1) / 2)) * width / units
}
main := "main"
secondary := "secondary"
swappingMainPanels := gui.State.Panels.LineByLine != nil && gui.State.Panels.LineByLine.SecondaryFocused
if swappingMainPanels {
main = "secondary"
secondary = "main"
}
v, err := g.SetView(main, leftSideWidth+panelSpacing, 0, panelSplitX, height-2, gocui.LEFT)
if err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
v.Title = gui.Tr.SLocalize("DiffTitle")
2018-08-05 22:00:02 +10:00
v.Wrap = true
v.FgColor = textColor
}
2018-05-19 11:16:34 +10:00
hiddenViewOffset := 0
if !gui.State.SplitMainPanel {
hiddenViewOffset = 9999
}
secondaryView, err := g.SetView(secondary, panelSplitX+1+hiddenViewOffset, hiddenViewOffset, width-1+hiddenViewOffset, height-2+hiddenViewOffset, gocui.LEFT)
if err != nil {
if err.Error() != "unknown view" {
return err
}
secondaryView.Title = gui.Tr.SLocalize("DiffTitle")
secondaryView.Wrap = true
secondaryView.FgColor = gocui.ColorWhite
}
if v, err := g.SetView("status", 0, 0, leftSideWidth, vHeights["status"]-1, gocui.BOTTOM|gocui.RIGHT); err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
v.Title = gui.Tr.SLocalize("StatusTitle")
v.FgColor = textColor
}
2018-06-01 23:23:31 +10:00
filesView, err := g.SetViewBeneath("files", "status", vHeights["files"])
if err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
2018-08-05 22:00:02 +10:00
filesView.Highlight = true
filesView.Title = gui.Tr.SLocalize("FilesTitle")
v.FgColor = textColor
}
2018-05-26 13:23:39 +10:00
branchesView, err := g.SetViewBeneath("branches", "files", vHeights["branches"])
if err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
branchesView.Title = gui.Tr.SLocalize("BranchesTitle")
2019-11-18 09:38:36 +11:00
branchesView.Tabs = []string{"Local Branches", "Remotes", "Tags"}
branchesView.FgColor = textColor
}
2018-05-21 20:52:48 +10:00
if v, err := g.SetViewBeneath("commitFiles", "branches", vHeights["commits"]); err != nil {
if err.Error() != "unknown view" {
return err
}
v.Title = gui.Tr.SLocalize("CommitFiles")
v.FgColor = textColor
}
commitsView, err := g.SetViewBeneath("commits", "branches", vHeights["commits"])
2019-02-25 22:11:35 +11:00
if err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
2019-02-25 22:11:35 +11:00
commitsView.Title = gui.Tr.SLocalize("CommitsTitle")
commitsView.FgColor = textColor
}
2018-06-05 18:48:46 +10:00
stashView, err := g.SetViewBeneath("stash", "commits", vHeights["stash"])
2019-02-25 22:11:35 +11:00
if err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
2019-02-25 22:11:35 +11:00
stashView.Title = gui.Tr.SLocalize("StashTitle")
stashView.FgColor = textColor
}
2018-05-21 20:52:48 +10:00
if v, err := g.SetView("options", appStatusOptionsBoundary-1, height-2, optionsVersionBoundary-1, height, 0); err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
return err
}
v.Frame = false
userConfig := gui.Config.GetUserConfig()
v.FgColor = theme.GetColor(userConfig.GetStringSlice("gui.theme.optionsTextColor"))
2018-08-08 08:32:52 +10:00
}
2018-12-08 16:54:54 +11:00
if gui.getCommitMessageView() == nil {
2018-08-11 15:09:37 +10:00
// doesn't matter where this view starts because it will be hidden
if commitMessageView, err := g.SetView("commitMessage", width, height, width*2, height*2, 0); err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
2018-08-11 15:09:37 +10:00
return err
}
g.SetViewOnBottom("commitMessage")
commitMessageView.Title = gui.Tr.SLocalize("CommitMessage")
commitMessageView.FgColor = textColor
2018-08-11 15:09:37 +10:00
commitMessageView.Editable = true
2019-12-07 16:10:49 +11:00
commitMessageView.Editor = gocui.EditorFunc(gui.commitMessageEditor)
2018-09-02 17:08:59 +02:00
}
2018-08-11 15:09:37 +10:00
}
if check, _ := g.View("credentials"); check == nil {
2018-10-20 17:37:55 +02:00
// doesn't matter where this view starts because it will be hidden
if credentialsView, err := g.SetView("credentials", width, height, width*2, height*2, 0); err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
2018-10-20 17:37:55 +02:00
return err
}
_, err := g.SetViewOnBottom("credentials")
2018-10-20 18:58:37 +02:00
if err != nil {
return err
}
2018-12-10 08:04:22 +01:00
credentialsView.Title = gui.Tr.SLocalize("CredentialsUsername")
credentialsView.FgColor = textColor
credentialsView.Editable = true
2018-10-20 17:37:55 +02:00
}
}
if appStatusView, err := g.SetView("appStatus", -1, height-2, width, height, 0); err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
2018-08-23 18:43:16 +10:00
return err
}
2018-08-25 15:55:49 +10:00
appStatusView.BgColor = gocui.ColorDefault
appStatusView.FgColor = gocui.ColorCyan
appStatusView.Frame = false
2018-08-25 17:38:03 +10:00
if _, err := g.SetViewOnBottom("appStatus"); err != nil {
return err
}
2018-08-23 18:43:16 +10:00
}
if v, err := g.SetView("information", optionsVersionBoundary-1, height-2, width, height, 0); err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
2018-08-08 08:32:52 +10:00
return err
}
v.BgColor = gocui.ColorDefault
v.FgColor = gocui.ColorGreen
v.Frame = false
if err := gui.renderString(g, "information", information); err != nil {
return err
}
2018-06-05 18:48:46 +10:00
2019-03-12 21:43:56 +11:00
// doing this here because it'll only happen once
2019-11-16 12:41:04 +11:00
if err := gui.onInitialViewsCreation(); err != nil {
2018-09-07 09:41:15 +10:00
return err
}
2019-03-12 21:43:56 +11:00
}
2018-09-07 09:41:15 +10:00
2019-03-12 21:43:56 +11:00
if gui.g.CurrentView() == nil {
if _, err := gui.g.SetCurrentView(gui.getFilesView().Name()); err != nil {
2018-12-07 18:52:31 +11:00
return err
}
2019-03-12 21:43:56 +11:00
if err := gui.switchFocus(gui.g, nil, gui.getFilesView()); err != nil {
2018-12-07 18:52:31 +11:00
return err
}
2019-03-12 21:43:56 +11:00
}
2018-12-07 18:52:31 +11:00
type listViewState struct {
selectedLine int
lineCount int
2019-11-16 16:38:38 +11:00
view *gocui.View
context string
}
2019-11-16 16:38:38 +11:00
listViews := []listViewState{
{view: filesView, context: "", selectedLine: gui.State.Panels.Files.SelectedLine, lineCount: len(gui.State.Files)},
{view: branchesView, context: "local-branches", selectedLine: gui.State.Panels.Branches.SelectedLine, lineCount: len(gui.State.Branches)},
{view: branchesView, context: "remotes", selectedLine: gui.State.Panels.Remotes.SelectedLine, lineCount: len(gui.State.Remotes)},
2019-11-16 17:35:59 +11:00
{view: branchesView, context: "remote-branches", selectedLine: gui.State.Panels.RemoteBranches.SelectedLine, lineCount: len(gui.State.Remotes)},
2019-11-16 16:38:38 +11:00
{view: commitsView, context: "", selectedLine: gui.State.Panels.Commits.SelectedLine, lineCount: len(gui.State.Commits)},
{view: stashView, context: "", selectedLine: gui.State.Panels.Stash.SelectedLine, lineCount: len(gui.State.StashEntries)},
2019-02-25 22:11:35 +11:00
}
// menu view might not exist so we check to be safe
if menuView, err := gui.g.View("menu"); err == nil {
2019-11-16 16:38:38 +11:00
listViews = append(listViews, listViewState{view: menuView, context: "", selectedLine: gui.State.Panels.Menu.SelectedLine, lineCount: gui.State.MenuItemCount})
}
2019-11-16 16:38:38 +11:00
for _, listView := range listViews {
// ignore views where the context doesn't match up with the selected line we're trying to focus
if listView.context != "" && (listView.view.Context != listView.context) {
continue
}
// check if the selected line is now out of view and if so refocus it
2019-11-16 16:38:38 +11:00
if err := gui.focusPoint(0, listView.selectedLine, listView.lineCount, listView.view); err != nil {
return err
}
}
2018-12-07 18:52:31 +11:00
// here is a good place log some stuff
// if you download humanlog and do tail -f development.log | humanlog
// this will let you see these branches as prettified json
// gui.Log.Info(utils.AsJson(gui.State.Branches[0:4]))
2018-09-05 19:07:46 +10:00
return gui.resizeCurrentPopupPanel(g)
2018-05-19 11:16:34 +10:00
}
2019-11-16 12:41:04 +11:00
func (gui *Gui) onInitialViewsCreation() error {
if err := gui.changeMainViewsContext("normal"); err != nil {
return err
}
2019-11-16 14:00:27 +11:00
gui.getBranchesView().Context = "local-branches"
2019-11-16 12:41:04 +11:00
return gui.loadNewRepo()
}
2019-03-12 21:43:56 +11:00
func (gui *Gui) loadNewRepo() error {
gui.Updater.CheckForNewUpdate(gui.onBackgroundUpdateCheckFinish, false)
if err := gui.updateRecentRepoList(); err != nil {
return err
}
gui.waitForIntro.Done()
if err := gui.refreshSidePanels(gui.g); err != nil {
return err
}
2019-11-10 22:07:45 +11:00
return nil
}
func (gui *Gui) showInitialPopups(tasks []func(chan struct{}) error) {
gui.waitForIntro.Add(len(tasks))
done := make(chan struct{})
go func() {
for _, task := range tasks {
go func() {
if err := task(done); err != nil {
_ = gui.createErrorPanel(gui.g, err.Error())
}
}()
<-done
gui.waitForIntro.Done()
2019-03-12 21:43:56 +11:00
}
2019-11-10 22:07:45 +11:00
}()
}
func (gui *Gui) showShamelessSelfPromotionMessage(done chan struct{}) error {
onConfirm := func(g *gocui.Gui, v *gocui.View) error {
done <- struct{}{}
return gui.Config.WriteToUserConfig("startupPopupVersion", StartupPopupVersion)
2019-03-12 21:43:56 +11:00
}
2019-11-10 22:07:45 +11:00
return gui.createConfirmationPanel(gui.g, nil, true, gui.Tr.SLocalize("ShamelessSelfPromotionTitle"), gui.Tr.SLocalize("ShamelessSelfPromotionMessage"), onConfirm, onConfirm)
2019-03-12 21:43:56 +11:00
}
2019-11-10 22:07:45 +11:00
func (gui *Gui) promptAnonymousReporting(done chan struct{}) error {
return gui.createConfirmationPanel(gui.g, nil, true, gui.Tr.SLocalize("AnonymousReportingTitle"), gui.Tr.SLocalize("AnonymousReportingPrompt"), func(g *gocui.Gui, v *gocui.View) error {
2019-11-10 22:07:45 +11:00
done <- struct{}{}
return gui.Config.WriteToUserConfig("reporting", "on")
2018-08-26 15:46:18 +10:00
}, func(g *gocui.Gui, v *gocui.View) error {
2019-11-10 22:07:45 +11:00
done <- struct{}{}
return gui.Config.WriteToUserConfig("reporting", "off")
2018-08-26 15:46:18 +10:00
})
}
2018-12-18 22:29:07 +11:00
func (gui *Gui) fetch(g *gocui.Gui, v *gocui.View, canAskForCredentials bool) (unamePassOpend bool, err error) {
2018-12-07 14:56:29 +01:00
unamePassOpend = false
err = gui.GitCommand.Fetch(func(passOrUname string) string {
unamePassOpend = true
2018-12-06 09:05:51 +01:00
return gui.waitForPassUname(gui.g, v, passOrUname)
2018-12-18 22:29:07 +11:00
}, canAskForCredentials)
2018-11-25 13:15:36 +01:00
2018-12-18 22:29:07 +11:00
if canAskForCredentials && err != nil && strings.Contains(err.Error(), "exit status 128") {
2018-11-25 13:15:36 +01:00
colorFunction := color.New(color.FgRed).SprintFunc()
coloredMessage := colorFunction(strings.TrimSpace(gui.Tr.SLocalize("PassUnameWrong")))
close := func(g *gocui.Gui, v *gocui.View) error {
return nil
}
_ = gui.createConfirmationPanel(g, v, true, gui.Tr.SLocalize("Error"), coloredMessage, close, close)
2018-11-25 13:15:36 +01:00
}
2018-08-13 21:16:21 +10:00
gui.refreshStatus(g)
2018-12-07 14:56:29 +01:00
return unamePassOpend, err
2018-06-02 13:51:03 +10:00
}
2018-12-08 16:54:54 +11:00
func (gui *Gui) renderAppStatus() error {
2018-08-25 15:55:49 +10:00
appStatus := gui.statusManager.getStatusString()
if appStatus != "" {
return gui.renderString(gui.g, "appStatus", appStatus)
}
return nil
}
2018-12-07 18:52:31 +11:00
func (gui *Gui) renderGlobalOptions() error {
return gui.renderOptionsMap(map[string]string{
"PgUp/PgDn": gui.Tr.SLocalize("scroll"),
"← → ↑ ↓": gui.Tr.SLocalize("navigate"),
"esc/q": gui.Tr.SLocalize("close"),
2018-09-05 15:55:24 +02:00
"x": gui.Tr.SLocalize("menu"),
"1-5": gui.Tr.SLocalize("jump"),
})
}
2018-12-08 16:54:54 +11:00
func (gui *Gui) goEvery(interval time.Duration, function func() error) {
go func() {
for range time.Tick(interval) {
2019-03-02 13:22:02 +11:00
_ = function()
}
}()
}
func (gui *Gui) startBackgroundFetch() {
2019-07-19 13:56:53 +02:00
gui.waitForIntro.Wait()
isNew := gui.Config.GetIsNewRepo()
if !isNew {
time.After(60 * time.Second)
}
2019-07-19 14:40:23 +02:00
_, err := gui.fetch(gui.g, gui.g.CurrentView(), false)
2019-07-19 13:56:53 +02:00
if err != nil && strings.Contains(err.Error(), "exit status 128") && isNew {
_ = gui.createConfirmationPanel(gui.g, gui.g.CurrentView(), true, gui.Tr.SLocalize("NoAutomaticGitFetchTitle"), gui.Tr.SLocalize("NoAutomaticGitFetchBody"), nil, nil)
2019-07-19 13:56:53 +02:00
} else {
gui.goEvery(time.Second*60, func() error {
_, err := gui.fetch(gui.g, gui.g.CurrentView(), false)
return err
})
}
}
2018-08-13 20:26:02 +10:00
// Run setup the gui with keybindings and start the mainloop
2018-08-13 21:16:21 +10:00
func (gui *Gui) Run() error {
g, err := gocui.NewGui(gocui.OutputNormal, OverlappingEdges)
if err != nil {
2018-08-13 21:16:21 +10:00
return err
}
defer g.Close()
2018-05-19 17:04:33 +10:00
if gui.Config.GetUserConfig().GetBool("gui.mouseEvents") {
g.Mouse = true
}
2019-02-25 22:11:35 +11:00
2018-08-14 08:33:40 +10:00
gui.g = g // TODO: always use gui.g rather than passing g around everywhere
if err := gui.setColorScheme(); err != nil {
2018-08-18 13:53:58 +10:00
return err
}
2018-08-11 15:09:37 +10:00
2019-11-10 22:07:45 +11:00
popupTasks := []func(chan struct{}) error{}
2018-12-07 19:22:22 +01:00
if gui.Config.GetUserConfig().GetString("reporting") == "undetermined" {
2019-11-10 22:07:45 +11:00
popupTasks = append(popupTasks, gui.promptAnonymousReporting)
}
configPopupVersion := gui.Config.GetUserConfig().GetInt("StartupPopupVersion")
// -1 means we've disabled these popups
if configPopupVersion != -1 && configPopupVersion < StartupPopupVersion {
popupTasks = append(popupTasks, gui.showShamelessSelfPromotionMessage)
2018-12-07 19:22:22 +01:00
}
2019-11-10 22:07:45 +11:00
gui.showInitialPopups(popupTasks)
2018-12-07 19:22:22 +01:00
2019-11-10 22:07:45 +11:00
gui.waitForIntro.Add(1)
2019-08-06 14:47:24 +02:00
if gui.Config.GetUserConfig().GetBool("git.autoFetch") {
2019-07-19 13:56:53 +02:00
go gui.startBackgroundFetch()
}
2019-11-10 22:07:45 +11:00
2019-02-11 21:07:12 +11:00
gui.goEvery(time.Second*10, gui.refreshFiles)
gui.goEvery(time.Millisecond*50, gui.renderAppStatus)
g.SetManager(gocui.ManagerFunc(gui.layout), gocui.ManagerFunc(gui.getFocusLayout()))
2018-05-19 17:04:33 +10:00
2018-08-13 20:26:02 +10:00
if err = gui.keybindings(g); err != nil {
2018-08-13 21:16:21 +10:00
return err
}
2018-05-19 17:04:33 +10:00
2019-11-10 22:07:45 +11:00
gui.Log.Warn("starting main loop")
2018-08-07 18:05:43 +10:00
err = g.MainLoop()
2018-08-13 21:16:21 +10:00
return err
2018-08-13 20:26:02 +10: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() error {
2018-08-13 20:26:02 +10:00
for {
if err := gui.Run(); err != nil {
if err == gocui.ErrQuit {
if !gui.State.RetainOriginalDir {
if err := gui.recordCurrentDirectory(); err != nil {
return err
}
}
gui.fileWatcher.Close()
2018-08-13 20:26:02 +10:00
break
2018-09-07 09:41:15 +10:00
} else if err == gui.Errors.ErrSwitchRepo {
continue
} else if err == gui.Errors.ErrSubProcess {
if err := gui.runCommand(); err != nil {
2019-03-12 21:43:56 +11:00
return err
}
2018-08-13 20:26:02 +10:00
} else {
return err
2018-08-13 20:26:02 +10:00
}
}
}
return nil
2018-05-19 17:04:33 +10:00
}
func (gui *Gui) runCommand() error {
gui.SubProcess.Stdout = os.Stdout
gui.SubProcess.Stderr = os.Stdout
gui.SubProcess.Stdin = os.Stdin
2019-03-12 21:43:56 +11:00
fmt.Fprintf(os.Stdout, "\n%s\n\n", utils.ColoredString("+ "+strings.Join(gui.SubProcess.Args, " "), color.FgBlue))
if err := gui.SubProcess.Run(); err != nil {
2019-03-12 21:43:56 +11:00
// not handling the error explicitly because usually we're going to see it
// in the output anyway
gui.Log.Error(err)
}
gui.SubProcess.Stdout = ioutil.Discard
gui.SubProcess.Stderr = ioutil.Discard
gui.SubProcess.Stdin = nil
gui.SubProcess = nil
fmt.Fprintf(os.Stdout, "\n%s", utils.ColoredString(gui.Tr.SLocalize("pressEnterToReturn"), color.FgGreen))
fmt.Scanln() // wait for enter press
return nil
2019-03-12 21:43:56 +11:00
}
2019-02-25 22:11:35 +11:00
func (gui *Gui) handleDonate(g *gocui.Gui, v *gocui.View) error {
if !gui.g.Mouse {
return nil
}
2019-02-25 22:11:35 +11:00
cx, _ := v.Cursor()
if cx > len(gui.Tr.SLocalize("Donate")) {
return nil
}
2019-11-10 22:07:45 +11:00
return gui.OSCommand.OpenLink("https://github.com/sponsors/jesseduffield")
2019-02-25 22:11:35 +11:00
}
// setColorScheme sets the color scheme for the app based on the user config
func (gui *Gui) setColorScheme() error {
userConfig := gui.Config.GetUserConfig()
theme.UpdateTheme(userConfig)
gui.g.FgColor = theme.InactiveBorderColor
gui.g.SelFgColor = theme.ActiveBorderColor
return nil
}
2019-11-10 16:20:35 +11:00
func (gui *Gui) handleMouseDownMain(g *gocui.Gui, v *gocui.View) error {
if gui.popupPanelFocused() {
return nil
}
switch g.CurrentView().Name() {
case "files":
return gui.enterFile(false, v.SelectedLineIdx())
case "commitFiles":
return gui.enterCommitFile(v.SelectedLineIdx())
}
return nil
}
func (gui *Gui) handleMouseDownSecondary(g *gocui.Gui, v *gocui.View) error {
if gui.popupPanelFocused() {
return nil
}
switch g.CurrentView().Name() {
case "files":
return gui.enterFile(true, v.SelectedLineIdx())
}
return nil
}