2022-12-27 07:27:36 +02:00
|
|
|
package components
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2022-12-27 13:52:20 +02:00
|
|
|
"strings"
|
2022-12-27 07:27:36 +02:00
|
|
|
|
|
|
|
"github.com/jesseduffield/gocui"
|
|
|
|
)
|
|
|
|
|
|
|
|
type View struct {
|
|
|
|
// context is prepended to any error messages e.g. 'context: "current view"'
|
|
|
|
context string
|
|
|
|
getView func() *gocui.View
|
2022-12-27 12:35:36 +02:00
|
|
|
t *TestDriver
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// asserts that the view has the expected title
|
|
|
|
func (self *View) Title(expected *matcher) *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
2022-12-27 07:27:36 +02:00
|
|
|
actual := self.getView().Title
|
|
|
|
return expected.context(fmt.Sprintf("%s title", self.context)).test(actual)
|
|
|
|
})
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// asserts that the view has lines matching the given matchers. So if three matchers
|
|
|
|
// are passed, we only check the first three lines of the view.
|
|
|
|
// This method is convenient when you have a list of commits but you only want to
|
|
|
|
// assert on the first couple of commits.
|
|
|
|
func (self *View) TopLines(matchers ...*matcher) *View {
|
2022-12-27 13:52:20 +02:00
|
|
|
if len(matchers) < 1 {
|
|
|
|
self.t.fail("TopLines method requires at least one matcher. If you are trying to assert that there are no lines, use .IsEmpty()")
|
|
|
|
}
|
|
|
|
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
2022-12-27 07:27:36 +02:00
|
|
|
lines := self.getView().BufferLines()
|
|
|
|
return len(lines) >= len(matchers), fmt.Sprintf("unexpected number of lines in view. Expected at least %d, got %d", len(matchers), len(lines))
|
|
|
|
})
|
|
|
|
|
|
|
|
return self.assertLines(matchers...)
|
|
|
|
}
|
|
|
|
|
|
|
|
// asserts that the view has lines matching the given matchers. One matcher must be passed for each line.
|
|
|
|
// If you only care about the top n lines, use the TopLines method instead.
|
|
|
|
func (self *View) Lines(matchers ...*matcher) *View {
|
2022-12-27 13:52:20 +02:00
|
|
|
self.LineCount(len(matchers))
|
2022-12-27 07:27:36 +02:00
|
|
|
|
|
|
|
return self.assertLines(matchers...)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *View) assertLines(matchers ...*matcher) *View {
|
|
|
|
view := self.getView()
|
|
|
|
|
|
|
|
for i, matcher := range matchers {
|
|
|
|
checkIsSelected, matcher := matcher.checkIsSelected()
|
|
|
|
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.matchString(matcher, fmt.Sprintf("Unexpected content in view '%s'.", view.Name()),
|
2022-12-27 07:27:36 +02:00
|
|
|
func() string {
|
|
|
|
return view.BufferLines()[i]
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
if checkIsSelected {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
2022-12-27 07:27:36 +02:00
|
|
|
lineIdx := view.SelectedLineIdx()
|
|
|
|
return lineIdx == i, fmt.Sprintf("Unexpected selected line index in view '%s'. Expected %d, got %d", view.Name(), i, lineIdx)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// asserts on the content of the view i.e. the stuff within the view's frame.
|
|
|
|
func (self *View) Content(matcher *matcher) *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.matchString(matcher, fmt.Sprintf("%s: Unexpected content.", self.context),
|
2022-12-27 07:27:36 +02:00
|
|
|
func() string {
|
|
|
|
return self.getView().Buffer()
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// asserts on the selected line of the view
|
|
|
|
func (self *View) SelectedLine(matcher *matcher) *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.matchString(matcher, fmt.Sprintf("%s: Unexpected selected line.", self.context),
|
2022-12-27 07:27:36 +02:00
|
|
|
func() string {
|
|
|
|
return self.getView().SelectedLine()
|
|
|
|
},
|
|
|
|
)
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// asserts on the index of the selected line. 0 is the first index, representing the line at the top of the view.
|
|
|
|
func (self *View) SelectedLineIdx(expected int) *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
2022-12-27 07:27:36 +02:00
|
|
|
actual := self.getView().SelectedLineIdx()
|
|
|
|
return expected == actual, fmt.Sprintf("%s: Expected selected line index to be %d, got %d", self.context, expected, actual)
|
|
|
|
})
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// focus the view (assumes the view is a side-view that can be focused via a keybinding)
|
|
|
|
func (self *View) Focus() *View {
|
|
|
|
// we can easily change focus by switching to the view's window, but this assumes that the desired view
|
|
|
|
// is at the top of that window. So for now we'll switch to the window then assert that the desired
|
|
|
|
// view is on top (i.e. that it's the current view).
|
|
|
|
// If we want to support other views e.g. the tags view, we'll need to add more logic here.
|
|
|
|
viewName := self.getView().Name()
|
|
|
|
|
|
|
|
// using a map rather than a slice because we might add other views which share a window index later
|
|
|
|
windowIndexMap := map[string]int{
|
|
|
|
"status": 0,
|
|
|
|
"files": 1,
|
|
|
|
"localBranches": 2,
|
|
|
|
"commits": 3,
|
|
|
|
"stash": 4,
|
|
|
|
}
|
|
|
|
|
|
|
|
index, ok := windowIndexMap[viewName]
|
|
|
|
if !ok {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.fail(fmt.Sprintf("Cannot focus view %s: Focus() method not implemented", viewName))
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.press(self.t.keys.Universal.JumpToBlock[index])
|
2022-12-27 07:27:36 +02:00
|
|
|
|
|
|
|
// assert that we land in the expected view
|
|
|
|
self.IsFocused()
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// asserts that the view is focused
|
|
|
|
func (self *View) IsFocused() *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
2022-12-27 07:27:36 +02:00
|
|
|
expected := self.getView().Name()
|
2022-12-27 12:35:36 +02:00
|
|
|
actual := self.t.gui.CurrentContext().GetView().Name()
|
2022-12-27 07:27:36 +02:00
|
|
|
return actual == expected, fmt.Sprintf("%s: Unexpected view focused. Expected %s, got %s", self.context, expected, actual)
|
|
|
|
})
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *View) Press(keyStr string) *View {
|
|
|
|
self.IsFocused()
|
|
|
|
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.press(keyStr)
|
2022-12-27 07:27:36 +02:00
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
// i.e. pressing down arrow
|
|
|
|
func (self *View) SelectNextItem() *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
return self.Press(self.t.keys.Universal.NextItem)
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// i.e. pressing up arrow
|
|
|
|
func (self *View) SelectPreviousItem() *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
return self.Press(self.t.keys.Universal.PrevItem)
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// i.e. pressing space
|
|
|
|
func (self *View) PressPrimaryAction() *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
return self.Press(self.t.keys.Universal.Select)
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// i.e. pressing space
|
|
|
|
func (self *View) PressEnter() *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
return self.Press(self.t.keys.Universal.Confirm)
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// i.e. pressing escape
|
|
|
|
func (self *View) PressEscape() *View {
|
2022-12-27 12:35:36 +02:00
|
|
|
return self.Press(self.t.keys.Universal.Return)
|
2022-12-27 07:27:36 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (self *View) NavigateToListItem(matcher *matcher) *View {
|
|
|
|
self.IsFocused()
|
|
|
|
|
2022-12-27 12:35:36 +02:00
|
|
|
self.t.navigateToListItem(matcher)
|
2022-12-27 07:27:36 +02:00
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
2022-12-27 12:25:11 +02:00
|
|
|
|
2022-12-27 13:52:20 +02:00
|
|
|
// returns true if the view is a list view and it contains no items
|
|
|
|
func (self *View) IsEmpty() *View {
|
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
|
|
|
actual := strings.TrimSpace(self.getView().Buffer())
|
|
|
|
return actual == "", fmt.Sprintf("%s: Unexpected content in view: expected no content. Content: %s", self.context, actual)
|
|
|
|
})
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *View) LineCount(expectedCount int) *View {
|
|
|
|
if expectedCount == 0 {
|
|
|
|
return self.IsEmpty()
|
|
|
|
}
|
|
|
|
|
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
|
|
|
lines := self.getView().BufferLines()
|
|
|
|
return len(lines) == expectedCount, fmt.Sprintf("unexpected number of lines in view. Expected %d, got %d", expectedCount, len(lines))
|
|
|
|
})
|
|
|
|
|
|
|
|
self.t.assertWithRetries(func() (bool, string) {
|
|
|
|
lines := self.getView().BufferLines()
|
|
|
|
|
|
|
|
// if the view has a single blank line (often the case) we want to treat that as having no lines
|
|
|
|
if len(lines) == 1 && expectedCount == 1 {
|
|
|
|
actual := strings.TrimSpace(self.getView().Buffer())
|
2022-12-28 01:29:32 +02:00
|
|
|
return actual != "", "unexpected number of lines in view. Expected 1, got 0"
|
2022-12-27 13:52:20 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return len(lines) == expectedCount, fmt.Sprintf("unexpected number of lines in view. Expected %d, got %d", expectedCount, len(lines))
|
|
|
|
})
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|
|
|
|
|
2022-12-27 12:25:11 +02:00
|
|
|
// for when you want to make some assertion unrelated to the current view
|
|
|
|
// without breaking the method chain
|
|
|
|
func (self *View) Tap(f func()) *View {
|
|
|
|
f()
|
|
|
|
|
|
|
|
return self
|
|
|
|
}
|