2020-09-26 15:23:28 +10:00
package gui
import (
2020-09-26 17:15:13 +10:00
"log"
2020-09-27 09:37:22 +10:00
"strings"
2020-09-26 15:23:28 +10:00
2020-09-27 09:13:31 +10:00
"github.com/fatih/color"
2020-09-26 17:15:13 +10:00
"github.com/jesseduffield/gocui"
2020-09-29 20:28:39 +10:00
"github.com/jesseduffield/lazygit/pkg/commands/models"
2020-10-03 14:54:55 +10:00
"github.com/jesseduffield/lazygit/pkg/config"
2020-09-27 09:13:31 +10:00
"github.com/jesseduffield/lazygit/pkg/utils"
2020-09-26 15:23:28 +10:00
)
type CustomCommandObjects struct {
2020-09-29 18:36:54 +10:00
SelectedLocalCommit * models . Commit
SelectedReflogCommit * models . Commit
SelectedSubCommit * models . Commit
2020-09-29 18:45:00 +10:00
SelectedFile * models . File
2021-03-15 23:00:20 +11:00
SelectedPath string
2020-09-29 18:34:01 +10:00
SelectedLocalBranch * models . Branch
2020-09-29 18:41:06 +10:00
SelectedRemoteBranch * models . RemoteBranch
SelectedRemote * models . Remote
2020-09-29 18:38:56 +10:00
SelectedTag * models . Tag
2020-09-29 18:46:45 +10:00
SelectedStashEntry * models . StashEntry
2020-09-29 18:48:38 +10:00
SelectedCommitFile * models . CommitFile
2020-09-29 18:34:01 +10:00
CheckedOutBranch * models . Branch
2020-09-26 19:48:13 +10:00
PromptResponses [ ] string
}
func ( gui * Gui ) resolveTemplate ( templateStr string , promptResponses [ ] string ) ( string , error ) {
objects := CustomCommandObjects {
SelectedFile : gui . getSelectedFile ( ) ,
2021-03-15 23:00:20 +11:00
SelectedPath : gui . getSelectedPath ( ) ,
2020-09-26 19:48:13 +10:00
SelectedLocalCommit : gui . getSelectedLocalCommit ( ) ,
SelectedReflogCommit : gui . getSelectedReflogCommit ( ) ,
SelectedLocalBranch : gui . getSelectedBranch ( ) ,
SelectedRemoteBranch : gui . getSelectedRemoteBranch ( ) ,
SelectedRemote : gui . getSelectedRemote ( ) ,
SelectedTag : gui . getSelectedTag ( ) ,
SelectedStashEntry : gui . getSelectedStashEntry ( ) ,
SelectedCommitFile : gui . getSelectedCommitFile ( ) ,
SelectedSubCommit : gui . getSelectedSubCommit ( ) ,
CheckedOutBranch : gui . currentBranch ( ) ,
PromptResponses : promptResponses ,
}
2020-09-30 21:12:03 +10:00
return utils . ResolveTemplate ( templateStr , objects )
2020-09-26 15:23:28 +10:00
}
2020-10-03 14:54:55 +10:00
func ( gui * Gui ) handleCustomCommandKeybinding ( customCommand config . CustomCommand ) func ( ) error {
2020-09-26 15:23:28 +10:00
return func ( ) error {
2020-09-26 19:48:13 +10:00
promptResponses := make ( [ ] string , len ( customCommand . Prompts ) )
2020-09-26 15:23:28 +10:00
2020-09-26 19:48:13 +10:00
f := func ( ) error {
cmdStr , err := gui . resolveTemplate ( customCommand . Command , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
2020-09-26 15:23:28 +10:00
2020-09-26 19:48:13 +10:00
if customCommand . Subprocess {
gui . PrepareSubProcess ( cmdStr )
return nil
}
2020-09-26 15:23:28 +10:00
2020-09-27 09:21:20 +10:00
loadingText := customCommand . LoadingText
if loadingText == "" {
2020-10-04 11:00:48 +11:00
loadingText = gui . Tr . LcRunningCustomCommandStatus
2020-09-27 09:21:20 +10:00
}
return gui . WithWaitingStatus ( loadingText , func ( ) error {
2020-09-26 19:48:13 +10:00
gui . OSCommand . PrepareSubProcess ( cmdStr )
2020-09-26 15:23:28 +10:00
2020-09-26 19:48:13 +10:00
if err := gui . OSCommand . RunCommand ( cmdStr ) ; err != nil {
return gui . surfaceError ( err )
}
return gui . refreshSidePanels ( refreshOptions { } )
} )
2020-09-26 17:15:13 +10:00
}
2020-09-26 19:48:13 +10:00
// if we have prompts we'll recursively wrap our confirm handlers with more prompts
// until we reach the actual command
for reverseIdx := range customCommand . Prompts {
idx := len ( customCommand . Prompts ) - 1 - reverseIdx
2020-09-26 17:15:13 +10:00
2020-09-26 19:48:13 +10:00
// going backwards so the outermost prompt is the first one
prompt := customCommand . Prompts [ idx ]
2020-09-26 21:47:01 +10:00
// need to do this because f's value will change with each iteration
wrappedF := f
2020-09-26 20:32:19 +10:00
switch prompt . Type {
2020-09-27 09:13:31 +10:00
case "input" :
2020-09-26 20:32:19 +10:00
f = func ( ) error {
2020-09-26 21:47:01 +10:00
title , err := gui . resolveTemplate ( prompt . Title , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
initialValue , err := gui . resolveTemplate ( prompt . InitialValue , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
2020-11-28 13:35:58 +11:00
return gui . prompt ( promptOpts {
title : title ,
initialContent : initialValue ,
handleConfirm : func ( str string ) error {
2020-09-26 20:32:19 +10:00
promptResponses [ idx ] = str
return wrappedF ( )
} ,
2020-11-28 13:35:58 +11:00
} )
2020-09-26 20:32:19 +10:00
}
case "menu" :
2020-09-26 21:47:01 +10:00
f = func ( ) error {
// need to make a menu here some how
menuItems := make ( [ ] * menuItem , len ( prompt . Options ) )
for i , option := range prompt . Options {
option := option
2020-09-27 09:11:19 +10:00
nameTemplate := option . Name
if nameTemplate == "" {
// this allows you to only pass values rather than bother with names/descriptions
nameTemplate = option . Value
}
name , err := gui . resolveTemplate ( nameTemplate , promptResponses )
2020-09-26 21:47:01 +10:00
if err != nil {
return gui . surfaceError ( err )
}
description , err := gui . resolveTemplate ( option . Description , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
value , err := gui . resolveTemplate ( option . Value , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
menuItems [ i ] = & menuItem {
2020-09-27 09:13:31 +10:00
displayStrings : [ ] string { name , utils . ColoredString ( description , color . FgYellow ) } ,
2020-09-26 21:47:01 +10:00
onPress : func ( ) error {
promptResponses [ idx ] = value
return wrappedF ( )
} ,
}
}
2020-09-26 20:32:19 +10:00
2020-09-26 21:47:01 +10:00
title , err := gui . resolveTemplate ( prompt . Title , promptResponses )
if err != nil {
return gui . surfaceError ( err )
2020-09-26 20:32:19 +10:00
}
2020-09-26 21:47:01 +10:00
return gui . createMenu ( title , menuItems , createMenuOptions { showCancel : true } )
2020-09-26 20:32:19 +10:00
}
default :
2020-09-27 09:13:31 +10:00
return gui . createErrorPanel ( "custom command prompt must have a type of 'input' or 'menu'" )
2020-09-26 15:23:28 +10:00
}
2020-09-26 20:32:19 +10:00
2020-09-26 19:48:13 +10:00
}
return f ( )
2020-09-26 15:23:28 +10:00
}
}
2020-09-26 17:15:13 +10:00
func ( gui * Gui ) GetCustomCommandKeybindings ( ) [ ] * Binding {
bindings := [ ] * Binding { }
2020-10-03 14:54:55 +10:00
customCommands := gui . Config . GetUserConfig ( ) . CustomCommands
2020-09-26 17:15:13 +10:00
for _ , customCommand := range customCommands {
var viewName string
2020-09-27 11:04:57 +10:00
var contexts [ ] string
2020-09-27 09:37:22 +10:00
switch customCommand . Context {
case "global" :
2020-09-26 17:15:13 +10:00
viewName = ""
2020-09-27 09:37:22 +10:00
case "" :
log . Fatalf ( "Error parsing custom command keybindings: context not provided (use context: 'global' for the global context). Key: %s, Command: %s" , customCommand . Key , customCommand . Command )
default :
context , ok := gui . contextForContextKey ( customCommand . Context )
if ! ok {
log . Fatalf ( "Error when setting custom command keybindings: unknown context: %s. Key: %s, Command: %s.\nPermitted contexts: %s" , customCommand . Context , customCommand . Key , customCommand . Command , strings . Join ( allContextKeys , ", " ) )
2020-09-26 17:15:13 +10:00
}
// here we assume that a given context will always belong to the same view.
// Currently this is a safe bet but it's by no means guaranteed in the long term
// and we might need to make some changes in the future to support it.
viewName = context . GetViewName ( )
2020-09-27 11:04:57 +10:00
contexts = [ ] string { customCommand . Context }
2020-09-26 17:15:13 +10:00
}
2020-09-27 11:29:10 +10:00
description := customCommand . Description
if description == "" {
description = customCommand . Command
}
2020-09-26 17:15:13 +10:00
bindings = append ( bindings , & Binding {
ViewName : viewName ,
2020-09-27 11:04:57 +10:00
Contexts : contexts ,
2020-09-26 17:15:13 +10:00
Key : gui . getKey ( customCommand . Key ) ,
Modifier : gocui . ModNone ,
Handler : gui . wrappedHandler ( gui . handleCustomCommandKeybinding ( customCommand ) ) ,
2020-09-27 11:29:10 +10:00
Description : description ,
2020-09-26 17:15:13 +10:00
} )
}
return bindings
}