2018-05-26 15:44:44 +10:00
package main
import (
2018-07-21 15:51:18 +10:00
"fmt"
2018-08-07 12:49:31 +10:00
"strings"
2018-06-01 23:23:31 +10:00
2018-07-21 15:51:18 +10:00
"github.com/jesseduffield/gocui"
2018-05-26 15:44:44 +10:00
)
func handleBranchPress ( g * gocui . Gui , v * gocui . View ) error {
2018-07-21 15:51:18 +10:00
index := getItemPosition ( v )
if index == 0 {
return createErrorPanel ( g , "You have already checked out this branch" )
}
branch := getSelectedBranch ( v )
if output , err := gitCheckout ( branch . Name , false ) ; err != nil {
createErrorPanel ( g , output )
}
return refreshSidePanels ( g )
2018-05-27 16:32:09 +10:00
}
func handleForceCheckout ( g * gocui . Gui , v * gocui . View ) error {
2018-07-29 10:40:54 +10:00
branch := getSelectedBranch ( v )
return createConfirmationPanel ( g , v , "Force Checkout Branch" , "Are you sure you want force checkout? You will lose all local changes" , func ( g * gocui . Gui , v * gocui . View ) error {
if output , err := gitCheckout ( branch . Name , true ) ; err != nil {
createErrorPanel ( g , output )
}
return refreshSidePanels ( g )
} , nil )
2018-05-26 15:44:44 +10:00
}
2018-06-09 19:30:59 +10:00
func handleCheckoutByName ( g * gocui . Gui , v * gocui . View ) error {
2018-07-21 15:51:18 +10:00
createPromptPanel ( g , v , "Branch Name:" , func ( g * gocui . Gui , v * gocui . View ) error {
if output , err := gitCheckout ( trimmedContent ( v ) , false ) ; err != nil {
return createErrorPanel ( g , output )
}
return refreshSidePanels ( g )
} )
return nil
2018-06-09 19:30:59 +10:00
}
2018-06-02 13:51:03 +10:00
func handleNewBranch ( g * gocui . Gui , v * gocui . View ) error {
2018-07-21 15:51:18 +10:00
branch := state . Branches [ 0 ]
createPromptPanel ( g , v , "New Branch Name (Branch is off of " + branch . Name + ")" , func ( g * gocui . Gui , v * gocui . View ) error {
if output , err := gitNewBranch ( trimmedContent ( v ) ) ; err != nil {
return createErrorPanel ( g , output )
}
refreshSidePanels ( g )
return handleBranchSelect ( g , v )
} )
return nil
2018-06-02 13:51:03 +10:00
}
2018-06-09 19:06:33 +10:00
func handleMerge ( g * gocui . Gui , v * gocui . View ) error {
2018-07-21 15:51:18 +10:00
checkedOutBranch := state . Branches [ 0 ]
selectedBranch := getSelectedBranch ( v )
defer refreshSidePanels ( g )
if checkedOutBranch . Name == selectedBranch . Name {
return createErrorPanel ( g , "You cannot merge a branch into itself" )
}
if output , err := gitMerge ( selectedBranch . Name ) ; err != nil {
return createErrorPanel ( g , output )
}
return nil
2018-06-09 19:06:33 +10:00
}
2018-05-26 15:44:44 +10:00
func getSelectedBranch ( v * gocui . View ) Branch {
2018-07-21 15:51:18 +10:00
lineNumber := getItemPosition ( v )
return state . Branches [ lineNumber ]
2018-05-26 15:44:44 +10:00
}
2018-06-09 19:06:33 +10:00
func renderBranchesOptions ( g * gocui . Gui ) error {
2018-07-21 15:51:18 +10:00
return renderOptionsMap ( g , map [ string ] string {
2018-07-29 10:40:54 +10:00
"space" : "checkout" ,
"f" : "force checkout" ,
"m" : "merge" ,
"c" : "checkout by name" ,
"n" : "new branch" ,
"← → ↑ ↓" : "navigate" ,
2018-07-21 15:51:18 +10:00
} )
2018-06-09 19:06:33 +10:00
}
2018-06-02 08:35:49 +10:00
// may want to standardise how these select methods work
2018-05-26 15:44:44 +10:00
func handleBranchSelect ( g * gocui . Gui , v * gocui . View ) error {
2018-07-21 15:51:18 +10:00
if err := renderBranchesOptions ( g ) ; err != nil {
return err
}
// This really shouldn't happen: there should always be a master branch
if len ( state . Branches ) == 0 {
return renderString ( g , "main" , "No branches for this repo" )
}
go func ( ) {
branch := getSelectedBranch ( v )
2018-08-10 21:33:49 +10:00
diff , err := getBranchGraph ( branch . Name )
2018-08-07 12:49:31 +10:00
if err != nil && strings . HasPrefix ( diff , "fatal: ambiguous argument" ) {
diff = "There is no tracking for this branch"
}
2018-07-21 15:51:18 +10:00
renderString ( g , "main" , diff )
} ( )
return nil
2018-05-26 15:44:44 +10:00
}
2018-06-01 23:23:31 +10:00
// refreshStatus is called at the end of this because that's when we can
// be sure there is a state.Branches array to pick the current branch from
2018-05-27 16:32:09 +10:00
func refreshBranches ( g * gocui . Gui ) error {
2018-07-21 15:51:18 +10:00
g . Update ( func ( g * gocui . Gui ) error {
v , err := g . View ( "branches" )
if err != nil {
panic ( err )
}
state . Branches = getGitBranches ( )
v . Clear ( )
for _ , branch := range state . Branches {
2018-08-10 21:33:49 +10:00
fmt . Fprintln ( v , branch . getDisplayString ( ) )
2018-07-21 15:51:18 +10:00
}
resetOrigin ( v )
return refreshStatus ( g )
} )
return nil
2018-05-26 15:44:44 +10:00
}