1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-10 04:07:18 +02:00
lazygit/pkg/gui/undoing.go

211 lines
7.0 KiB
Go
Raw Normal View History

2020-03-21 10:57:57 +02:00
package gui
import (
2020-11-16 11:38:26 +02:00
"github.com/jesseduffield/lazygit/pkg/commands"
"github.com/jesseduffield/lazygit/pkg/commands/oscommands"
"github.com/jesseduffield/lazygit/pkg/utils"
2020-03-21 10:57:57 +02:00
)
// Quick summary of how this all works:
// when you want to undo or redo, we start from the top of the reflog and work
// down until we've reached the last user-initiated reflog entry that hasn't already been undone
// we then do the reverse of what that reflog describes.
// When we do this, we create a new reflog entry, and tag it as either an undo or redo
// Then, next time we want to undo, we'll use those entries to know which user-initiated
// actions we can skip. E.g. if I do do three things, A, B, and C, and hit undo twice,
// the reflog will read UUCBA, and when I read the first two undos, I know to skip the following
// two user actions, meaning we end up undoing reflog entry C. Redoing works in a similar way.
type ReflogActionKind int
2020-03-21 10:57:57 +02:00
const (
CHECKOUT ReflogActionKind = iota
COMMIT
REBASE
CURRENT_REBASE
2020-03-21 10:57:57 +02:00
)
type reflogAction struct {
kind ReflogActionKind
from string
to string
2020-03-21 10:57:57 +02:00
}
// Here we're going through the reflog and maintaining a counter that represents how many
// undos/redos/user actions we've seen. when we hit a user action we call the callback specifying
// what the counter is up to and the nature of the action.
// If we find ourselves mid-rebase, we just return because undo/redo mid rebase
// requires knowledge of previous TODO file states, which you can't just get from the reflog.
// Though we might support this later, hence the use of the CURRENT_REBASE action kind.
func (gui *Gui) parseReflogForActions(onUserAction func(counter int, action reflogAction) (bool, error)) error {
counter := 0
reflogCommits := gui.State.FilteredReflogCommits
rebaseFinishCommitSha := ""
var action *reflogAction
for reflogCommitIdx, reflogCommit := range reflogCommits {
action = nil
prevCommitSha := ""
if len(reflogCommits)-1 >= reflogCommitIdx+1 {
prevCommitSha = reflogCommits[reflogCommitIdx+1].Sha
}
if rebaseFinishCommitSha == "" {
if ok, _ := utils.FindStringSubmatch(reflogCommit.Name, `^\[lazygit undo\]`); ok {
counter++
} else if ok, _ := utils.FindStringSubmatch(reflogCommit.Name, `^\[lazygit redo\]`); ok {
counter--
} else if ok, _ := utils.FindStringSubmatch(reflogCommit.Name, `^rebase -i \(abort\)|^rebase -i \(finish\)`); ok {
rebaseFinishCommitSha = reflogCommit.Sha
} else if ok, match := utils.FindStringSubmatch(reflogCommit.Name, `^checkout: moving from ([\S]+) to ([\S]+)`); ok {
action = &reflogAction{kind: CHECKOUT, from: match[1], to: match[2]}
} else if ok, _ := utils.FindStringSubmatch(reflogCommit.Name, `^commit|^reset: moving to|^pull`); ok {
action = &reflogAction{kind: COMMIT, from: prevCommitSha, to: reflogCommit.Sha}
} else if ok, _ := utils.FindStringSubmatch(reflogCommit.Name, `^rebase -i \(start\)`); ok {
// if we're here then we must be currently inside an interactive rebase
action = &reflogAction{kind: CURRENT_REBASE, from: prevCommitSha}
}
} else if ok, _ := utils.FindStringSubmatch(reflogCommit.Name, `^rebase -i \(start\)`); ok {
action = &reflogAction{kind: REBASE, from: prevCommitSha, to: rebaseFinishCommitSha}
rebaseFinishCommitSha = ""
}
if action != nil {
if action.kind != CURRENT_REBASE && action.from == action.to {
// if we're going from one place to the same place we'll ignore the action.
continue
}
ok, err := onUserAction(counter, *action)
if ok {
return err
}
counter--
}
2020-03-21 10:57:57 +02:00
}
return nil
2020-03-21 10:57:57 +02:00
}
func (gui *Gui) reflogUndo() error {
undoEnvVars := []string{"GIT_REFLOG_ACTION=[lazygit undo]"}
2020-10-04 02:00:48 +02:00
undoingStatus := gui.Tr.UndoingStatus
2020-11-16 11:38:26 +02:00
if gui.GitCommand.WorkingTreeState() == commands.REBASE_MODE_REBASING {
2020-10-04 02:00:48 +02:00
return gui.createErrorPanel(gui.Tr.LcCantUndoWhileRebasing)
}
2021-04-11 11:35:42 +02:00
span := gui.Tr.Spans.Undo
2021-04-10 08:25:45 +02:00
return gui.parseReflogForActions(func(counter int, action reflogAction) (bool, error) {
if counter != 0 {
2020-03-21 10:57:57 +02:00
return false, nil
}
switch action.kind {
case COMMIT, REBASE:
return true, gui.handleHardResetWithAutoStash(action.from, handleHardResetWithAutoStashOptions{
EnvVars: undoEnvVars,
WaitingStatus: undoingStatus,
2021-04-10 08:25:45 +02:00
span: span,
})
case CHECKOUT:
return true, gui.handleCheckoutRef(action.from, handleCheckoutRefOptions{
EnvVars: undoEnvVars,
WaitingStatus: undoingStatus,
2021-04-10 08:25:45 +02:00
span: span,
})
}
gui.Log.Error("didn't match on the user action when trying to undo")
return true, nil
2020-03-21 10:57:57 +02:00
})
}
func (gui *Gui) reflogRedo() error {
redoEnvVars := []string{"GIT_REFLOG_ACTION=[lazygit redo]"}
2020-10-04 02:00:48 +02:00
redoingStatus := gui.Tr.RedoingStatus
2020-11-16 11:38:26 +02:00
if gui.GitCommand.WorkingTreeState() == commands.REBASE_MODE_REBASING {
2020-10-04 02:00:48 +02:00
return gui.createErrorPanel(gui.Tr.LcCantRedoWhileRebasing)
}
2021-04-11 11:35:42 +02:00
span := gui.Tr.Spans.Redo
2021-04-10 08:25:45 +02:00
return gui.parseReflogForActions(func(counter int, action reflogAction) (bool, error) {
// if we're redoing and the counter is zero, we just return
2020-03-21 10:57:57 +02:00
if counter == 0 {
return true, nil
} else if counter > 1 {
2020-03-21 10:57:57 +02:00
return false, nil
}
switch action.kind {
case COMMIT, REBASE:
return true, gui.handleHardResetWithAutoStash(action.to, handleHardResetWithAutoStashOptions{
EnvVars: redoEnvVars,
WaitingStatus: redoingStatus,
2021-04-10 08:25:45 +02:00
span: span,
})
case CHECKOUT:
return true, gui.handleCheckoutRef(action.to, handleCheckoutRefOptions{
EnvVars: redoEnvVars,
WaitingStatus: redoingStatus,
2021-04-10 08:25:45 +02:00
span: span,
})
2020-03-21 10:57:57 +02:00
}
gui.Log.Error("didn't match on the user action when trying to redo")
return true, nil
})
2020-03-21 10:57:57 +02:00
}
type handleHardResetWithAutoStashOptions struct {
WaitingStatus string
EnvVars []string
2021-04-10 08:25:45 +02:00
span string
2020-03-21 10:57:57 +02:00
}
// only to be used in the undo flow for now
func (gui *Gui) handleHardResetWithAutoStash(commitSha string, options handleHardResetWithAutoStashOptions) error {
2021-04-10 08:25:45 +02:00
gitCommand := gui.GitCommand.WithSpan(options.span)
2020-03-21 10:57:57 +02:00
reset := func() error {
2021-04-10 08:25:45 +02:00
if err := gui.resetToRef(commitSha, "hard", options.span, oscommands.RunCommandOptions{EnvVars: options.EnvVars}); err != nil {
2020-03-28 02:47:54 +02:00
return gui.surfaceError(err)
2020-03-21 10:57:57 +02:00
}
return nil
}
// if we have any modified tracked files we need to ask the user if they want us to stash for them
2021-04-10 08:25:45 +02:00
dirtyWorkingTree := len(gui.trackedFiles()) > 0 || len(gui.stagedFiles()) > 0
2020-03-21 10:57:57 +02:00
if dirtyWorkingTree {
// offer to autostash changes
2020-08-15 08:38:16 +02:00
return gui.ask(askOpts{
2020-10-04 02:00:48 +02:00
title: gui.Tr.AutoStashTitle,
prompt: gui.Tr.AutoStashPrompt,
2020-08-15 08:36:39 +02:00
handleConfirm: func() error {
return gui.WithWaitingStatus(options.WaitingStatus, func() error {
2021-04-10 08:25:45 +02:00
if err := gitCommand.StashSave(gui.Tr.StashPrefix + commitSha); err != nil {
2020-08-15 08:36:39 +02:00
return gui.surfaceError(err)
}
if err := reset(); err != nil {
2020-03-21 10:57:57 +02:00
return err
}
2020-08-15 08:36:39 +02:00
2021-04-10 08:25:45 +02:00
err := gitCommand.StashDo(0, "pop")
if err := gui.refreshSidePanels(refreshOptions{}); err != nil {
return err
}
if err != nil {
2020-08-15 08:36:39 +02:00
return gui.surfaceError(err)
}
return nil
})
},
})
2020-03-21 10:57:57 +02:00
}
return gui.WithWaitingStatus(options.WaitingStatus, func() error {
return reset()
2020-03-21 10:57:57 +02:00
})
}