1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-04-21 12:16:54 +02:00
lazygit/pkg/utils/lines_test.go
Stefan Haller 1f2cb35cc9 Refactor: move wrapMessageToWidth to utils/lines.go
to make it more generally usable by clients other than ConfirmationHelper, which
we will do later in this branch. Rename it to WrapViewLinesToWidth while we're
at it.

Add tests; in particular, add a sanity check that we wrap lines the same way as
gocui does. The tests that are added here are the same ones as in gocui for its
lineWrap function, but we'll extend them a bit in later commits in this branch.
2024-12-23 12:24:09 +01:00

352 lines
6.7 KiB
Go

package utils
import (
"bufio"
"strings"
"testing"
"github.com/jesseduffield/gocui"
"github.com/stretchr/testify/assert"
)
// TestSplitLines is a function.
func TestSplitLines(t *testing.T) {
type scenario struct {
multilineString string
expected []string
}
scenarios := []scenario{
{
"",
[]string{},
},
{
"\n",
[]string{},
},
{
"hello world !\nhello universe !\n",
[]string{
"hello world !",
"hello universe !",
},
},
}
for _, s := range scenarios {
assert.EqualValues(t, s.expected, SplitLines(s.multilineString))
}
}
func TestSplitNul(t *testing.T) {
type scenario struct {
multilineString string
expected []string
}
scenarios := []scenario{
{
"",
[]string{},
},
{
"\x00",
[]string{
"",
},
},
{
"hello world !\x00hello universe !\x00",
[]string{
"hello world !",
"hello universe !",
},
},
}
for _, s := range scenarios {
assert.EqualValues(t, s.expected, SplitNul(s.multilineString))
}
}
// TestNormalizeLinefeeds is a function.
func TestNormalizeLinefeeds(t *testing.T) {
type scenario struct {
byteArray []byte
expected []byte
}
scenarios := []scenario{
{
// \r\n
[]byte{97, 115, 100, 102, 13, 10},
[]byte{97, 115, 100, 102, 10},
},
{
// bash\r\nblah
[]byte{97, 115, 100, 102, 13, 10, 97, 115, 100, 102},
[]byte{97, 115, 100, 102, 10, 97, 115, 100, 102},
},
{
// \r
[]byte{97, 115, 100, 102, 13},
[]byte{97, 115, 100, 102},
},
{
// \n
[]byte{97, 115, 100, 102, 10},
[]byte{97, 115, 100, 102, 10},
},
}
for _, s := range scenarios {
assert.EqualValues(t, string(s.expected), NormalizeLinefeeds(string(s.byteArray)))
}
}
func TestScanLinesAndTruncateWhenLongerThanBuffer(t *testing.T) {
type scenario struct {
input string
expectedLines []string
}
scenarios := []scenario{
{
"",
[]string{},
},
{
"\n",
[]string{""},
},
{
"abc",
[]string{"abc"},
},
{
"abc\ndef",
[]string{"abc", "def"},
},
{
"abc\n\ndef",
[]string{"abc", "", "def"},
},
{
"abc\r\ndef\r",
[]string{"abc", "def"},
},
{
"abcdef",
[]string{"abcde"},
},
{
"abcdef\n",
[]string{"abcde"},
},
{
"abcdef\nghijkl\nx",
[]string{"abcde", "ghijk", "x"},
},
{
"abc\ndefghijklmnopqrstuvw\nx",
[]string{"abc", "defgh", "x"},
},
}
for _, s := range scenarios {
scanner := bufio.NewScanner(strings.NewReader(s.input))
scanner.Buffer(make([]byte, 5), 5)
scanner.Split(ScanLinesAndTruncateWhenLongerThanBuffer(5))
result := []string{}
for scanner.Scan() {
result = append(result, scanner.Text())
}
assert.NoError(t, scanner.Err())
assert.EqualValues(t, s.expectedLines, result)
}
}
func TestWrapViewLinesToWidth(t *testing.T) {
tests := []struct {
name string
wrap bool
text string
width int
expectedWrappedLines []string
}{
{
name: "Wrap on space",
wrap: true,
text: "Hello World",
width: 5,
expectedWrappedLines: []string{
"Hello",
"World",
},
},
{
name: "Wrap on hyphen",
wrap: true,
text: "Hello-World",
width: 6,
expectedWrappedLines: []string{
"Hello-",
"World",
},
},
{
name: "Wrap on hyphen 2",
wrap: true,
text: "Blah Hello-World",
width: 12,
expectedWrappedLines: []string{
"Blah Hello-",
"World",
},
},
{
name: "Wrap on hyphen 3",
wrap: true,
text: "Blah Hello-World",
width: 11,
expectedWrappedLines: []string{
"Blah Hello-",
"World",
},
},
{
name: "Wrap on hyphen 4",
wrap: true,
text: "Blah Hello-World",
width: 10,
expectedWrappedLines: []string{
"Blah Hello",
"-World",
},
},
{
name: "Wrap on space 2",
wrap: true,
text: "Blah Hello World",
width: 10,
expectedWrappedLines: []string{
"Blah Hello",
"World",
},
},
{
name: "Wrap on space with more words",
wrap: true,
text: "Longer word here",
width: 10,
expectedWrappedLines: []string{
"Longer",
"word here",
},
},
{
name: "Split word that's too long",
wrap: true,
text: "ThisWordIsWayTooLong",
width: 10,
expectedWrappedLines: []string{
"ThisWordIs",
"WayTooLong",
},
},
{
name: "Split word that's too long over multiple lines",
wrap: true,
text: "ThisWordIsWayTooLong",
width: 5,
expectedWrappedLines: []string{
"ThisW",
"ordIs",
"WayTo",
"oLong",
},
},
{
name: "Lots of hyphens",
wrap: true,
text: "one-two-three-four-five",
width: 8,
expectedWrappedLines: []string{
"one-two-",
"three-",
"four-",
"five",
},
},
{
name: "Several lines using all the available width",
wrap: true,
text: "aaa bb cc ddd-ee ff",
width: 5,
expectedWrappedLines: []string{
"aaa",
"bb cc",
"ddd-",
"ee ff",
},
},
{
name: "Several lines using all the available width, with multi-cell runes",
wrap: true,
text: "🐤🐤🐤 🐝🐝 🙉🙉 🦊🦊🦊-🐬🐬 🦢🦢",
width: 9,
expectedWrappedLines: []string{
"🐤🐤🐤",
"🐝🐝 🙉🙉",
"🦊🦊🦊-",
"🐬🐬 🦢🦢",
},
},
{
name: "Space in last column",
wrap: true,
text: "hello world",
width: 6,
expectedWrappedLines: []string{
"hello",
"world",
},
},
{
name: "Hyphen in last column",
wrap: true,
text: "hello-world",
width: 6,
expectedWrappedLines: []string{
"hello-",
"world",
},
},
{
name: "English text",
wrap: true,
text: "+The sea reach of the Thames stretched before us like the bedinnind of an interminable waterway. In the offind the sea and the sky were welded todether without a joint, and in the luminous space the tanned sails of the bardes drifting blah blah",
width: 81,
expectedWrappedLines: []string{
"+The sea reach of the Thames stretched before us like the bedinnind of an",
"interminable waterway. In the offind the sea and the sky were welded todether",
"without a joint, and in the luminous space the tanned sails of the bardes",
"drifting blah blah",
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
wrappedLines := WrapViewLinesToWidth(tt.wrap, tt.text, tt.width)
assert.Equal(t, tt.expectedWrappedLines, wrappedLines)
// As a sanity check, also test that gocui's line wrapping behaves the same way
view := gocui.NewView("", 0, 0, tt.width+1, 1000, gocui.OutputNormal)
assert.Equal(t, tt.width, view.InnerWidth())
view.Wrap = tt.wrap
view.SetContent(tt.text)
assert.Equal(t, wrappedLines, view.ViewBufferLines())
})
}
}