1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-04-27 12:32:37 +02:00

Support selecting file range in patch builder

test: add move_range_to_index

test: add toggle_range
This commit is contained in:
Aaron Hoffman 2024-01-24 19:16:30 -06:00 committed by Jesse Duffield
parent 9b2a5f636a
commit 510f9a1ae1
6 changed files with 247 additions and 24 deletions

View File

@ -80,13 +80,15 @@ func (p *PatchBuilder) PatchToApply(reverse bool) string {
} }
func (p *PatchBuilder) addFileWhole(info *fileInfo) { func (p *PatchBuilder) addFileWhole(info *fileInfo) {
info.mode = WHOLE if info.mode != WHOLE {
lineCount := len(strings.Split(info.diff, "\n")) info.mode = WHOLE
// add every line index lineCount := len(strings.Split(info.diff, "\n"))
// TODO: add tests and then use lo.Range to simplify // add every line index
info.includedLineIndices = make([]int, lineCount) // TODO: add tests and then use lo.Range to simplify
for i := 0; i < lineCount; i++ { info.includedLineIndices = make([]int, lineCount)
info.includedLineIndices[i] = i for i := 0; i < lineCount; i++ {
info.includedLineIndices[i] = i
}
} }
} }

View File

@ -1,6 +1,8 @@
package controllers package controllers
import ( import (
"strings"
"github.com/jesseduffield/gocui" "github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/commands/git_commands" "github.com/jesseduffield/lazygit/pkg/commands/git_commands"
"github.com/jesseduffield/lazygit/pkg/commands/models" "github.com/jesseduffield/lazygit/pkg/commands/models"
@ -10,6 +12,7 @@ import (
"github.com/jesseduffield/lazygit/pkg/gui/filetree" "github.com/jesseduffield/lazygit/pkg/gui/filetree"
"github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/gui/types"
"github.com/jesseduffield/lazygit/pkg/utils" "github.com/jesseduffield/lazygit/pkg/utils"
"github.com/samber/lo"
) )
type CommitFilesController struct { type CommitFilesController struct {
@ -76,8 +79,8 @@ func (self *CommitFilesController) GetKeybindings(opts types.KeybindingsOpts) []
}, },
{ {
Key: opts.GetKey(opts.Config.Universal.Select), Key: opts.GetKey(opts.Config.Universal.Select),
Handler: self.withItem(self.toggleForPatch), Handler: self.withItems(self.toggleForPatch),
GetDisabledReason: self.require(self.singleItemSelected()), GetDisabledReason: self.require(self.itemsSelected()),
Description: self.c.Tr.ToggleAddToPatch, Description: self.c.Tr.ToggleAddToPatch,
Tooltip: utils.ResolvePlaceholderString(self.c.Tr.ToggleAddToPatchTooltip, Tooltip: utils.ResolvePlaceholderString(self.c.Tr.ToggleAddToPatchTooltip,
map[string]string{"doc": constants.Links.Docs.CustomPatchDemo}, map[string]string{"doc": constants.Links.Docs.CustomPatchDemo},
@ -240,7 +243,7 @@ func (self *CommitFilesController) openDiffTool(node *filetree.CommitFileNode) e
return err return err
} }
func (self *CommitFilesController) toggleForPatch(node *filetree.CommitFileNode) error { func (self *CommitFilesController) toggleForPatch(selectedNodes []*filetree.CommitFileNode) error {
toggle := func() error { toggle := func() error {
return self.c.WithWaitingStatus(self.c.Tr.UpdatingPatch, func(gocui.Task) error { return self.c.WithWaitingStatus(self.c.Tr.UpdatingPatch, func(gocui.Task) error {
if !self.c.Git().Patch.PatchBuilder.Active() { if !self.c.Git().Patch.PatchBuilder.Active() {
@ -249,21 +252,29 @@ func (self *CommitFilesController) toggleForPatch(node *filetree.CommitFileNode)
} }
} }
// if there is any file that hasn't been fully added we'll fully add everything, selectedNodes = normalisedSelectedCommitFileNodes(selectedNodes)
// otherwise we'll remove everything
adding := node.SomeFile(func(file *models.CommitFile) bool { // Find if any file in the selection is unselected or partially added
return self.c.Git().Patch.PatchBuilder.GetFileStatus(file.Name, self.context().GetRef().RefName()) != patch.WHOLE adding := lo.SomeBy(selectedNodes, func(node *filetree.CommitFileNode) bool {
return node.SomeFile(func(file *models.CommitFile) bool {
fileStatus := self.c.Git().Patch.PatchBuilder.GetFileStatus(file.Name, self.context().GetRef().RefName())
return fileStatus == patch.PART || fileStatus == patch.UNSELECTED
})
}) })
err := node.ForEachFile(func(file *models.CommitFile) error { patchOperationFunction := self.c.Git().Patch.PatchBuilder.RemoveFile
if adding {
return self.c.Git().Patch.PatchBuilder.AddFileWhole(file.Name) if adding {
} else { patchOperationFunction = self.c.Git().Patch.PatchBuilder.AddFileWhole
return self.c.Git().Patch.PatchBuilder.RemoveFile(file.Name) }
for _, node := range selectedNodes {
err := node.ForEachFile(func(file *models.CommitFile) error {
return patchOperationFunction(file.Name)
})
if err != nil {
return self.c.Error(err)
} }
})
if err != nil {
return self.c.Error(err)
} }
if self.c.Git().Patch.PatchBuilder.IsEmpty() { if self.c.Git().Patch.PatchBuilder.IsEmpty() {
@ -290,7 +301,7 @@ func (self *CommitFilesController) toggleForPatch(node *filetree.CommitFileNode)
func (self *CommitFilesController) toggleAllForPatch(_ *filetree.CommitFileNode) error { func (self *CommitFilesController) toggleAllForPatch(_ *filetree.CommitFileNode) error {
root := self.context().CommitFileTreeViewModel.GetRoot() root := self.context().CommitFileTreeViewModel.GetRoot()
return self.toggleForPatch(root) return self.toggleForPatch([]*filetree.CommitFileNode{root})
} }
func (self *CommitFilesController) startPatchBuilder() error { func (self *CommitFilesController) startPatchBuilder() error {
@ -354,3 +365,23 @@ func (self *CommitFilesController) toggleTreeView() error {
return self.c.PostRefreshUpdate(self.context()) return self.c.PostRefreshUpdate(self.context())
} }
// NOTE: these functions are identical to those in files_controller.go (except for types) and
// could also be cleaned up with some generics
func normalisedSelectedCommitFileNodes(selectedNodes []*filetree.CommitFileNode) []*filetree.CommitFileNode {
return lo.Filter(selectedNodes, func(node *filetree.CommitFileNode, _ int) bool {
return !isDescendentOfSelectedCommitFileNodes(node, selectedNodes)
})
}
func isDescendentOfSelectedCommitFileNodes(node *filetree.CommitFileNode, selectedNodes []*filetree.CommitFileNode) bool {
for _, selectedNode := range selectedNodes {
selectedNodePath := selectedNode.GetPath()
nodePath := node.GetPath()
if strings.HasPrefix(nodePath, selectedNodePath) && nodePath != selectedNodePath {
return true
}
}
return false
}

View File

@ -80,7 +80,18 @@ func (self *CommitFileTreeViewModel) GetSelectedItemId() string {
} }
func (self *CommitFileTreeViewModel) GetSelectedItems() ([]*CommitFileNode, int, int) { func (self *CommitFileTreeViewModel) GetSelectedItems() ([]*CommitFileNode, int, int) {
panic("Not implemented") if self.Len() == 0 {
return nil, 0, 0
}
startIdx, endIdx := self.GetSelectionRange()
nodes := []*CommitFileNode{}
for i := startIdx; i <= endIdx; i++ {
nodes = append(nodes, self.Get(i))
}
return nodes, startIdx, endIdx
} }
func (self *CommitFileTreeViewModel) GetSelectedItemIds() ([]string, int, int) { func (self *CommitFileTreeViewModel) GetSelectedItemIds() ([]string, int, int) {

View File

@ -0,0 +1,70 @@
package patch_building
import (
"github.com/jesseduffield/lazygit/pkg/config"
. "github.com/jesseduffield/lazygit/pkg/integration/components"
)
var MoveRangeToIndex = NewIntegrationTest(NewIntegrationTestArgs{
Description: "Apply a custom patch",
ExtraCmdArgs: []string{},
Skip: false,
SetupConfig: func(config *config.AppConfig) {},
SetupRepo: func(shell *Shell) {
shell.CreateFileAndAdd("file1", "first line\n")
shell.Commit("first commit")
shell.UpdateFileAndAdd("file1", "first line\nsecond line\n")
shell.CreateFileAndAdd("file2", "file two content\n")
shell.CreateFileAndAdd("file3", "file three content\n")
shell.Commit("second commit")
},
Run: func(t *TestDriver, keys config.KeybindingConfig) {
t.Views().Commits().
Focus().
Lines(
Contains("second commit").IsSelected(),
Contains("first commit"),
).
PressEnter()
t.Views().CommitFiles().
IsFocused().
Lines(
Contains("M file1").IsSelected(),
Contains("A file2"),
Contains("A file3"),
).
Press(keys.Universal.ToggleRangeSelect).
NavigateToLine(Contains("file2")).
PressPrimaryAction()
t.Views().Information().Content(Contains("Building patch"))
t.Views().PatchBuildingSecondary().Content(Contains("second line"))
t.Views().PatchBuildingSecondary().Content(Contains("file two content"))
t.Common().SelectPatchOption(MatchesRegexp(`Move patch out into index$`))
t.Views().CommitFiles().
IsFocused().
Lines(
Contains("file3").IsSelected(),
).PressEscape()
t.Views().Files().
Focus().
Lines(
Contains("file1").IsSelected(),
Contains("file2"),
)
t.Views().Main().
Content(Contains("second line"))
t.Views().Files().Focus().NavigateToLine(Contains("file2"))
t.Views().Main().
Content(Contains("file two content"))
},
})

View File

@ -0,0 +1,107 @@
package patch_building
import (
"github.com/jesseduffield/lazygit/pkg/config"
. "github.com/jesseduffield/lazygit/pkg/integration/components"
)
var ToggleRange = NewIntegrationTest(NewIntegrationTestArgs{
Description: "Check multi select toggle logic",
ExtraCmdArgs: []string{},
Skip: false,
SetupConfig: func(config *config.AppConfig) {},
SetupRepo: func(shell *Shell) {
shell.CreateDir("dir1")
shell.CreateFileAndAdd("dir1/file1-a", "d2f1 first line\nsecond line\nthird line\n")
shell.CreateFileAndAdd("dir1/file2-a", "d1f2 first line\n")
shell.CreateFileAndAdd("dir1/file3-a", "d1f3 first line\n")
shell.CreateDir("dir2")
shell.CreateFileAndAdd("dir2/file1-b", "d2f1 first line\nsecond line\nthird line\n")
shell.CreateFileAndAdd("dir2/file2-b", "d2f2 first line\n")
shell.CreateFileAndAdd("dir2/file3-b", "d2f3 first line\nsecond line\n")
shell.Commit("first commit")
},
Run: func(t *TestDriver, keys config.KeybindingConfig) {
t.Views().Commits().
Focus().
Lines(
Contains("first commit").IsSelected(),
).
PressEnter()
t.Views().CommitFiles().
IsFocused().
Lines(
Contains("▼ dir1").IsSelected(),
Contains(" A").Contains("file1-a"),
Contains(" A").Contains("file2-a"),
Contains(" A").Contains("file3-a"),
Contains("▼ dir2"),
Contains(" A").Contains("file1-b"),
Contains(" A").Contains("file2-b"),
Contains(" A").Contains("file3-b"),
).
NavigateToLine(Contains("file1-a")).
Press(keys.Universal.ToggleRangeSelect).
NavigateToLine(Contains("file3-a")).
PressPrimaryAction().
Lines(
Contains("▼ dir1"),
Contains(" ●").Contains("file1-a").IsSelected(),
Contains(" ●").Contains("file2-a").IsSelected(),
Contains(" ●").Contains("file3-a").IsSelected(),
Contains("▼ dir2"),
Contains(" A").Contains("file1-b"),
Contains(" A").Contains("file2-b"),
Contains(" A").Contains("file3-b"),
).
PressEscape().
NavigateToLine(Contains("file3-b")).
PressEnter()
t.Views().Main().IsFocused().
NavigateToLine(Contains("second line")).
PressPrimaryAction().
PressEscape()
t.Views().CommitFiles().
IsFocused().
Lines(
Contains("▼ dir1"),
Contains(" ●").Contains("file1-a"),
Contains(" ●").Contains("file2-a"),
Contains(" ●").Contains("file3-a"),
Contains("▼ dir2"),
Contains(" A").Contains("file1-b"),
Contains(" A").Contains("file2-b"),
Contains(" ◐").Contains("file3-b").IsSelected(),
).
NavigateToLine(Contains("dir1")).
Press(keys.Universal.ToggleRangeSelect).
NavigateToLine(Contains("dir2")).
PressPrimaryAction().
Lines(
Contains("▼ dir1").IsSelected(),
Contains(" ●").Contains("file1-a").IsSelected(),
Contains(" ●").Contains("file2-a").IsSelected(),
Contains(" ●").Contains("file3-a").IsSelected(),
Contains("▼ dir2").IsSelected(),
Contains(" ●").Contains("file1-b"),
Contains(" ●").Contains("file2-b"),
Contains(" ●").Contains("file3-b"),
).
PressPrimaryAction().
Lines(
Contains("▼ dir1").IsSelected(),
Contains(" A").Contains("file1-a").IsSelected(),
Contains(" A").Contains("file2-a").IsSelected(),
Contains(" A").Contains("file3-a").IsSelected(),
Contains("▼ dir2").IsSelected(),
Contains(" A").Contains("file1-b"),
Contains(" A").Contains("file2-b"),
Contains(" A").Contains("file3-b"),
)
},
})

View File

@ -194,6 +194,7 @@ var tests = []*components.IntegrationTest{
patch_building.Apply, patch_building.Apply,
patch_building.ApplyInReverse, patch_building.ApplyInReverse,
patch_building.ApplyInReverseWithConflict, patch_building.ApplyInReverseWithConflict,
patch_building.MoveRangeToIndex,
patch_building.MoveToEarlierCommit, patch_building.MoveToEarlierCommit,
patch_building.MoveToEarlierCommitNoKeepEmpty, patch_building.MoveToEarlierCommitNoKeepEmpty,
patch_building.MoveToIndex, patch_building.MoveToIndex,
@ -209,6 +210,7 @@ var tests = []*components.IntegrationTest{
patch_building.SelectAllFiles, patch_building.SelectAllFiles,
patch_building.SpecificSelection, patch_building.SpecificSelection,
patch_building.StartNewPatch, patch_building.StartNewPatch,
patch_building.ToggleRange,
reflog.Checkout, reflog.Checkout,
reflog.CherryPick, reflog.CherryPick,
reflog.DoNotShowBranchMarkersInReflogSubcommits, reflog.DoNotShowBranchMarkersInReflogSubcommits,