1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-01-22 05:29:44 +02:00

combining nodes when only one child exists

This commit is contained in:
Jesse Duffield 2021-03-14 19:38:59 +11:00
parent 77a7619690
commit 79079b54ea
4 changed files with 173 additions and 13 deletions

View File

@ -1,6 +1,7 @@
package models package models
import ( import (
"fmt"
"sort" "sort"
) )
@ -106,11 +107,11 @@ func (s *StatusLineNode) Flatten() []*StatusLineNode {
return arr return arr
} }
func (s *StatusLineNode) SortTree() { func (s *StatusLineNode) Sort() {
s.sortChildren() s.sortChildren()
for _, child := range s.Children { for _, child := range s.Children {
child.SortTree() child.Sort()
} }
} }
@ -149,3 +150,35 @@ func (s *StatusLineNode) GetIsTracked() bool {
func (s *StatusLineNode) GetPath() string { func (s *StatusLineNode) GetPath() string {
return s.Path return s.Path
} }
func (s *StatusLineNode) HasExactlyOneChild() bool {
return len(s.Children) == 1
}
func (s *StatusLineNode) Compress() {
if s == nil {
return
}
s.compressAux()
}
func (s *StatusLineNode) compressAux() *StatusLineNode {
if s.IsLeaf() {
return s
}
for i, child := range s.Children {
if child.HasExactlyOneChild() {
grandchild := child.Children[0]
grandchild.Name = fmt.Sprintf("%s/%s", child.Name, grandchild.Name)
s.Children[i] = grandchild
}
}
for i, child := range s.Children {
s.Children[i] = child.compressAux()
}
return s
}

View File

@ -6,16 +6,16 @@ import (
"github.com/stretchr/testify/assert" "github.com/stretchr/testify/assert"
) )
func TestRender(t *testing.T) { func TestCompress(t *testing.T) {
scenarios := []struct { scenarios := []struct {
name string name string
root *StatusLineNode root *StatusLineNode
expected []string expected *StatusLineNode
}{ }{
{ {
name: "nil node", name: "nil node",
root: nil, root: nil,
expected: []string{}, expected: nil,
}, },
{ {
name: "leaf node", name: "leaf node",
@ -25,7 +25,12 @@ func TestRender(t *testing.T) {
{File: &File{Name: "test", ShortStatus: " M", HasStagedChanges: true}, Name: "test"}, {File: &File{Name: "test", ShortStatus: " M", HasStagedChanges: true}, Name: "test"},
}, },
}, },
expected: []string{" M test"}, expected: &StatusLineNode{
Name: "",
Children: []*StatusLineNode{
{File: &File{Name: "test", ShortStatus: " M", HasStagedChanges: true}, Name: "test"},
},
},
}, },
{ {
name: "big example", name: "big example",
@ -33,44 +38,78 @@ func TestRender(t *testing.T) {
Name: "", Name: "",
Children: []*StatusLineNode{ Children: []*StatusLineNode{
{ {
Name: "dir1", Name: "dir1",
Collapsed: true, Path: "dir1",
Children: []*StatusLineNode{ Children: []*StatusLineNode{
{ {
File: &File{Name: "file2", ShortStatus: "M ", HasUnstagedChanges: true}, File: &File{Name: "file2", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file2", Name: "file2",
Path: "dir1/file2",
}, },
}, },
}, },
{ {
Name: "dir2", Name: "dir2",
Path: "dir2",
Children: []*StatusLineNode{ Children: []*StatusLineNode{
{ {
File: &File{Name: "file3", ShortStatus: " M", HasStagedChanges: true}, File: &File{Name: "file3", ShortStatus: " M", HasStagedChanges: true},
Name: "file3", Name: "file3",
Path: "dir2/file3",
}, },
{ {
File: &File{Name: "file4", ShortStatus: "M ", HasUnstagedChanges: true}, File: &File{Name: "file4", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file4", Name: "file4",
Path: "dir2/file4",
}, },
}, },
}, },
{ {
File: &File{Name: "file1", ShortStatus: "M ", HasUnstagedChanges: true}, File: &File{Name: "file1", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file1", Name: "file1",
Path: "file1",
},
},
},
expected: &StatusLineNode{
Name: "",
Children: []*StatusLineNode{
{
Name: "dir1/file2",
File: &File{Name: "file2", ShortStatus: "M ", HasUnstagedChanges: true},
Path: "dir1/file2",
},
{
Name: "dir2",
Path: "dir2",
Children: []*StatusLineNode{
{
File: &File{Name: "file3", ShortStatus: " M", HasStagedChanges: true},
Name: "file3",
Path: "dir2/file3",
},
{
File: &File{Name: "file4", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file4",
Path: "dir2/file4",
},
},
},
{
File: &File{Name: "file1", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file1",
Path: "file1",
}, },
}, },
}, },
expected: []string{"M dir1 ►", "MM dir2 ▼", " M file3", " M file4", "M file1"},
}, },
} }
for _, s := range scenarios { for _, s := range scenarios {
s := s s := s
t.Run(s.name, func(t *testing.T) { t.Run(s.name, func(t *testing.T) {
result := s.root.Render()[1:] s.root.Compress()
assert.EqualValues(t, s.expected, result) assert.EqualValues(t, s.expected, s.root)
}) })
} }
} }

View File

@ -0,0 +1,87 @@
package gui
import (
"testing"
"github.com/jesseduffield/lazygit/pkg/commands/models"
"github.com/stretchr/testify/assert"
)
func TestRender(t *testing.T) {
scenarios := []struct {
name string
root *models.StatusLineNode
expected []string
}{
{
name: "nil node",
root: nil,
expected: []string{},
},
{
name: "leaf node",
root: &models.StatusLineNode{
Name: "",
Children: []*models.StatusLineNode{
{File: &models.File{Name: "test", ShortStatus: " M", HasStagedChanges: true}, Name: "test"},
},
},
expected: []string{" M test"},
},
{
name: "big example",
root: &models.StatusLineNode{
Name: "",
Children: []*models.StatusLineNode{
{
Name: "dir1",
Collapsed: true,
Children: []*models.StatusLineNode{
{
File: &models.File{Name: "file2", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file2",
},
},
},
{
Name: "dir2",
Children: []*models.StatusLineNode{
{
Name: "dir2",
Children: []*models.StatusLineNode{
{
File: &models.File{Name: "file3", ShortStatus: " M", HasStagedChanges: true},
Name: "file3",
},
{
File: &models.File{Name: "file4", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file4",
},
},
},
{
File: &models.File{Name: "file5", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file5",
},
},
},
{
File: &models.File{Name: "file1", ShortStatus: "M ", HasUnstagedChanges: true},
Name: "file1",
},
},
},
expected: []string{" M dir1 ►", "MM dir2 ▼", "├─ MM dir2 ▼", "│ ├─ M file3", "│ └─ M file4", "└─ M file5", "M file1"},
},
}
for _, s := range scenarios {
s := s
t.Run(s.name, func(t *testing.T) {
mngr := &StatusLineManager{Tree: s.root}
result := mngr.Render("", nil)
assert.EqualValues(t, s.expected, result)
})
}
}

View File

@ -43,7 +43,8 @@ func GetTreeFromStatusFiles(files []*models.File) *models.StatusLineNode {
} }
} }
root.SortTree() root.Sort()
root.Compress()
return root return root
} }