2020-09-26 15:23:28 +10:00
package gui
import (
2021-07-19 11:46:29 +01:00
"bytes"
"errors"
2020-09-26 17:15:13 +10:00
"log"
2021-07-18 18:38:06 +01:00
"regexp"
"strconv"
2020-09-27 09:37:22 +10:00
"strings"
2021-07-19 11:46:29 +01:00
"text/template"
2020-09-26 15:23:28 +10:00
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"
2021-07-27 15:00:37 +02:00
"github.com/jesseduffield/lazygit/pkg/gui/style"
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 {
2021-03-31 22:08:55 +11:00
SelectedLocalCommit * models . Commit
SelectedReflogCommit * models . Commit
SelectedSubCommit * models . Commit
SelectedFile * models . File
SelectedPath string
SelectedLocalBranch * models . Branch
SelectedRemoteBranch * models . RemoteBranch
SelectedRemote * models . Remote
SelectedTag * models . Tag
SelectedStashEntry * models . StashEntry
SelectedCommitFile * models . CommitFile
SelectedCommitFilePath string
CheckedOutBranch * models . Branch
PromptResponses [ ] string
2020-09-26 19:48:13 +10:00
}
func ( gui * Gui ) resolveTemplate ( templateStr string , promptResponses [ ] string ) ( string , error ) {
objects := CustomCommandObjects {
2021-03-31 22:08:55 +11:00
SelectedFile : gui . getSelectedFile ( ) ,
SelectedPath : gui . getSelectedPath ( ) ,
SelectedLocalCommit : gui . getSelectedLocalCommit ( ) ,
SelectedReflogCommit : gui . getSelectedReflogCommit ( ) ,
SelectedLocalBranch : gui . getSelectedBranch ( ) ,
SelectedRemoteBranch : gui . getSelectedRemoteBranch ( ) ,
SelectedRemote : gui . getSelectedRemote ( ) ,
SelectedTag : gui . getSelectedTag ( ) ,
SelectedStashEntry : gui . getSelectedStashEntry ( ) ,
SelectedCommitFile : gui . getSelectedCommitFile ( ) ,
SelectedCommitFilePath : gui . getSelectedCommitFilePath ( ) ,
SelectedSubCommit : gui . getSelectedSubCommit ( ) ,
CheckedOutBranch : gui . currentBranch ( ) ,
PromptResponses : promptResponses ,
2020-09-26 19:48:13 +10:00
}
2020-09-30 21:12:03 +10:00
return utils . ResolveTemplate ( templateStr , objects )
2020-09-26 15:23:28 +10:00
}
2021-07-20 20:59:03 +01:00
func ( gui * Gui ) inputPrompt ( prompt config . CustomCommandPrompt , promptResponses [ ] string , responseIdx int , wrappedF func ( ) error ) error {
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 )
}
return gui . prompt ( promptOpts {
title : title ,
initialContent : initialValue ,
handleConfirm : func ( str string ) error {
promptResponses [ responseIdx ] = str
return wrappedF ( )
} ,
} )
}
func ( gui * Gui ) menuPrompt ( prompt config . CustomCommandPrompt , promptResponses [ ] string , responseIdx int , wrappedF func ( ) error ) error {
// need to make a menu here some how
menuItems := make ( [ ] * menuItem , len ( prompt . Options ) )
for i , option := range prompt . Options {
option := option
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 )
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 {
2021-07-27 15:00:37 +02:00
displayStrings : [ ] string { name , style . FgYellow . Sprint ( description ) } ,
2021-07-20 20:59:03 +01:00
onPress : func ( ) error {
promptResponses [ responseIdx ] = value
return wrappedF ( )
} ,
}
}
title , err := gui . resolveTemplate ( prompt . Title , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
return gui . createMenu ( title , menuItems , createMenuOptions { showCancel : true } )
}
2021-07-22 15:44:16 +01:00
func ( gui * Gui ) GenerateMenuCandidates ( commandOutput string , filter string , format string ) ( [ ] string , error ) {
2021-07-20 20:59:03 +01:00
candidates := [ ] string { }
reg , err := regexp . Compile ( filter )
if err != nil {
return candidates , gui . surfaceError ( errors . New ( "unable to parse filter regex, error: " + err . Error ( ) ) )
}
buff := bytes . NewBuffer ( nil )
temp , err := template . New ( "format" ) . Parse ( format )
if err != nil {
return candidates , gui . surfaceError ( errors . New ( "unable to parse format, error: " + err . Error ( ) ) )
}
for _ , str := range strings . Split ( string ( commandOutput ) , "\n" ) {
if str == "" {
continue
}
tmplData := map [ string ] string { }
out := reg . FindAllStringSubmatch ( str , - 1 )
if len ( out ) > 0 {
for groupIdx , group := range reg . SubexpNames ( ) {
// Record matched group with group ids
matchName := "group_" + strconv . Itoa ( groupIdx )
2021-07-22 19:45:43 +02:00
tmplData [ matchName ] = out [ 0 ] [ groupIdx ]
2021-07-20 20:59:03 +01:00
// Record last named group non-empty matches as group matches
if group != "" {
tmplData [ group ] = out [ 0 ] [ groupIdx ]
}
}
}
err = temp . Execute ( buff , tmplData )
if err != nil {
return candidates , gui . surfaceError ( err )
}
candidates = append ( candidates , strings . TrimSpace ( buff . String ( ) ) )
buff . Reset ( )
}
return candidates , err
}
func ( gui * Gui ) menuPromptFromCommand ( prompt config . CustomCommandPrompt , promptResponses [ ] string , responseIdx int , wrappedF func ( ) error ) error {
// Collect cmd to run from config
cmdStr , err := gui . resolveTemplate ( prompt . Command , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
// Collect Filter regexp
filter , err := gui . resolveTemplate ( prompt . Filter , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
// Run and save output
message , err := gui . GitCommand . RunCommandWithOutput ( cmdStr )
if err != nil {
return gui . surfaceError ( err )
}
// Need to make a menu out of what the cmd has displayed
2021-07-22 15:44:16 +01:00
candidates , err := gui . GenerateMenuCandidates ( message , filter , prompt . Format )
if err != nil {
return gui . surfaceError ( err )
}
2021-07-20 20:59:03 +01:00
menuItems := make ( [ ] * menuItem , len ( candidates ) )
for i := range candidates {
menuItems [ i ] = & menuItem {
displayStrings : [ ] string { candidates [ i ] } ,
onPress : func ( ) error {
promptResponses [ responseIdx ] = candidates [ i ]
return wrappedF ( )
} ,
}
}
title , err := gui . resolveTemplate ( prompt . Title , promptResponses )
if err != nil {
return gui . surfaceError ( err )
}
return gui . createMenu ( title , menuItems , createMenuOptions { showCancel : true } )
}
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 {
2021-04-10 11:40:42 +10:00
return gui . runSubprocessWithSuspenseAndRefresh ( gui . OSCommand . PrepareShellSubProcess ( cmdStr ) )
2020-09-26 19:48:13 +10:00
}
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 {
2021-04-11 19:35:42 +10:00
if err := gui . OSCommand . WithSpan ( gui . Tr . Spans . CustomCommand ) . RunShellCommand ( cmdStr ) ; err != nil {
2020-09-26 19:48:13 +10:00
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 {
2021-07-20 20:59:03 +01:00
return gui . inputPrompt ( prompt , promptResponses , idx , wrappedF )
2020-09-26 20:32:19 +10:00
}
case "menu" :
2020-09-26 21:47:01 +10:00
f = func ( ) error {
2021-07-20 20:59:03 +01:00
return gui . menuPrompt ( prompt , promptResponses , idx , wrappedF )
2021-07-17 18:02:11 +01:00
}
case "menuFromCommand" :
f = func ( ) error {
2021-07-20 20:59:03 +01:00
return gui . menuPromptFromCommand ( prompt , promptResponses , idx , wrappedF )
2020-09-26 20:32:19 +10:00
}
default :
2021-07-17 18:02:11 +01:00
return gui . createErrorPanel ( "custom command prompt must have a type of 'input', 'menu' or 'menuFromCommand'" )
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 :
2021-04-04 23:51:59 +10:00
context , ok := gui . contextForContextKey ( ContextKey ( customCommand . Context ) )
// stupid golang making me build an array of strings for this.
allContextKeyStrings := make ( [ ] string , len ( allContextKeys ) )
for i := range allContextKeys {
allContextKeyStrings [ i ] = string ( allContextKeys [ i ] )
}
2020-09-27 09:37:22 +10:00
if ! ok {
2021-04-04 23:51:59 +10:00
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 ( allContextKeyStrings , ", " ) )
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 ,
2021-04-02 19:20:40 +11:00
Handler : gui . handleCustomCommandKeybinding ( customCommand ) ,
2020-09-27 11:29:10 +10:00
Description : description ,
2020-09-26 17:15:13 +10:00
} )
}
return bindings
}