summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--go.mod3
-rw-r--r--go.sum12
-rw-r--r--pkg/commands/patch/patch_modifier.go8
-rw-r--r--pkg/commands/patch/patch_parser.go4
-rw-r--r--pkg/gui/gui_driver.go9
-rw-r--r--pkg/integration/components/matcher.go5
-rw-r--r--pkg/integration/components/test_driver.go13
-rw-r--r--pkg/integration/components/test_test.go4
-rw-r--r--pkg/integration/components/viewDriver.go109
-rw-r--r--pkg/integration/components/views.go86
-rw-r--r--pkg/integration/tests/staging/diff_context_change.go81
-rw-r--r--pkg/integration/tests/staging/discard_all_changes.go6
-rw-r--r--pkg/integration/tests/staging/stage_hunks.go88
-rw-r--r--pkg/integration/tests/staging/stage_lines.go34
-rw-r--r--pkg/integration/tests/staging/stage_ranges.go63
-rw-r--r--pkg/integration/types/types.go1
-rw-r--r--vendor/github.com/jesseduffield/gocui/view.go3
-rw-r--r--vendor/modules.txt4
18 files changed, 417 insertions, 116 deletions
diff --git a/go.mod b/go.mod
index 8a24e093e..3522a4caf 100644
--- a/go.mod
+++ b/go.mod
@@ -18,7 +18,7 @@ require (
github.com/integrii/flaggy v1.4.0
github.com/jesseduffield/generics v0.0.0-20220320043834-727e535cbe68
github.com/jesseduffield/go-git/v5 v5.1.2-0.20221018185014-fdd53fef665d
- github.com/jesseduffield/gocui v0.3.1-0.20230219034834-06a1f1e95da5
+ github.com/jesseduffield/gocui v0.3.1-0.20230225001450-38a4deaa7f24
github.com/jesseduffield/kill v0.0.0-20220618033138-bfbe04675d10
github.com/jesseduffield/lazycore v0.0.0-20221012050358-03d2e40243c5
github.com/jesseduffield/minimal/gitignore v0.3.3-0.20211018110810-9cde264e6b1e
@@ -46,7 +46,6 @@ require (
github.com/emirpasic/gods v1.12.0 // indirect
github.com/fatih/color v1.9.0 // indirect
github.com/gdamore/encoding v1.0.0 // indirect
- github.com/gdamore/tcell v1.4.0 // indirect
github.com/go-git/gcfg v1.5.0 // indirect
github.com/go-git/go-billy/v5 v5.0.0 // indirect
github.com/go-logfmt/logfmt v0.5.0 // indirect
diff --git a/go.sum b/go.sum
index d4c892dd5..6af49803a 100644
--- a/go.sum
+++ b/go.sum
@@ -34,11 +34,7 @@ github.com/fsnotify/fsnotify v1.4.7 h1:IXs+QLmnXW2CcXuY+8Mzv/fWEsPGWxqefPtCP5CnV
github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo=
github.com/gdamore/encoding v1.0.0 h1:+7OoQ1Bc6eTm5niUzBa0Ctsh6JbMW6Ra+YNuAtDBdko=
github.com/gdamore/encoding v1.0.0/go.mod h1:alR0ol34c49FCSBLjhosxzcPHQbf2trDkoo5dl+VrEg=
-github.com/gdamore/tcell v1.4.0 h1:vUnHwJRvcPQa3tzi+0QI4U9JINXYJlOz9yiaiPQ2wMU=
-github.com/gdamore/tcell v1.4.0/go.mod h1:vxEiSDZdW3L+Uhjii9c3375IlDmR05bzxY404ZVSMo0=
github.com/gdamore/tcell/v2 v2.4.0/go.mod h1:cTTuF84Dlj/RqmaCIV5p4w8uG1zWdk0SF6oBpwHp4fU=
-github.com/gdamore/tcell/v2 v2.5.4 h1:TGU4tSjD3sCL788vFNeJnTdzpNKIw1H5dgLnJRQVv/k=
-github.com/gdamore/tcell/v2 v2.5.4/go.mod h1:dZgRy5v4iMobMEcWNYBtREnDZAT9DYmfqIkrgEMxLyw=
github.com/gdamore/tcell/v2 v2.6.0 h1:OKbluoP9VYmJwZwq/iLb4BxwKcwGthaa1YNBJIyCySg=
github.com/gdamore/tcell/v2 v2.6.0/go.mod h1:be9omFATkdr0D9qewWW3d+MEvl5dha+Etb5y65J2H8Y=
github.com/gliderlabs/ssh v0.2.2 h1:6zsha5zo/TWhRhwqCD3+EarCAgZ2yN28ipRnGPnwkI0=
@@ -76,8 +72,8 @@ github.com/jesseduffield/generics v0.0.0-20220320043834-727e535cbe68 h1:EQP2Tv8T
github.com/jesseduffield/generics v0.0.0-20220320043834-727e535cbe68/go.mod h1:+LLj9/WUPAP8LqCchs7P+7X0R98HiFujVFANdNaxhGk=
github.com/jesseduffield/go-git/v5 v5.1.2-0.20221018185014-fdd53fef665d h1:bO+OmbreIv91rCe8NmscRwhFSqkDJtzWCPV4Y+SQuXE=
github.com/jesseduffield/go-git/v5 v5.1.2-0.20221018185014-fdd53fef665d/go.mod h1:nGNEErzf+NRznT+N2SWqmHnDnF9aLgANB1CUNEan09o=
-github.com/jesseduffield/gocui v0.3.1-0.20230219034834-06a1f1e95da5 h1:6mrOZa9I1bol14HhVK0tl4w9xvvGKLWPmShPRey1Lxg=
-github.com/jesseduffield/gocui v0.3.1-0.20230219034834-06a1f1e95da5/go.mod h1:znJuCDnF2Ph40YZSlBwdX/4GEofnIoWLGdT4mK5zRAU=
+github.com/jesseduffield/gocui v0.3.1-0.20230225001450-38a4deaa7f24 h1:1uSYSN8np7ym5IjuGi2hHVp/9GQebLbApfz7vudzaSM=
+github.com/jesseduffield/gocui v0.3.1-0.20230225001450-38a4deaa7f24/go.mod h1:znJuCDnF2Ph40YZSlBwdX/4GEofnIoWLGdT4mK5zRAU=
github.com/jesseduffield/kill v0.0.0-20220618033138-bfbe04675d10 h1:jmpr7KpX2+2GRiE91zTgfq49QvgiqB0nbmlwZ8UnOx0=
github.com/jesseduffield/kill v0.0.0-20220618033138-bfbe04675d10/go.mod h1:aA97kHeNA+sj2Hbki0pvLslmE4CbDyhBeSSTUUnOuVo=
github.com/jesseduffield/lazycore v0.0.0-20221012050358-03d2e40243c5 h1:CDuQmfOjAtb1Gms6a1p5L2P8RhbLUq5t8aL7PiQd2uY=
@@ -117,7 +113,6 @@ github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hd
github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE=
github.com/mattn/go-isatty v0.0.14 h1:yVuAays6BHfxijgZPzw+3Zlu5yQgKGP2/hcQbHb7S9Y=
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
-github.com/mattn/go-runewidth v0.0.7/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI=
github.com/mattn/go-runewidth v0.0.10/go.mod h1:RAqKPSqVFrSLVXbA8x7dzmKdmGzieGRCM46jaSJTDAk=
github.com/mattn/go-runewidth v0.0.14 h1:+xnbZSEeDbOIg5/mE6JF0w6n9duR1l3/WmbinWVwUuU=
github.com/mattn/go-runewidth v0.0.14/go.mod h1:Jdepj2loyihRzMpdS35Xk/zdY8IAYHsh153qUoGf23w=
@@ -141,7 +136,6 @@ github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZb
github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4=
github.com/rivo/uniseg v0.1.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
github.com/rivo/uniseg v0.2.0/go.mod h1:J6wj4VEh+S6ZtnVlnTBMWIodfgj8LQOQFoIToxlJtxc=
-github.com/rivo/uniseg v0.4.3 h1:utMvzDsuh3suAEnhH0RdHmoPbU648o6CvXxTx4SBMOw=
github.com/rivo/uniseg v0.4.3/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
github.com/rivo/uniseg v0.4.4 h1:8TfxU8dW6PdqD27gjM8MVNuicgxIjxpm4K7x4jp8sis=
github.com/rivo/uniseg v0.4.4/go.mod h1:FN3SvrM+Zdj16jyLfmOkMNblXMcoc8DfTHruCPUcx88=
@@ -203,7 +197,6 @@ golang.org/x/sys v0.0.0-20190221075227-b4e8571b14e0/go.mod h1:STP8DvDyc/dI5b8T5h
golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY=
golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
-golang.org/x/sys v0.0.0-20190626150813-e07cf5db2756/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
@@ -224,7 +217,6 @@ golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ=
golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk=
golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ=
golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ=
-golang.org/x/text v0.5.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
golang.org/x/text v0.7.0 h1:4BRB4x83lYWy72KwLD/qYDuTu7q9PjSagHvijDw7cLo=
golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8=
golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ=
diff --git a/pkg/commands/patch/patch_modifier.go b/pkg/commands/patch/patch_modifier.go
index 2d060ec18..fe0a896b1 100644
--- a/pkg/commands/patch/patch_modifier.go
+++ b/pkg/commands/patch/patch_modifier.go
@@ -27,7 +27,9 @@ func GetHunksFromDiff(diff string) []*PatchHunk {
var hunkLines []string //nolint:prealloc
pastDiffHeader := false
- for lineIdx, line := range strings.SplitAfter(diff, "\n") {
+ lines := strings.SplitAfter(diff, "\n")
+
+ for lineIdx, line := range lines {
isHunkHeader := strings.HasPrefix(line, "@@ -")
if isHunkHeader {
@@ -44,6 +46,10 @@ func GetHunksFromDiff(diff string) []*PatchHunk {
continue
}
+ if lineIdx == len(lines)-1 && line == "" { // skip the trailing newline
+ continue
+ }
+
hunkLines = append(hunkLines, line)
}
diff --git a/pkg/commands/patch/patch_parser.go b/pkg/commands/patch/patch_parser.go
index 1fd3c107a..3f191f40a 100644
--- a/pkg/commands/patch/patch_parser.go
+++ b/pkg/commands/patch/patch_parser.go
@@ -134,7 +134,8 @@ func coloredString(textStyle style.TextStyle, str string, selected bool, include
}
func parsePatch(patch string) ([]int, []int, []*PatchLine) {
- lines := strings.Split(patch, "\n")
+ // ignore trailing newline.
+ lines := strings.Split(strings.TrimSuffix(patch, "\n"), "\n")
hunkStarts := []int{}
stageableLines := []int{}
pastFirstHunkHeader := false
@@ -179,6 +180,7 @@ func parsePatch(patch string) ([]int, []int, []*PatchLine) {
}
patchLines[index] = &PatchLine{Kind: lineKind, Content: line}
}
+
return hunkStarts, stageableLines, patchLines
}
diff --git a/pkg/gui/gui_driver.go b/pkg/gui/gui_driver.go
index d9735c2fe..35b6a870d 100644
--- a/pkg/gui/gui_driver.go
+++ b/pkg/gui/gui_driver.go
@@ -49,6 +49,15 @@ func (self *GuiDriver) CurrentContext() types.Context {
return self.gui.c.CurrentContext()
}
+func (self *GuiDriver) ContextForView(viewName string) types.Context {
+ context, ok := self.gui.contextForView(viewName)
+ if !ok {
+ return nil
+ }
+
+ return context
+}
+
func (self *GuiDriver) Fail(message string) {
currentView := self.gui.g.CurrentView()
fullMessage := fmt.Sprintf(
diff --git a/pkg/integration/components/matcher.go b/pkg/integration/components/matcher.go
index 85a26bbde..ebec084a8 100644
--- a/pkg/integration/components/matcher.go
+++ b/pkg/integration/components/matcher.go
@@ -61,6 +61,11 @@ func (self *matcher) Contains(target string) *matcher {
return self.appendRule(matcherRule{
name: fmt.Sprintf("contains '%s'", target),
testFn: func(value string) (bool, string) {
+ // everything contains the empty string so we unconditionally return true here
+ if target == "" {
+ return true, ""
+ }
+
return strings.Contains(value, target), fmt.Sprintf("Expected '%s' to be found in '%s'", target, value)
},
})
diff --git a/pkg/integration/components/test_driver.go b/pkg/integration/components/test_driver.go
index 43618f23c..694a59dbf 100644
--- a/pkg/integration/components/test_driver.go
+++ b/pkg/integration/components/test_driver.go
@@ -7,7 +7,6 @@ import (
"github.com/atotto/clipboard"
"github.com/jesseduffield/lazygit/pkg/config"
- "github.com/jesseduffield/lazygit/pkg/gui/types"
integrationTypes "github.com/jesseduffield/lazygit/pkg/integration/types"
)
@@ -84,9 +83,7 @@ func (self *TestDriver) Shell() *Shell {
// in the current page and failing that, jump to the top of the view and iterate through all of it,
// looking for the item.
func (self *TestDriver) navigateToListItem(matcher *matcher) {
- self.inListContext()
-
- currentContext := self.gui.CurrentContext().(types.IListContext)
+ currentContext := self.gui.CurrentContext()
view := currentContext.GetView()
@@ -133,14 +130,6 @@ func (self *TestDriver) navigateToListItem(matcher *matcher) {
}
}
-func (self *TestDriver) inListContext() {
- self.assertWithRetries(func() (bool, string) {
- currentContext := self.gui.CurrentContext()
- _, ok := currentContext.(types.IListContext)
- return ok, fmt.Sprintf("Expected current context to be a list context, but got %s", currentContext.GetKey())
- })
-}
-
// for making assertions on lazygit views
func (self *TestDriver) Views() *Views {
return &Views{t: self}
diff --git a/pkg/integration/components/test_test.go b/pkg/integration/components/test_test.go
index 08823f525..c40fc3368 100644
--- a/pkg/integration/components/test_test.go
+++ b/pkg/integration/components/test_test.go
@@ -30,6 +30,10 @@ func (self *fakeGuiDriver) CurrentContext() types.Context {
return nil
}
+func (self *fakeGuiDriver) ContextForView(viewName string) types.Context {
+ return nil
+}
+
func (self *fakeGuiDriver) Fail(message string) {
self.failureMessage = message
}
diff --git a/pkg/integration/components/viewDriver.go b/pkg/integration/components/viewDriver.go
index 04c58fc81..3f0b9726b 100644
--- a/pkg/integration/components/viewDriver.go
+++ b/pkg/integration/components/viewDriver.go
@@ -10,9 +10,10 @@ import (
type ViewDriver struct {
// context is prepended to any error messages e.g. 'context: "current view"'
- context string
- getView func() *gocui.View
- t *TestDriver
+ context string
+ getView func() *gocui.View
+ t *TestDriver
+ getSelectedLinesFn func() ([]string, error)
}
// asserts that the view has the expected title
@@ -50,6 +51,74 @@ func (self *ViewDriver) Lines(matchers ...*matcher) *ViewDriver {
return self.assertLines(matchers...)
}
+func (self *ViewDriver) getSelectedLines() ([]string, error) {
+ if self.getSelectedLinesFn == nil {
+ view := self.t.gui.View(self.getView().Name())
+
+ return []string{view.SelectedLine()}, nil
+ }
+
+ return self.getSelectedLinesFn()
+}
+
+func (self *ViewDriver) SelectedLines(matchers ...*matcher) *ViewDriver {
+ self.t.assertWithRetries(func() (bool, string) {
+ selectedLines, err := self.getSelectedLines()
+ if err != nil {
+ return false, err.Error()
+ }
+
+ selectedContent := strings.Join(selectedLines, "\n")
+ expectedContent := expectedContentFromMatchers(matchers)
+
+ if len(selectedLines) != len(matchers) {
+ return false, fmt.Sprintf("Expected the following to be selected:\n-----\n%s\n-----\nBut got:\n-----\n%s\n-----", expectedContent, selectedContent)
+ }
+
+ for i, line := range selectedLines {
+ ok, message := matchers[i].test(line)
+ if !ok {
+ return false, fmt.Sprintf("Error: %s. Expected the following to be selected:\n-----\n%s\n-----\nBut got:\n-----\n%s\n-----", message, expectedContent, selectedContent)
+ }
+ }
+
+ return true, ""
+ })
+
+ return self
+}
+
+func (self *ViewDriver) ContainsLines(matchers ...*matcher) *ViewDriver {
+ self.t.assertWithRetries(func() (bool, string) {
+ content := self.getView().Buffer()
+ lines := strings.Split(content, "\n")
+
+ for i := 0; i < len(lines)-len(matchers)+1; i++ {
+ matches := true
+ for j, matcher := range matchers {
+ ok, _ := matcher.test(lines[i+j])
+ if !ok {
+ matches = false
+ break
+ }
+ }
+ if matches {
+ return true, ""
+ }
+ }
+
+ expectedContent := expectedContentFromMatchers(matchers)
+
+ return false, fmt.Sprintf(
+ "Expected the following to be contained in the staging panel:\n-----\n%s\n-----\nBut got:\n-----\n%s\n-----",
+ expectedContent,
+ content,
+ )
+ })
+
+ return self
+}
+
func (self *ViewDriver) assertLines(matchers ...*matcher) *ViewDriver {
view := self.getView()
@@ -86,9 +155,35 @@ func (self *ViewDriver) Content(matcher *matcher) *ViewDriver {
// asserts on the selected line of the view
func (self *ViewDriver) SelectedLine(matcher *matcher) *ViewDriver {
+ self.t.assertWithRetries(func() (bool, string) {
+ selectedLines, err := self.getSelectedLines()
+ if err != nil {
+ return false, err.Error()
+ }
+
+ if len(selectedLines) == 0 {
+ return false, "No line selected. Expected exactly one line to be selected"
+ } else if len(selectedLines) > 1 {
+ return false, fmt.Sprintf(
+ "Multiple lines selected. Expected only a single line to be selected. Selected lines:\n---\n%s\n---\n\nExpected line: %s",
+ strings.Join(selectedLines, "\n"),
+ matcher.name(),
+ )
+ }
+
+ value := selectedLines[0]
+ return matcher.context(fmt.Sprintf("%s: Unexpected selected line.", self.context)).test(value)
+ })
+
self.t.matchString(matcher, fmt.Sprintf("%s: Unexpected selected line.", self.context),
func() string {
- return self.getView().SelectedLine()
+ selectedLines, err := self.getSelectedLines()
+ if err != nil {
+ self.t.gui.Fail(err.Error())
+ return "<failed to obtain selected line>"
+ }
+
+ return selectedLines[0]
},
)
@@ -253,3 +348,9 @@ func (self *ViewDriver) Tap(f func()) *ViewDriver {
return self
}
+
+func expectedContentFromMatchers(matchers []*matcher) string {
+ return strings.Join(lo.Map(matchers, func(matcher *matcher, _ int) string {
+ return matcher.name()
+ }), "\n")
+}
diff --git a/pkg/integration/components/views.go b/pkg/integration/components/views.go
index ea5257602..1061d6262 100644
--- a/pkg/integration/components/views.go
+++ b/pkg/integration/components/views.go
@@ -2,8 +2,11 @@ package components
import (
"fmt"
+ "strings"
+ "github.com/go-errors/errors"
"github.com/jesseduffield/gocui"
+ "github.com/jesseduffield/lazygit/pkg/gui/context"
)
type Views struct {
@@ -36,98 +39,129 @@ func (self *Views) Secondary() *ViewDriver {
}
}
-func (self *Views) byName(viewName string) *ViewDriver {
+func (self *Views) regularView(viewName string) *ViewDriver {
+ return self.newStaticViewDriver(viewName, nil)
+}
+
+func (self *Views) patchExplorerViewByName(viewName string) *ViewDriver {
+ return self.newStaticViewDriver(viewName, func() ([]string, error) {
+ ctx := self.t.gui.ContextForView(viewName).(*context.PatchExplorerContext)
+ state := ctx.GetState()
+ if state == nil {
+ return nil, errors.New("Expected patch explorer to be activated")
+ }
+ selectedContent := state.PlainRenderSelected()
+ // the above method returns a string with a trailing newline so we need to remove that before splitting
+ selectedLines := strings.Split(strings.TrimSuffix(selectedContent, "\n"), "\n")
+ return selectedLines, nil
+ })
+}
+
+// 'static' because it'll always refer to the same view, as opposed to the 'main' view which could actually be
+// one of several views, or the 'current' view which depends on focus.
+func (self *Views) newStaticViewDriver(viewName string, getSelectedLinesFn func() ([]string, error)) *ViewDriver {
return &ViewDriver{
- context: fmt.Sprintf("%s view", viewName),
- getView: func() *gocui.View { return self.t.gui.View(viewName) },
- t: self.t,
+ context: fmt.Sprintf("%s view", viewName),
+ getView: func() *gocui.View { return self.t.gui.View(viewName) },
+ getSelectedLinesFn: getSelectedLinesFn,
+ t: self.t,
}
}
func (self *Views) Commits() *ViewDriver {
- return self.byName("commits")
+ return self.regularView("commits")
}
func (self *Views) Files() *ViewDriver {
- return self.byName("files")
+ return self.regularView("files")
}
func (self *Views) Status() *ViewDriver {
- return self.byName("status")
+ return self.regularView("status")
}
func (self *Views) Submodules() *ViewDriver {
- return self.byName("submodules")
+ return self.regularView("submodules")
}
func (self *Views) Information() *ViewDriver {
- return self.byName("information")
+ return self.regularView("information")
}
func (self *Views) AppStatus() *ViewDriver {
- return self.byName("appStatus")
+ return self.regularView("appStatus")
}
func (self *Views) Branches() *ViewDriver {
- return self.byName("localBranches")
+ return self.regularView("localBranches")
}
func (self *Views) Remotes() *ViewDriver {
- return self.byName("remotes")
+ return self.regularView("remotes")
}
func (self *Views) RemoteBranches() *ViewDriver {
- return self.byName("remoteBranches")
+ return self.regularView("remoteBranches")
}
func (self *Views) Tags() *ViewDriver {
- return self.byName("tags")
+ return self.regularView("tags")
}
func (self *Views) ReflogCommits() *ViewDriver {
- return self.byName("reflogCommits")
+ return self.regularView("reflogCommits")
}
func (self *Views) SubCommits() *ViewDriver {
- return self.byName("subCommits")
+ return self.regularView("subCommits")
}
func (self *Views) CommitFiles() *ViewDriver {
- return self.byName("commitFiles")
+ return self.regularView("commitFiles")
}
func (self *Views) Stash() *ViewDriver {
- return self.byName("stash")
+ return self.regularView("stash")
}
func (self *Views) Staging() *ViewDriver {
- return self.byName("staging")
+ return self.patchExplorerViewByName("staging")
}
func (self *Views) StagingSecondary() *ViewDriver {
- return self.byName("stagingSecondary")
+ return self.patchExplorerViewByName("stagingSecondary")
+}
+
+func (self *Views) PatchBuilding() *ViewDriver {
+ return self.patchExplorerViewByName("patchBuilding")
+}
+
+func (self *Views) PatchBuildingSecondary() *ViewDriver {
+ // this is not a patch explorer view because you can't actually focus it: it
+ // just renders content
+ return self.regularView("patchBuildingSecondary")
}
func (self *Views) Menu() *ViewDriver {
- return self.byName("menu")
+ return self.regularView("menu")
}
func (self *Views) Confirmation() *ViewDriver {
- return self.byName("confirmation")
+ return self.regularView("confirmation")
}
func (self *Views) CommitMessage() *ViewDriver {
- return self.byName("commitMessage")
+ return self.regularView("commitMessage")
}
func (self *Views) Suggestions() *ViewDriver {
- return self.byName("suggestions")
+ return self.regularView("suggestions")
}
func (self *Views) MergeConflicts() *ViewDriver {
- return self.byName("mergeConflicts")
+ return self.regularView("mergeConflicts")
}
func (self *Views) Search() *ViewDriver {
- return self.byName("search")
+ return self.regularView("search")
}
diff --git a/pkg/integration/tests/staging/diff_context_change.go b/pkg/integration/tests/staging/diff_context_change.go
index 8ddf3d37f..a88b99848 100644
--- a/pkg/integration/tests/staging/diff_context_change.go
+++ b/pkg/integration/tests/staging/diff_context_change.go
@@ -50,31 +50,80 @@ var DiffContextChange = NewIntegrationTest(NewIntegrationTestArgs{
t.Views().Staging().
IsFocused().
- Content(Contains("@@ -1,6 +1,6 @@").DoesNotContain(" 7a")).
- SelectedLine(Contains("-3a")).
+ Press(keys.Main.ToggleSelectHunk).
+ SelectedLines(
+ Contains(`@@ -1,6 +1,6 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ Contains(` 6a`),
+ ).
Press(keys.Universal.IncreaseContextInDiffView).
- // still on the same line
- SelectedLine(Contains("-3a")).
- // '7a' is now visible
- Content(Contains("@@ -1,7 +1,7 @@").Contains(" 7a")).
+ SelectedLines(
+ Contains(`@@ -1,7 +1,7 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ Contains(` 6a`),
+ Contains(` 7a`),
+ ).
Press(keys.Universal.DecreaseContextInDiffView).
- SelectedLine(Contains("-3a")).
- Content(Contains("@@ -1,6 +1,6 @@").DoesNotContain(" 7a")).
+ SelectedLines(
+ Contains(`@@ -1,6 +1,6 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ Contains(` 6a`),
+ ).
Press(keys.Universal.DecreaseContextInDiffView).
- SelectedLine(Contains("-3a")).
- Content(Contains("@@ -1,5 +1,5 @@").DoesNotContain(" 6a")).
+ SelectedLines(
+ Contains(`@@ -1,5 +1,5 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ ).
Press(keys.Universal.DecreaseContextInDiffView).
- // arguably we should still be on -3a, but at the moment the logic puts us on on +3b
- SelectedLine(Contains("+3b")).
- Content(Contains("@@ -2,3 +2,3 @@").DoesNotContain(" 5a")).
+ SelectedLines(
+ Contains(`@@ -2,3 +2,3 @@`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ ).
PressPrimaryAction().
- Content(DoesNotContain("+3b")).
Press(keys.Universal.TogglePanel)
t.Views().StagingSecondary().
IsFocused().
- Content(Contains("@@ -3,2 +3,3 @@\n 3a\n+3b\n 4a")).
+ Press(keys.Main.ToggleSelectHunk).
+ SelectedLines(
+ Contains(`@@ -2,3 +2,3 @@`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ ).
Press(keys.Universal.IncreaseContextInDiffView).
- Content(Contains("@@ -2,4 +2,5 @@\n 2a\n 3a\n+3b\n 4a\n 5a"))
+ SelectedLines(
+ Contains(`@@ -1,5 +1,5 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ )
},
})
diff --git a/pkg/integration/tests/staging/discard_all_changes.go b/pkg/integration/tests/staging/discard_all_changes.go
index 7d27e7ec2..48cd0b62d 100644
--- a/pkg/integration/tests/staging/discard_all_changes.go
+++ b/pkg/integration/tests/staging/discard_all_changes.go
@@ -29,13 +29,13 @@ var DiscardAllChanges = NewIntegrationTest(NewIntegrationTestArgs{
t.Views().Staging().
IsFocused().
- SelectedLine(Contains("+three")).
+ SelectedLines(Contains("+three")).
// discard the line
Press(keys.Universal.Remove).
Tap(func() {
t.Actions().ConfirmDiscardLines()
}).
- SelectedLine(Contains("+four")).
+ SelectedLines(Contains("+four")).
// discard the other line
Press(keys.Universal.Remove).
Tap(func() {
@@ -49,6 +49,6 @@ var DiscardAllChanges = NewIntegrationTest(NewIntegrationTestArgs{
}).
// assert we are still in the staging panel, but now looking at the changes of the other file
IsFocused().
- SelectedLine(Contains("+3"))
+ SelectedLines(Contains("+3"))
},
})
diff --git a/pkg/integration/tests/staging/stage_hunks.go b/pkg/integration/tests/staging/stage_hunks.go
index b79819108..c2f85661a 100644
--- a/pkg/integration/tests/staging/stage_hunks.go
+++ b/pkg/integration/tests/staging/stage_hunks.go
@@ -50,42 +50,108 @@ var StageHunks = NewIntegrationTest(NewIntegrationTestArgs{
t.Views().Staging().
IsFocused().
- SelectedLine(Contains("-3a")).
+ SelectedLines(
+ Contains("-3a"),
+ ).
Press(keys.Universal.NextBlock).
- SelectedLine(Contains("-13a")).
+ SelectedLines(
+ Contains("-13a"),
+ ).
Press(keys.Main.ToggleSelectHunk).
+ SelectedLines(
+ Contains("@@ -10,6 +10,6 @@"),
+ Contains(" 10a"),
+ Contains(" 11a"),
+ Contains(" 12a"),
+ Contains("-13a"),
+ Contains("+13b"),
+ Contains(" 14a"),
+ Contains(" 15a"),
+ Contains(`\ No newline at end of file`),
+ ).
// when in hunk mode, pressing up/down moves us up/down by a hunk
SelectPreviousItem().
- SelectedLine(Contains("-3a")).
+ SelectedLines(
+ Contains(`@@ -1,6 +1,6 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ Contains(` 6a`),
+ ).
SelectNextItem().
- SelectedLine(Contains("-13a")).
+ SelectedLines(
+ Contains("@@ -10,6 +10,6 @@"),
+ Contains(" 10a"),
+ Contains(" 11a"),
+ Contains(" 12a"),
+ Contains("-13a"),
+ Contains("+13b"),
+ Contains(" 14a"),
+ Contains(" 15a"),
+ Contains(`\ No newline at end of file`),
+ ).
// stage the second hunk
PressPrimaryAction().
- Content(Contains("-3a\n+3b")).
+ ContainsLines(
+ Contains("-3a"),
+ Contains("+3b"),
+ ).
Tap(func() {
t.Views().StagingSecondary().
- Content(Contains("-13a\n+13b"))
+ ContainsLines(
+ Contains("-13a"),
+ Contains("+13b"),
+ )
}).
Press(keys.Universal.TogglePanel)
t.Views().StagingSecondary().
IsFocused().
- SelectedLine(Contains("-13a")).
// after toggling panel, we're back to only having selected a single line
+ SelectedLines(
+ Contains("-13a"),
+ ).
PressPrimaryAction().
- SelectedLine(Contains("+13b")).
+ SelectedLines(
+ Contains("+13b"),
+ ).
PressPrimaryAction().
IsEmpty()
t.Views().Staging().
IsFocused().
- SelectedLine(Contains("-3a")).
+ SelectedLines(
+ Contains("-3a"),
+ ).
Press(keys.Main.ToggleSelectHunk).
+ SelectedLines(
+ Contains(`@@ -1,6 +1,6 @@`),
+ Contains(` 1a`),
+ Contains(` 2a`),
+ Contains(`-3a`),
+ Contains(`+3b`),
+ Contains(` 4a`),
+ Contains(` 5a`),
+ Contains(` 6a`),
+ ).
Press(keys.Universal.Remove).
Tap(func() {
t.Actions().ConfirmDiscardLines()
}).
- SelectedLine(Contains("-13a")).
- Content(DoesNotContain("-3a").DoesNotContain("+3b"))
+ Content(DoesNotContain("-3a").DoesNotContain("+3b")).
+ SelectedLines(
+ Contains("@@ -10,6 +10,6 @@"),
+ Contains(" 10a"),
+ Contains(" 11a"),
+ Contains(" 12a"),
+ Contains("-13a"),
+ Contains("+13b"),
+ Contains(" 14a"),
+ Contains(" 15a"),
+ Contains(`\ No newline at end of file`),
+ )
},
})
diff --git a/pkg/integration/tests/staging/stage_lines.go b/pkg/integration/tests/staging/stage_lines.go
index f1777fb36..f517440b8 100644
--- a/pkg/integration/tests/staging/stage_lines.go
+++ b/pkg/integration/tests/staging/stage_lines.go
@@ -26,16 +26,18 @@ var StageLines = NewIntegrationTest(NewIntegrationTestArgs{
t.Views().Staging().
IsFocused().
- SelectedLine(Contains("+three")).
+ SelectedLines(Contains("+three")).
// stage 'three'
PressPrimaryAction().
// 'three' moves over to the staging secondary panel