1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-22 05:29:44 +02:00
lazygit/pkg/gui/confirmation_panel.go

191 lines
6.7 KiB
Go
Raw Normal View History

2018-05-26 13:23:39 +10:00
// lots of this has been directly ported from one of the example files, will brush up later
// Copyright 2014 The gocui Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
2018-08-14 11:05:26 +02:00
package gui
2018-05-26 13:23:39 +10:00
import (
2018-08-05 22:00:02 +10:00
"strings"
2018-12-05 19:33:46 +11:00
"time"
2018-05-26 13:23:39 +10:00
2018-08-05 22:00:02 +10:00
"github.com/fatih/color"
"github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/theme"
2018-05-26 13:23:39 +10:00
)
func (gui *Gui) wrappedConfirmationFunction(function func(*gocui.Gui, *gocui.View) error, returnFocusOnClose bool) func(*gocui.Gui, *gocui.View) error {
2018-08-05 22:00:02 +10:00
return func(g *gocui.Gui, v *gocui.View) error {
2019-11-17 12:02:39 +11:00
2018-08-05 22:00:02 +10:00
if function != nil {
if err := function(g, v); err != nil {
2018-08-23 18:43:16 +10:00
return err
2018-08-05 22:00:02 +10:00
}
}
2019-11-17 12:02:39 +11:00
2019-11-17 17:58:24 +11:00
return gui.closeConfirmationPrompt(g, returnFocusOnClose)
2018-08-05 22:00:02 +10:00
}
2018-05-27 16:32:09 +10:00
}
func (gui *Gui) closeConfirmationPrompt(g *gocui.Gui, returnFocusOnClose bool) error {
2018-08-05 22:00:02 +10:00
view, err := g.View("confirmation")
if err != nil {
2018-12-07 18:52:31 +11:00
return nil // if it's already been closed we can just return
2018-08-05 22:00:02 +10:00
}
2019-11-17 17:58:24 +11:00
view.Editable = false
if returnFocusOnClose {
if err := gui.returnFocus(g, view); err != nil {
panic(err)
}
2018-08-05 22:00:02 +10:00
}
g.DeleteKeybindings("confirmation")
return g.DeleteView("confirmation")
2018-05-27 16:32:09 +10:00
}
func (gui *Gui) getMessageHeight(wrap bool, message string, width int) int {
2018-08-05 22:00:02 +10:00
lines := strings.Split(message, "\n")
lineCount := 0
// if we need to wrap, calculate height to fit content within view's width
if wrap {
for _, line := range lines {
lineCount += len(line)/width + 1
}
} else {
lineCount = len(lines)
2018-08-05 22:00:02 +10:00
}
return lineCount
2018-05-26 13:23:39 +10:00
}
func (gui *Gui) getConfirmationPanelDimensions(g *gocui.Gui, wrap bool, prompt string) (int, int, int, int) {
2018-08-05 22:00:02 +10:00
width, height := g.Size()
2019-11-10 22:07:45 +11:00
panelWidth := 4 * width / 7
panelHeight := gui.getMessageHeight(wrap, prompt, panelWidth)
2018-08-05 22:00:02 +10:00
return width/2 - panelWidth/2,
height/2 - panelHeight/2 - panelHeight%2 - 1,
width/2 + panelWidth/2,
height/2 + panelHeight/2
2018-05-26 13:23:39 +10:00
}
2019-02-16 12:03:22 +11:00
func (gui *Gui) prepareConfirmationPanel(currentView *gocui.View, title, prompt string, hasLoader bool) (*gocui.View, error) {
x0, y0, x1, y1 := gui.getConfirmationPanelDimensions(gui.g, true, prompt)
2018-08-25 15:55:49 +10:00
confirmationView, err := gui.g.SetView("confirmation", x0, y0, x1, y1, 0)
if err != nil {
2019-02-16 12:03:22 +11:00
if err.Error() != "unknown view" {
2018-08-25 15:55:49 +10:00
return nil, err
2018-08-05 22:00:02 +10:00
}
2019-02-16 12:03:22 +11:00
confirmationView.HasLoader = hasLoader
if hasLoader {
gui.g.StartTicking()
}
2018-08-05 22:00:02 +10:00
confirmationView.Title = title
2018-12-05 19:33:46 +11:00
confirmationView.Wrap = true
confirmationView.FgColor = theme.GocuiDefaultTextColor
2018-08-05 22:00:02 +10:00
}
2018-12-07 18:52:31 +11:00
gui.g.Update(func(g *gocui.Gui) error {
return gui.switchFocus(gui.g, currentView, confirmationView)
})
2018-08-25 15:55:49 +10:00
return confirmationView, nil
2018-05-27 16:32:09 +10:00
}
2018-08-15 21:49:38 +10:00
func (gui *Gui) onNewPopupPanel() {
2019-02-11 21:07:12 +11:00
viewNames := []string{"commitMessage",
"credentials",
"menu"}
for _, viewName := range viewNames {
_, _ = gui.g.SetViewOnBottom(viewName)
}
2018-08-15 21:49:38 +10:00
}
2019-11-17 17:58:24 +11:00
func (gui *Gui) createPopupPanel(g *gocui.Gui, currentView *gocui.View, title, prompt string, hasLoader bool, returnFocusOnClose bool, editable bool, handleConfirm, handleClose func(*gocui.Gui, *gocui.View) error) error {
2018-08-15 21:49:38 +10:00
gui.onNewPopupPanel()
2018-08-05 22:00:02 +10:00
g.Update(func(g *gocui.Gui) error {
// delete the existing confirmation panel if it exists
if view, _ := g.View("confirmation"); view != nil {
if err := gui.closeConfirmationPrompt(g, true); err != nil {
2019-11-17 17:58:24 +11:00
gui.Log.Error(err)
2018-08-05 22:00:02 +10:00
}
}
2019-02-16 12:03:22 +11:00
confirmationView, err := gui.prepareConfirmationPanel(currentView, title, prompt, hasLoader)
2018-08-25 15:55:49 +10:00
if err != nil {
return err
2018-08-05 22:00:02 +10:00
}
2019-11-17 17:58:24 +11:00
confirmationView.Editable = editable
if editable {
go func() {
// TODO: remove this wait (right now if you remove it the EditGotoToEndOfLine method doesn't seem to work)
time.Sleep(time.Millisecond)
gui.g.Update(func(g *gocui.Gui) error {
confirmationView.EditGotoToEndOfLine()
return nil
})
}()
}
2018-08-25 15:55:49 +10:00
if err := gui.renderString(g, "confirmation", prompt); err != nil {
return err
}
return gui.setKeyBindings(g, handleConfirm, handleClose, returnFocusOnClose)
2018-08-05 22:00:02 +10:00
})
return nil
}
2019-11-17 17:58:24 +11:00
func (gui *Gui) createLoaderPanel(g *gocui.Gui, currentView *gocui.View, prompt string) error {
return gui.createPopupPanel(g, currentView, "", prompt, true, true, false, nil, nil)
}
// it is very important that within this function we never include the original prompt in any error messages, because it may contain e.g. a user password
func (gui *Gui) createConfirmationPanel(g *gocui.Gui, currentView *gocui.View, returnFocusOnClose bool, title, prompt string, handleConfirm, handleClose func(*gocui.Gui, *gocui.View) error) error {
return gui.createPopupPanel(g, currentView, title, prompt, false, returnFocusOnClose, false, handleConfirm, handleClose)
}
func (gui *Gui) createPromptPanel(g *gocui.Gui, currentView *gocui.View, title string, initialContent string, handleConfirm func(*gocui.Gui, *gocui.View) error) error {
return gui.createPopupPanel(gui.g, currentView, title, initialContent, false, true, true, handleConfirm, nil)
}
func (gui *Gui) setKeyBindings(g *gocui.Gui, handleConfirm, handleClose func(*gocui.Gui, *gocui.View) error, returnFocusOnClose bool) error {
actions := gui.Tr.TemplateLocalize(
"CloseConfirm",
Teml{
"keyBindClose": "esc",
"keyBindConfirm": "enter",
},
)
2018-08-25 15:55:49 +10:00
if err := gui.renderString(g, "options", actions); err != nil {
return err
}
2019-11-16 12:41:04 +11:00
if err := g.SetKeybinding("confirmation", nil, gocui.KeyEnter, gocui.ModNone, gui.wrappedConfirmationFunction(handleConfirm, returnFocusOnClose)); err != nil {
2018-08-05 22:00:02 +10:00
return err
}
2019-11-16 12:41:04 +11:00
return g.SetKeybinding("confirmation", nil, gocui.KeyEsc, gocui.ModNone, gui.wrappedConfirmationFunction(handleClose, returnFocusOnClose))
2018-05-26 13:23:39 +10:00
}
2018-05-27 16:32:09 +10:00
2018-08-14 11:05:26 +02:00
func (gui *Gui) createMessagePanel(g *gocui.Gui, currentView *gocui.View, title, prompt string) error {
2019-11-17 17:58:24 +11:00
return gui.createPopupPanel(g, currentView, title, prompt, false, true, false, nil, nil)
2018-05-27 16:32:09 +10:00
}
2018-06-01 23:23:31 +10:00
// createSpecificErrorPanel allows you to create an error popup, specifying the
// view to be focused when the user closes the popup, and a boolean specifying
// whether we will log the error. If the message may include a user password,
// this function is to be used over the more generic createErrorPanel, with
// willLog set to false
func (gui *Gui) createSpecificErrorPanel(message string, nextView *gocui.View, willLog bool) error {
if willLog {
go func() {
// when reporting is switched on this log call sometimes introduces
// a delay on the error panel popping up. Here I'm adding a second wait
// so that the error is logged while the user is reading the error message
time.Sleep(time.Second)
gui.Log.Error(message)
}()
}
2018-08-05 22:00:02 +10:00
colorFunction := color.New(color.FgRed).SprintFunc()
coloredMessage := colorFunction(strings.TrimSpace(message))
return gui.createConfirmationPanel(gui.g, nextView, true, gui.Tr.SLocalize("Error"), coloredMessage, nil, nil)
}
func (gui *Gui) createErrorPanel(g *gocui.Gui, message string) error {
return gui.createSpecificErrorPanel(message, g.CurrentView(), true)
2018-06-01 23:23:31 +10:00
}