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

383 lines
9.2 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"
2018-05-26 05:23:39 +02:00
"github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/utils"
"github.com/spkg/bom"
2018-05-26 05:23:39 +02:00
)
var cyclableViews = []string{"status", "files", "branches", "commits", "stash"}
2018-06-06 04:17:49 +02:00
2018-08-14 11:05:26 +02:00
func (gui *Gui) refreshSidePanels(g *gocui.Gui) error {
2018-12-07 09:52:31 +02:00
if err := gui.refreshBranches(g); err != nil {
return err
}
2018-12-08 07:54:54 +02:00
if err := gui.refreshFiles(); err != nil {
2018-12-07 09:52:31 +02:00
return err
}
if err := gui.refreshCommits(g); err != nil {
return err
}
return gui.refreshStashEntries(g)
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
if v == nil || v.Name() == cyclableViews[len(cyclableViews)-1] {
focusedViewName = cyclableViews[0]
} else {
for i := range cyclableViews {
if v.Name() == 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{
2018-08-15 11:49:43 +02:00
"name": v.Name(),
},
)
gui.Log.Info(message)
return nil
}
}
}
focusedView, err := g.View(focusedViewName)
if err != nil {
panic(err)
}
2018-08-14 11:05:26 +02:00
return gui.switchFocus(g, 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 {
var focusedViewName string
2018-07-22 04:58:39 +02:00
if v == nil || v.Name() == cyclableViews[0] {
focusedViewName = cyclableViews[len(cyclableViews)-1]
} else {
for i := range cyclableViews {
if v.Name() == 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{
2018-08-15 11:49:43 +02:00
"name": v.Name(),
},
)
gui.Log.Info(message)
return nil
}
}
}
focusedView, err := g.View(focusedViewName)
if err != nil {
panic(err)
}
2018-08-14 11:05:26 +02:00
return gui.switchFocus(g, v, focusedView)
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) newLineFocused(g *gocui.Gui, v *gocui.View) error {
switch v.Name() {
2018-09-05 11:12:11 +02:00
case "menu":
2018-12-07 09:52:31 +02:00
return gui.handleMenuSelect(g, v)
case "status":
return gui.handleStatusSelect(g, v)
case "files":
return gui.handleFileSelect(g, v, false)
case "branches":
2018-08-14 11:05:26 +02:00
return gui.handleBranchSelect(g, v)
case "commits":
return gui.handleCommitSelect(g, v)
case "stash":
return gui.handleStashEntrySelect(g, v)
case "confirmation":
return nil
2018-08-11 07:09:37 +02:00
case "commitMessage":
2018-08-14 11:05:26 +02:00
return gui.handleCommitFocused(g, v)
case "credentials":
return gui.handleCredentialsViewFocused(g, v)
case "main":
if gui.State.Contexts["main"] == "merging" {
return gui.refreshMergePanel()
}
v.Highlight = false
return nil
default:
panic(gui.Tr.SLocalize("NoViewMachingNewLineFocusedSwitchStatement"))
}
2018-06-06 04:17:49 +02:00
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) returnFocus(g *gocui.Gui, v *gocui.View) error {
previousView, err := 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
previousView, err = g.View("files")
if err != nil {
gui.Log.Error(err)
}
}
2018-08-14 11:05:26 +02:00
return gui.switchFocus(g, v, previousView)
2018-05-26 05:23:39 +02:00
}
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
2018-08-14 11:05:26 +02:00
func (gui *Gui) switchFocus(g *gocui.Gui, oldView, newView *gocui.View) error {
// we assume we'll never want to return focus to a confirmation panel i.e.
// we should never stack confirmation panels
if oldView != nil && oldView.Name() != "confirmation" {
// second class panels should never have focus restored to them because
// once they lose focus they are effectively 'destroyed'
secondClassPanels := []string{"confirmation", "menu"}
if !utils.IncludesString(secondClassPanels, 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)
if _, err := g.SetCurrentView(newView.Name()); err != nil {
return err
}
if _, err := g.SetViewOnTop(newView.Name()); err != nil {
return err
}
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
}
return gui.newLineFocused(g, 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 {
if err := v.SetCursor(0, 0); err != nil {
return err
}
return v.SetOrigin(0, 0)
2018-06-09 11:06:33 +02:00
}
// if the cursor down past the last item, move it to the last line
func (gui *Gui) focusPoint(cx int, cy int, v *gocui.View) error {
if cy < 0 {
return nil
}
ox, oy := v.Origin()
_, 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
}
if err := v.SetOrigin(ox, 0); err != nil {
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
2018-05-26 05:23:39 +02:00
}
func (gui *Gui) cleanString(s string) string {
output := string(bom.Clean([]byte(s)))
return utils.NormalizeLinefeeds(output)
}
func (gui *Gui) setViewContent(g *gocui.Gui, v *gocui.View, s string) error {
v.Clear()
fmt.Fprint(v, gui.cleanString(s))
return nil
}
// renderString resets the origin of a view and sets its content
2018-08-14 11:05:26 +02:00
func (gui *Gui) renderString(g *gocui.Gui, viewName, s string) error {
g.Update(func(*gocui.Gui) error {
v, err := 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
}
return gui.setViewContent(gui.g, v, s)
})
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 {
return gui.renderString(gui.g, "options", gui.optionsMapToString(optionsMap))
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
}
2018-12-08 07:54:54 +02:00
func (gui *Gui) getStashView() *gocui.View {
v, _ := gui.g.View("stash")
return v
}
2018-08-14 11:05:26 +02:00
func (gui *Gui) trimmedContent(v *gocui.View) string {
return strings.TrimSpace(v.Buffer())
}
func (gui *Gui) currentViewName(g *gocui.Gui) string {
currentView := g.CurrentView()
return currentView.Name()
}
2018-09-05 11:07:46 +02:00
func (gui *Gui) resizeCurrentPopupPanel(g *gocui.Gui) error {
v := g.CurrentView()
if v.Name() == "commitMessage" || v.Name() == "credentials" || v.Name() == "confirmation" {
2018-09-05 11:07:46 +02:00
return gui.resizePopupPanel(g, v)
}
return nil
}
func (gui *Gui) resizePopupPanel(g *gocui.Gui, v *gocui.View) error {
// If the confirmation panel is already displayed, just resize the width,
// otherwise continue
content := v.Buffer()
x0, y0, x1, y1 := gui.getConfirmationPanelDimensions(g, v.Wrap, content)
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"))
_, err := g.SetView(v.Name(), x0, y0, x1, y1, 0)
return err
}
// 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
}
}
2018-12-07 09:52:31 +02:00
func (gui *Gui) renderListPanel(v *gocui.View, items interface{}) error {
gui.g.Update(func(g *gocui.Gui) error {
isFocused := gui.g.CurrentView().Name() == v.Name()
list, err := utils.RenderList(items, isFocused)
2018-12-07 09:52:31 +02:00
if err != nil {
return gui.createErrorPanel(gui.g, err.Error())
}
v.Clear()
fmt.Fprint(v, list)
return nil
})
return nil
}
func (gui *Gui) renderPanelOptions() error {
currentView := gui.g.CurrentView()
switch currentView.Name() {
case "menu":
return gui.renderMenuOptions()
case "main":
if gui.State.Contexts["main"] == "merging" {
return gui.renderMergeOptions()
}
2018-12-07 09:52:31 +02:00
}
return gui.renderGlobalOptions()
2018-12-07 09:52:31 +02:00
}