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"
2019-10-18 09:48:37 +02:00
"github.com/jesseduffield/lazygit/pkg/theme"
2018-05-26 13:23:39 +10:00
)
2019-11-05 15:19:43 +11: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
}
2019-11-05 15:19:43 +11: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
2019-11-05 15:19:43 +11:00
if returnFocusOnClose {
if err := gui . returnFocus ( g , view ) ; err != nil {
panic ( err )
}
2018-08-05 22:00:02 +10:00
}
2020-03-26 21:39:59 +11:00
g . DeleteKeybinding ( "confirmation" , gocui . KeyEnter , gocui . ModNone )
g . DeleteKeybinding ( "confirmation" , gocui . KeyEsc , gocui . ModNone )
2018-08-05 22:00:02 +10:00
return g . DeleteView ( "confirmation" )
2018-05-27 16:32:09 +10:00
}
2019-01-15 19:33:42 +11: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
2018-12-26 20:39:16 +09:00
// if we need to wrap, calculate height to fit content within view's width
2019-01-15 19:33:42 +11:00
if wrap {
2018-12-26 20:39:16 +09:00
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
}
2019-01-15 19:33:42 +11: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
2019-01-15 19:33:42 +11:00
panelHeight := gui . getMessageHeight ( wrap , prompt , panelWidth )
2020-03-26 21:39:59 +11:00
if panelHeight > height * 3 / 4 {
panelHeight = height * 3 / 4
}
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 ) {
2019-01-15 19:33:42 +11:00
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
2020-02-01 00:06:07 +11:00
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
2019-10-18 09:48:37 +02:00
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" ,
2019-02-11 21:02:53 +11:00
"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 {
2019-11-05 15:19:43 +11:00
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
} )
} ( )
}
2020-03-09 11:34:10 +11:00
gui . renderString ( g , "confirmation" , prompt )
2020-03-28 12:52:45 +11:00
return gui . setKeyBindings ( g , handleConfirm , handleClose , returnFocusOnClose )
2018-08-05 22:00:02 +10:00
} )
return nil
2018-06-05 18:47:57 +10:00
}
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 )
}
2019-11-05 15:19:43 +11:00
func ( gui * Gui ) setKeyBindings ( g * gocui . Gui , handleConfirm , handleClose func ( * gocui . Gui , * gocui . View ) error , returnFocusOnClose bool ) error {
2018-08-15 10:53:05 +02:00
actions := gui . Tr . TemplateLocalize (
"CloseConfirm" ,
2018-08-16 11:31:03 +02:00
Teml {
2018-08-15 10:53:05 +02:00
"keyBindClose" : "esc" ,
"keyBindConfirm" : "enter" ,
} ,
)
2020-03-26 21:39:59 +11:00
2020-03-09 11:34:10 +11:00
gui . renderString ( g , "options" , actions )
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-12-16 17:05:34 +11: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 ) )
2020-03-28 12:52:45 +11:00
if err := gui . refreshSidePanels ( refreshOptions { mode : ASYNC } ) ; err != nil {
return err
}
2019-11-05 15:19:43 +11:00
return gui . createConfirmationPanel ( gui . g , nextView , true , gui . Tr . SLocalize ( "Error" ) , coloredMessage , nil , nil )
2018-12-16 17:05:34 +11:00
}
2020-03-28 11:47:54 +11:00
func ( gui * Gui ) createErrorPanel ( message string ) error {
return gui . createSpecificErrorPanel ( message , gui . g . CurrentView ( ) , true )
}
func ( gui * Gui ) surfaceError ( err error ) error {
return gui . createErrorPanel ( err . Error ( ) )
2018-06-01 23:23:31 +10:00
}