mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-24 05:36:19 +02:00
185 lines
4.6 KiB
Go
185 lines
4.6 KiB
Go
package filetree
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
|
"github.com/samber/lo"
|
|
"github.com/sirupsen/logrus"
|
|
)
|
|
|
|
type FileTreeDisplayFilter int
|
|
|
|
const (
|
|
DisplayAll FileTreeDisplayFilter = iota
|
|
DisplayStaged
|
|
DisplayUnstaged
|
|
// this shows files with merge conflicts
|
|
DisplayConflicted
|
|
)
|
|
|
|
type ITree[T any] interface {
|
|
InTreeMode() bool
|
|
ExpandToPath(path string)
|
|
ToggleShowTree()
|
|
GetIndexForPath(path string) (int, bool)
|
|
Len() int
|
|
GetItem(index int) types.HasUrn
|
|
SetTree()
|
|
IsCollapsed(path string) bool
|
|
ToggleCollapsed(path string)
|
|
CollapsedPaths() *CollapsedPaths
|
|
}
|
|
|
|
type IFileTree interface {
|
|
ITree[models.File]
|
|
|
|
FilterFiles(test func(*models.File) bool) []*models.File
|
|
SetStatusFilter(filter FileTreeDisplayFilter)
|
|
Get(index int) *FileNode
|
|
GetFile(path string) *models.File
|
|
GetAllItems() []*FileNode
|
|
GetAllFiles() []*models.File
|
|
GetFilter() FileTreeDisplayFilter
|
|
GetRoot() *FileNode
|
|
}
|
|
|
|
type FileTree struct {
|
|
getFiles func() []*models.File
|
|
tree *Node[models.File]
|
|
showTree bool
|
|
log *logrus.Entry
|
|
filter FileTreeDisplayFilter
|
|
collapsedPaths *CollapsedPaths
|
|
}
|
|
|
|
var _ IFileTree = &FileTree{}
|
|
|
|
func NewFileTree(getFiles func() []*models.File, log *logrus.Entry, showTree bool) *FileTree {
|
|
return &FileTree{
|
|
getFiles: getFiles,
|
|
log: log,
|
|
showTree: showTree,
|
|
filter: DisplayAll,
|
|
collapsedPaths: NewCollapsedPaths(),
|
|
}
|
|
}
|
|
|
|
func (self *FileTree) InTreeMode() bool {
|
|
return self.showTree
|
|
}
|
|
|
|
func (self *FileTree) ExpandToPath(path string) {
|
|
self.collapsedPaths.ExpandToPath(path)
|
|
}
|
|
|
|
func (self *FileTree) getFilesForDisplay() []*models.File {
|
|
switch self.filter {
|
|
case DisplayAll:
|
|
return self.getFiles()
|
|
case DisplayStaged:
|
|
return self.FilterFiles(func(file *models.File) bool { return file.HasStagedChanges })
|
|
case DisplayUnstaged:
|
|
return self.FilterFiles(func(file *models.File) bool { return file.HasUnstagedChanges })
|
|
case DisplayConflicted:
|
|
return self.FilterFiles(func(file *models.File) bool { return file.HasMergeConflicts })
|
|
default:
|
|
panic(fmt.Sprintf("Unexpected files display filter: %d", self.filter))
|
|
}
|
|
}
|
|
|
|
func (self *FileTree) FilterFiles(test func(*models.File) bool) []*models.File {
|
|
return lo.Filter(self.getFiles(), func(file *models.File, _ int) bool { return test(file) })
|
|
}
|
|
|
|
func (self *FileTree) SetStatusFilter(filter FileTreeDisplayFilter) {
|
|
self.filter = filter
|
|
self.SetTree()
|
|
}
|
|
|
|
func (self *FileTree) ToggleShowTree() {
|
|
self.showTree = !self.showTree
|
|
self.SetTree()
|
|
}
|
|
|
|
func (self *FileTree) Get(index int) *FileNode {
|
|
// need to traverse the tree depth first until we get to the index.
|
|
return NewFileNode(self.tree.GetNodeAtIndex(index+1, self.collapsedPaths)) // ignoring root
|
|
}
|
|
|
|
func (self *FileTree) GetFile(path string) *models.File {
|
|
for _, file := range self.getFiles() {
|
|
if file.Name == path {
|
|
return file
|
|
}
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (self *FileTree) GetIndexForPath(path string) (int, bool) {
|
|
index, found := self.tree.GetIndexForPath(path, self.collapsedPaths)
|
|
return index - 1, found
|
|
}
|
|
|
|
// note: this gets all items when the filter is taken into consideration. There may
|
|
// be hidden files that aren't included here. Files off the screen however will
|
|
// be included
|
|
func (self *FileTree) GetAllItems() []*FileNode {
|
|
if self.tree == nil {
|
|
return nil
|
|
}
|
|
|
|
// ignoring root
|
|
return lo.Map(self.tree.Flatten(self.collapsedPaths)[1:], func(node *Node[models.File], _ int) *FileNode {
|
|
return NewFileNode(node)
|
|
})
|
|
}
|
|
|
|
func (self *FileTree) Len() int {
|
|
return self.tree.Size(self.collapsedPaths) - 1 // ignoring root
|
|
}
|
|
|
|
func (self *FileTree) GetItem(index int) types.HasUrn {
|
|
// Unimplemented because we don't yet need to show inlines statuses in commit file views
|
|
return nil
|
|
}
|
|
|
|
func (self *FileTree) GetAllFiles() []*models.File {
|
|
return self.getFiles()
|
|
}
|
|
|
|
func (self *FileTree) SetTree() {
|
|
filesForDisplay := self.getFilesForDisplay()
|
|
if self.showTree {
|
|
self.tree = BuildTreeFromFiles(filesForDisplay)
|
|
} else {
|
|
self.tree = BuildFlatTreeFromFiles(filesForDisplay)
|
|
}
|
|
}
|
|
|
|
func (self *FileTree) IsCollapsed(path string) bool {
|
|
return self.collapsedPaths.IsCollapsed(path)
|
|
}
|
|
|
|
func (self *FileTree) ToggleCollapsed(path string) {
|
|
self.collapsedPaths.ToggleCollapsed(path)
|
|
}
|
|
|
|
func (self *FileTree) Tree() *FileNode {
|
|
return NewFileNode(self.tree)
|
|
}
|
|
|
|
func (self *FileTree) GetRoot() *FileNode {
|
|
return NewFileNode(self.tree)
|
|
}
|
|
|
|
func (self *FileTree) CollapsedPaths() *CollapsedPaths {
|
|
return self.collapsedPaths
|
|
}
|
|
|
|
func (self *FileTree) GetFilter() FileTreeDisplayFilter {
|
|
return self.filter
|
|
}
|