2020-08-22 10:22:06 +10:00
|
|
|
package gui
|
|
|
|
|
2020-09-29 20:28:39 +10:00
|
|
|
import "github.com/jesseduffield/lazygit/pkg/commands/models"
|
2020-08-22 10:22:06 +10:00
|
|
|
|
2020-08-22 11:05:37 +10:00
|
|
|
// you can only copy from one context at a time, because the order and position of commits matter
|
|
|
|
|
2020-08-22 11:44:03 +10:00
|
|
|
func (gui *Gui) resetCherryPickingIfNecessary(context Context) error {
|
2021-06-05 15:08:36 +10:00
|
|
|
oldContextKey := ContextKey(gui.State.Modes.CherryPicking.ContextKey)
|
2020-08-22 11:44:03 +10:00
|
|
|
|
|
|
|
if oldContextKey != context.GetKey() {
|
|
|
|
// need to reset the cherry picking mode
|
2021-06-05 15:08:36 +10:00
|
|
|
gui.State.Modes.CherryPicking.ContextKey = string(context.GetKey())
|
2020-09-29 18:36:54 +10:00
|
|
|
gui.State.Modes.CherryPicking.CherryPickedCommits = make([]*models.Commit, 0)
|
2020-08-22 11:44:03 +10:00
|
|
|
|
|
|
|
return gui.rerenderContextViewIfPresent(oldContextKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:05:37 +10:00
|
|
|
func (gui *Gui) handleCopyCommit() error {
|
2020-08-22 10:22:06 +10:00
|
|
|
if ok, err := gui.validateNotInFilterMode(); err != nil || !ok {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// get currently selected commit, add the sha to state.
|
2021-04-04 23:51:59 +10:00
|
|
|
context := gui.currentSideListContext()
|
2020-08-22 11:05:37 +10:00
|
|
|
if context == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:44:03 +10:00
|
|
|
if err := gui.resetCherryPickingIfNecessary(context); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-08-22 15:56:30 +10:00
|
|
|
item, ok := context.SelectedItem()
|
2020-08-22 11:05:37 +10:00
|
|
|
if !ok {
|
2020-08-22 15:56:30 +10:00
|
|
|
return nil
|
2020-08-22 11:05:37 +10:00
|
|
|
}
|
2020-09-29 18:36:54 +10:00
|
|
|
commit, ok := item.(*models.Commit)
|
2020-08-22 15:56:30 +10:00
|
|
|
if !ok {
|
2020-08-22 11:05:37 +10:00
|
|
|
return nil
|
|
|
|
}
|
2020-08-22 10:22:06 +10:00
|
|
|
|
|
|
|
// we will un-copy it if it's already copied
|
2020-08-22 11:05:37 +10:00
|
|
|
for index, cherryPickedCommit := range gui.State.Modes.CherryPicking.CherryPickedCommits {
|
2020-08-22 10:22:06 +10:00
|
|
|
if commit.Sha == cherryPickedCommit.Sha {
|
2020-08-22 11:05:37 +10:00
|
|
|
gui.State.Modes.CherryPicking.CherryPickedCommits = append(gui.State.Modes.CherryPicking.CherryPickedCommits[0:index], gui.State.Modes.CherryPicking.CherryPickedCommits[index+1:]...)
|
|
|
|
return context.HandleRender()
|
2020-08-22 10:22:06 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:44:03 +10:00
|
|
|
gui.addCommitToCherryPickedCommits(context.GetPanelState().GetSelectedLineIdx())
|
2020-08-22 11:05:37 +10:00
|
|
|
return context.HandleRender()
|
2020-08-22 10:22:06 +10:00
|
|
|
}
|
|
|
|
|
2020-08-22 11:57:44 +10:00
|
|
|
func (gui *Gui) cherryPickedCommitShaMap() map[string]bool {
|
2020-08-22 10:22:06 +10:00
|
|
|
commitShaMap := map[string]bool{}
|
2020-08-22 11:05:37 +10:00
|
|
|
for _, commit := range gui.State.Modes.CherryPicking.CherryPickedCommits {
|
2020-08-22 10:22:06 +10:00
|
|
|
commitShaMap[commit.Sha] = true
|
|
|
|
}
|
|
|
|
return commitShaMap
|
|
|
|
}
|
|
|
|
|
2020-09-29 18:36:54 +10:00
|
|
|
func (gui *Gui) commitsListForContext() []*models.Commit {
|
2021-04-04 23:51:59 +10:00
|
|
|
context := gui.currentSideListContext()
|
2020-08-22 11:44:03 +10:00
|
|
|
if context == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// using a switch statement, but we should use polymorphism
|
|
|
|
switch context.GetKey() {
|
|
|
|
case BRANCH_COMMITS_CONTEXT_KEY:
|
|
|
|
return gui.State.Commits
|
|
|
|
case REFLOG_COMMITS_CONTEXT_KEY:
|
|
|
|
return gui.State.FilteredReflogCommits
|
|
|
|
case SUB_COMMITS_CONTEXT_KEY:
|
|
|
|
return gui.State.SubCommits
|
|
|
|
default:
|
|
|
|
gui.Log.Errorf("no commit list for context %s", context.GetKey())
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-22 10:22:06 +10:00
|
|
|
func (gui *Gui) addCommitToCherryPickedCommits(index int) {
|
2020-08-22 11:57:44 +10:00
|
|
|
commitShaMap := gui.cherryPickedCommitShaMap()
|
2020-08-22 11:44:03 +10:00
|
|
|
commitsList := gui.commitsListForContext()
|
|
|
|
commitShaMap[commitsList[index].Sha] = true
|
2020-08-22 10:22:06 +10:00
|
|
|
|
2020-09-29 18:36:54 +10:00
|
|
|
newCommits := []*models.Commit{}
|
2020-08-22 11:44:03 +10:00
|
|
|
for _, commit := range commitsList {
|
2020-08-22 10:22:06 +10:00
|
|
|
if commitShaMap[commit.Sha] {
|
|
|
|
// duplicating just the things we need to put in the rebase TODO list
|
2020-09-29 18:36:54 +10:00
|
|
|
newCommits = append(newCommits, &models.Commit{Name: commit.Name, Sha: commit.Sha})
|
2020-08-22 10:22:06 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:05:37 +10:00
|
|
|
gui.State.Modes.CherryPicking.CherryPickedCommits = newCommits
|
2020-08-22 10:22:06 +10:00
|
|
|
}
|
|
|
|
|
2020-08-22 11:05:37 +10:00
|
|
|
func (gui *Gui) handleCopyCommitRange() error {
|
2020-08-22 10:22:06 +10:00
|
|
|
if ok, err := gui.validateNotInFilterMode(); err != nil || !ok {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:44:03 +10:00
|
|
|
// get currently selected commit, add the sha to state.
|
2021-04-04 23:51:59 +10:00
|
|
|
context := gui.currentSideListContext()
|
2020-08-22 11:44:03 +10:00
|
|
|
if context == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:57:44 +10:00
|
|
|
if err := gui.resetCherryPickingIfNecessary(context); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-08-22 11:44:03 +10:00
|
|
|
|
2020-08-22 15:56:30 +10:00
|
|
|
commitShaMap := gui.cherryPickedCommitShaMap()
|
|
|
|
commitsList := gui.commitsListForContext()
|
|
|
|
selectedLineIdx := context.GetPanelState().GetSelectedLineIdx()
|
|
|
|
|
|
|
|
if selectedLineIdx > len(commitsList)-1 {
|
2020-08-22 11:44:03 +10:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-08-22 10:22:06 +10:00
|
|
|
// find the last commit that is copied that's above our position
|
|
|
|
// if there are none, startIndex = 0
|
|
|
|
startIndex := 0
|
2020-08-22 15:56:30 +10:00
|
|
|
for index, commit := range commitsList[0:selectedLineIdx] {
|
2020-08-22 10:22:06 +10:00
|
|
|
if commitShaMap[commit.Sha] {
|
|
|
|
startIndex = index
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-08-22 15:56:30 +10:00
|
|
|
for index := startIndex; index <= selectedLineIdx; index++ {
|
2020-08-22 10:22:06 +10:00
|
|
|
gui.addCommitToCherryPickedCommits(index)
|
|
|
|
}
|
|
|
|
|
2020-08-22 11:57:44 +10:00
|
|
|
return context.HandleRender()
|
2020-08-22 10:22:06 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
// HandlePasteCommits begins a cherry-pick rebase with the commits the user has copied
|
2020-08-22 11:05:37 +10:00
|
|
|
func (gui *Gui) HandlePasteCommits() error {
|
2020-08-22 10:22:06 +10:00
|
|
|
if ok, err := gui.validateNotInFilterMode(); err != nil || !ok {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return gui.ask(askOpts{
|
2020-10-04 11:00:48 +11:00
|
|
|
title: gui.Tr.CherryPick,
|
|
|
|
prompt: gui.Tr.SureCherryPick,
|
2020-08-22 10:22:06 +10:00
|
|
|
handleConfirm: func() error {
|
2020-10-04 11:00:48 +11:00
|
|
|
return gui.WithWaitingStatus(gui.Tr.CherryPickingStatus, func() error {
|
2021-04-11 19:35:42 +10:00
|
|
|
err := gui.GitCommand.WithSpan(gui.Tr.Spans.CherryPick).CherryPickCommits(gui.State.Modes.CherryPicking.CherryPickedCommits)
|
2020-08-22 10:22:06 +10:00
|
|
|
return gui.handleGenericMergeCommandResult(err)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2020-08-22 11:44:03 +10:00
|
|
|
|
|
|
|
func (gui *Gui) exitCherryPickingMode() error {
|
2021-06-05 15:08:36 +10:00
|
|
|
contextKey := ContextKey(gui.State.Modes.CherryPicking.ContextKey)
|
2020-08-22 11:44:03 +10:00
|
|
|
|
|
|
|
gui.State.Modes.CherryPicking.ContextKey = ""
|
|
|
|
gui.State.Modes.CherryPicking.CherryPickedCommits = nil
|
|
|
|
|
|
|
|
if contextKey == "" {
|
|
|
|
gui.Log.Warn("context key blank when trying to exit cherry picking mode")
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return gui.rerenderContextViewIfPresent(contextKey)
|
|
|
|
}
|
|
|
|
|
2021-04-04 23:51:59 +10:00
|
|
|
func (gui *Gui) rerenderContextViewIfPresent(contextKey ContextKey) error {
|
2020-08-22 11:44:03 +10:00
|
|
|
if contextKey == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2020-09-27 09:37:22 +10:00
|
|
|
context := gui.mustContextForContextKey(contextKey)
|
2020-08-22 11:44:03 +10:00
|
|
|
|
|
|
|
viewName := context.GetViewName()
|
|
|
|
|
|
|
|
view, err := gui.g.View(viewName)
|
|
|
|
if err != nil {
|
2020-09-26 10:23:10 +10:00
|
|
|
gui.Log.Error(err)
|
2020-08-22 11:44:03 +10:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-04-04 23:51:59 +10:00
|
|
|
if ContextKey(view.Context) == contextKey {
|
2020-08-22 11:44:03 +10:00
|
|
|
if err := context.HandleRender(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|