2022-02-06 15:54:26 +11:00
|
|
|
package helpers
|
2022-01-16 14:46:53 +11:00
|
|
|
|
|
|
|
import (
|
2022-11-27 01:00:51 +01:00
|
|
|
"fmt"
|
2022-11-27 01:10:24 +01:00
|
|
|
"regexp"
|
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
2022-11-27 01:00:51 +01:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/config"
|
2023-01-21 11:38:14 +00:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/context"
|
2022-11-27 01:10:24 +01:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
2022-01-16 14:46:53 +11:00
|
|
|
)
|
|
|
|
|
2022-01-31 22:11:34 +11:00
|
|
|
type IWorkingTreeHelper interface {
|
|
|
|
AnyStagedFiles() bool
|
|
|
|
AnyTrackedFiles() bool
|
|
|
|
IsWorkingTreeDirty() bool
|
|
|
|
FileForSubmodule(submodule *models.SubmoduleConfig) *models.File
|
|
|
|
}
|
|
|
|
|
2022-01-16 14:46:53 +11:00
|
|
|
type WorkingTreeHelper struct {
|
2023-01-21 11:38:14 +00:00
|
|
|
c *HelperCommon
|
|
|
|
refHelper *RefsHelper
|
|
|
|
commitsHelper *CommitsHelper
|
|
|
|
gpgHelper *GpgHelper
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
|
2022-11-27 01:00:51 +01:00
|
|
|
func NewWorkingTreeHelper(
|
2023-03-23 12:35:07 +11:00
|
|
|
c *HelperCommon,
|
2022-11-27 01:10:24 +01:00
|
|
|
refHelper *RefsHelper,
|
2023-01-21 11:38:14 +00:00
|
|
|
commitsHelper *CommitsHelper,
|
|
|
|
gpgHelper *GpgHelper,
|
2022-11-27 01:00:51 +01:00
|
|
|
) *WorkingTreeHelper {
|
2022-01-16 14:46:53 +11:00
|
|
|
return &WorkingTreeHelper{
|
2023-01-21 11:38:14 +00:00
|
|
|
c: c,
|
|
|
|
refHelper: refHelper,
|
|
|
|
commitsHelper: commitsHelper,
|
|
|
|
gpgHelper: gpgHelper,
|
2022-01-16 14:46:53 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) AnyStagedFiles() bool {
|
2023-03-23 12:53:18 +11:00
|
|
|
for _, file := range self.c.Model().Files {
|
2022-01-16 14:46:53 +11:00
|
|
|
if file.HasStagedChanges {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) AnyTrackedFiles() bool {
|
2023-03-23 12:53:18 +11:00
|
|
|
for _, file := range self.c.Model().Files {
|
2022-01-16 14:46:53 +11:00
|
|
|
if file.Tracked {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) IsWorkingTreeDirty() bool {
|
|
|
|
return self.AnyStagedFiles() || self.AnyTrackedFiles()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) FileForSubmodule(submodule *models.SubmoduleConfig) *models.File {
|
2023-03-23 12:53:18 +11:00
|
|
|
for _, file := range self.c.Model().Files {
|
2022-01-16 14:46:53 +11:00
|
|
|
if file.IsSubmodule([]*models.SubmoduleConfig{submodule}) {
|
|
|
|
return file
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2022-03-26 15:52:35 +11:00
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) OpenMergeTool() error {
|
2022-03-30 08:48:29 +02:00
|
|
|
return self.c.Confirm(types.ConfirmOpts{
|
2022-03-26 15:52:35 +11:00
|
|
|
Title: self.c.Tr.MergeToolTitle,
|
|
|
|
Prompt: self.c.Tr.MergeToolPrompt,
|
|
|
|
HandleConfirm: func() error {
|
|
|
|
self.c.LogAction(self.c.Tr.Actions.OpenMergeTool)
|
|
|
|
return self.c.RunSubprocessAndRefresh(
|
2023-03-23 12:53:18 +11:00
|
|
|
self.c.Git().WorkingTree.OpenMergeToolCmdObj(),
|
2022-03-26 15:52:35 +11:00
|
|
|
)
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
2022-11-27 01:00:51 +01:00
|
|
|
|
2023-01-21 11:38:14 +00:00
|
|
|
func (self *WorkingTreeHelper) HandleCommitPressWithMessage(initialMessage string) error {
|
2022-11-27 01:00:51 +01:00
|
|
|
if err := self.prepareFilesForCommit(); err != nil {
|
|
|
|
return self.c.Error(err)
|
|
|
|
}
|
|
|
|
|
2023-03-23 12:53:18 +11:00
|
|
|
if len(self.c.Model().Files) == 0 {
|
2022-11-27 01:00:51 +01:00
|
|
|
return self.c.ErrorMsg(self.c.Tr.NoFilesStagedTitle)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !self.AnyStagedFiles() {
|
|
|
|
return self.PromptToStageAllAndRetry(self.HandleCommitPress)
|
|
|
|
}
|
|
|
|
|
2023-01-21 11:38:14 +00:00
|
|
|
return self.commitsHelper.OpenCommitMessagePanel(
|
|
|
|
&OpenCommitMessagePanelOpts{
|
2023-07-22 14:05:42 +10:00
|
|
|
CommitIndex: context.NoCommitIndex,
|
|
|
|
InitialMessage: initialMessage,
|
|
|
|
SummaryTitle: self.c.Tr.CommitSummaryTitle,
|
|
|
|
DescriptionTitle: self.c.Tr.CommitDescriptionTitle,
|
|
|
|
PreserveMessage: true,
|
|
|
|
OnConfirm: self.handleCommit,
|
2023-01-21 11:38:14 +00:00
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
2022-11-27 01:00:51 +01:00
|
|
|
|
2023-07-22 14:05:42 +10:00
|
|
|
func (self *WorkingTreeHelper) handleCommit(summary string, description string) error {
|
|
|
|
cmdObj := self.c.Git().Commit.CommitCmdObj(summary, description)
|
2023-01-21 11:38:14 +00:00
|
|
|
self.c.LogAction(self.c.Tr.Actions.Commit)
|
|
|
|
return self.gpgHelper.WithGpgHandling(cmdObj, self.c.Tr.CommittingStatus, func() error {
|
|
|
|
self.commitsHelper.OnCommitSuccess()
|
|
|
|
return nil
|
|
|
|
})
|
2022-11-27 01:00:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// HandleCommitEditorPress - handle when the user wants to commit changes via
|
|
|
|
// their editor rather than via the popup panel
|
|
|
|
func (self *WorkingTreeHelper) HandleCommitEditorPress() error {
|
2023-03-23 12:53:18 +11:00
|
|
|
if len(self.c.Model().Files) == 0 {
|
2022-11-27 01:00:51 +01:00
|
|
|
return self.c.ErrorMsg(self.c.Tr.NoFilesStagedTitle)
|
|
|
|
}
|
|
|
|
|
|
|
|
if !self.AnyStagedFiles() {
|
|
|
|
return self.PromptToStageAllAndRetry(self.HandleCommitEditorPress)
|
|
|
|
}
|
|
|
|
|
|
|
|
self.c.LogAction(self.c.Tr.Actions.Commit)
|
|
|
|
return self.c.RunSubprocessAndRefresh(
|
2023-03-23 12:53:18 +11:00
|
|
|
self.c.Git().Commit.CommitEditorCmdObj(),
|
2022-11-27 01:00:51 +01:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) HandleWIPCommitPress() error {
|
|
|
|
skipHookPrefix := self.c.UserConfig.Git.SkipHookPrefix
|
|
|
|
if skipHookPrefix == "" {
|
|
|
|
return self.c.ErrorMsg(self.c.Tr.SkipHookPrefixNotConfigured)
|
|
|
|
}
|
|
|
|
|
2023-01-21 11:38:14 +00:00
|
|
|
return self.HandleCommitPressWithMessage(skipHookPrefix)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) HandleCommitPress() error {
|
|
|
|
message := self.c.Contexts().CommitMessage.GetPreservedMessage()
|
|
|
|
|
2023-05-09 05:10:27 +00:00
|
|
|
if message == "" {
|
2023-01-21 11:38:14 +00:00
|
|
|
commitPrefixConfig := self.commitPrefixConfigForRepo()
|
|
|
|
if commitPrefixConfig != nil {
|
|
|
|
prefixPattern := commitPrefixConfig.Pattern
|
|
|
|
prefixReplace := commitPrefixConfig.Replace
|
|
|
|
rgx, err := regexp.Compile(prefixPattern)
|
|
|
|
if err != nil {
|
2023-05-25 21:11:51 +10:00
|
|
|
return self.c.ErrorMsg(fmt.Sprintf("%s: %s", self.c.Tr.CommitPrefixPatternError, err.Error()))
|
2023-01-21 11:38:14 +00:00
|
|
|
}
|
|
|
|
prefix := rgx.ReplaceAllString(self.refHelper.GetCheckedOutRef().Name, prefixReplace)
|
|
|
|
message = prefix
|
|
|
|
}
|
|
|
|
}
|
2022-11-27 01:00:51 +01:00
|
|
|
|
2023-01-21 11:38:14 +00:00
|
|
|
return self.HandleCommitPressWithMessage(message)
|
2022-11-27 01:00:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) PromptToStageAllAndRetry(retry func() error) error {
|
|
|
|
return self.c.Confirm(types.ConfirmOpts{
|
|
|
|
Title: self.c.Tr.NoFilesStagedTitle,
|
|
|
|
Prompt: self.c.Tr.NoFilesStagedPrompt,
|
|
|
|
HandleConfirm: func() error {
|
|
|
|
self.c.LogAction(self.c.Tr.Actions.StageAllFiles)
|
2023-03-23 12:53:18 +11:00
|
|
|
if err := self.c.Git().WorkingTree.StageAll(); err != nil {
|
2022-11-27 01:00:51 +01:00
|
|
|
return self.c.Error(err)
|
|
|
|
}
|
|
|
|
if err := self.syncRefresh(); err != nil {
|
|
|
|
return self.c.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return retry()
|
|
|
|
},
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// for when you need to refetch files before continuing an action. Runs synchronously.
|
|
|
|
func (self *WorkingTreeHelper) syncRefresh() error {
|
|
|
|
return self.c.Refresh(types.RefreshOptions{Mode: types.SYNC, Scope: []types.RefreshableView{types.FILES}})
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) prepareFilesForCommit() error {
|
|
|
|
noStagedFiles := !self.AnyStagedFiles()
|
|
|
|
if noStagedFiles && self.c.UserConfig.Gui.SkipNoStagedFilesWarning {
|
|
|
|
self.c.LogAction(self.c.Tr.Actions.StageAllFiles)
|
2023-03-23 12:53:18 +11:00
|
|
|
err := self.c.Git().WorkingTree.StageAll()
|
2022-11-27 01:00:51 +01:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-11-27 01:10:24 +01:00
|
|
|
return self.syncRefresh()
|
2022-11-27 01:00:51 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *WorkingTreeHelper) commitPrefixConfigForRepo() *config.CommitPrefixConfig {
|
2023-07-28 18:27:14 +10:00
|
|
|
cfg, ok := self.c.UserConfig.Git.CommitPrefixes[self.c.Git().RepoPaths.RepoName()]
|
2022-11-27 01:00:51 +01:00
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return &cfg
|
|
|
|
}
|