2022-01-22 00:13:51 +11:00
|
|
|
package filetree
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
2022-01-30 14:46:46 +11:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/context/traits"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/types"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
2022-01-22 00:13:51 +11:00
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
type IFileTreeViewModel interface {
|
|
|
|
IFileTree
|
|
|
|
types.IListCursor
|
|
|
|
}
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
// This combines our FileTree struct with a cursor that retains information about
|
|
|
|
// which item is selected. It also contains logic for repositioning that cursor
|
|
|
|
// after the files are refreshed
|
2022-01-22 00:13:51 +11:00
|
|
|
type FileTreeViewModel struct {
|
|
|
|
sync.RWMutex
|
2022-01-30 14:46:46 +11:00
|
|
|
IFileTree
|
|
|
|
types.IListCursor
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
var _ IFileTreeViewModel = &FileTreeViewModel{}
|
|
|
|
|
2022-01-30 13:08:09 +11:00
|
|
|
func NewFileTreeViewModel(getFiles func() []*models.File, log *logrus.Entry, showTree bool) *FileTreeViewModel {
|
2022-01-30 14:46:46 +11:00
|
|
|
fileTree := NewFileTree(getFiles, log, showTree)
|
|
|
|
listCursor := traits.NewListCursor(fileTree)
|
|
|
|
return &FileTreeViewModel{
|
|
|
|
IFileTree: fileTree,
|
|
|
|
IListCursor: listCursor,
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-19 09:31:52 +11:00
|
|
|
func (self *FileTreeViewModel) GetSelected() *FileNode {
|
|
|
|
if self.Len() == 0 {
|
2022-01-30 14:46:46 +11:00
|
|
|
return nil
|
|
|
|
}
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-03-19 09:31:52 +11:00
|
|
|
return self.Get(self.GetSelectedLineIdx())
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-02-24 13:29:48 +11:00
|
|
|
func (self *FileTreeViewModel) GetSelectedFile() *models.File {
|
2022-03-19 09:31:52 +11:00
|
|
|
node := self.GetSelected()
|
2022-02-24 13:29:48 +11:00
|
|
|
if node == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return node.File
|
|
|
|
}
|
|
|
|
|
|
|
|
func (self *FileTreeViewModel) GetSelectedPath() string {
|
2022-03-19 09:31:52 +11:00
|
|
|
node := self.GetSelected()
|
2022-02-24 13:29:48 +11:00
|
|
|
if node == nil {
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
return node.GetPath()
|
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
func (self *FileTreeViewModel) SetTree() {
|
|
|
|
newFiles := self.GetAllFiles()
|
2022-03-19 09:31:52 +11:00
|
|
|
selectedNode := self.GetSelected()
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
// for when you stage the old file of a rename and the new file is in a collapsed dir
|
|
|
|
for _, file := range newFiles {
|
|
|
|
if selectedNode != nil && selectedNode.Path != "" && file.PreviousName == selectedNode.Path {
|
|
|
|
self.ExpandToPath(file.Name)
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
prevNodes := self.GetAllItems()
|
|
|
|
prevSelectedLineIdx := self.GetSelectedLineIdx()
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
self.IFileTree.SetTree()
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
if selectedNode != nil {
|
|
|
|
newNodes := self.GetAllItems()
|
|
|
|
newIdx := self.findNewSelectedIdx(prevNodes[prevSelectedLineIdx:], newNodes)
|
|
|
|
if newIdx != -1 && newIdx != prevSelectedLineIdx {
|
|
|
|
self.SetSelectedLineIdx(newIdx)
|
2022-01-26 01:20:19 +11:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
self.RefreshSelectedIdx()
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
// Let's try to find our file again and move the cursor to that.
|
|
|
|
// If we can't find our file, it was probably just removed by the user. In that
|
|
|
|
// case, we go looking for where the next file has been moved to. Given that the
|
|
|
|
// user could have removed a whole directory, we continue iterating through the old
|
|
|
|
// nodes until we find one that exists in the new set of nodes, then move the cursor
|
|
|
|
// to that.
|
|
|
|
// prevNodes starts from our previously selected node because we don't need to consider anything above that
|
|
|
|
func (self *FileTreeViewModel) findNewSelectedIdx(prevNodes []*FileNode, currNodes []*FileNode) int {
|
|
|
|
getPaths := func(node *FileNode) []string {
|
|
|
|
if node == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if node.File != nil && node.File.IsRename() {
|
|
|
|
return node.File.Names()
|
|
|
|
} else {
|
|
|
|
return []string{node.Path}
|
|
|
|
}
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
for _, prevNode := range prevNodes {
|
|
|
|
selectedPaths := getPaths(prevNode)
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
for idx, node := range currNodes {
|
|
|
|
paths := getPaths(node)
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
// If you started off with a rename selected, and now it's broken in two, we want you to jump to the new file, not the old file.
|
|
|
|
// This is because the new should be in the same position as the rename was meaning less cursor jumping
|
|
|
|
foundOldFileInRename := prevNode.File != nil && prevNode.File.IsRename() && node.Path == prevNode.File.PreviousName
|
|
|
|
foundNode := utils.StringArraysOverlap(paths, selectedPaths) && !foundOldFileInRename
|
|
|
|
if foundNode {
|
|
|
|
return idx
|
|
|
|
}
|
|
|
|
}
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
return -1
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
func (self *FileTreeViewModel) SetFilter(filter FileTreeDisplayFilter) {
|
|
|
|
self.IFileTree.SetFilter(filter)
|
|
|
|
self.IListCursor.SetSelectedLineIdx(0)
|
2022-01-22 00:13:51 +11:00
|
|
|
}
|
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
// If we're going from flat to tree we want to select the same file.
|
|
|
|
// If we're going from tree to flat and we have a file selected we want to select that.
|
|
|
|
// If instead we've selected a directory we need to select the first file in that directory.
|
|
|
|
func (self *FileTreeViewModel) ToggleShowTree() {
|
2022-03-19 09:31:52 +11:00
|
|
|
selectedNode := self.GetSelected()
|
2022-01-22 00:13:51 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
self.IFileTree.ToggleShowTree()
|
2022-01-26 19:45:42 +11:00
|
|
|
|
2022-01-30 14:46:46 +11:00
|
|
|
if selectedNode == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
path := selectedNode.Path
|
|
|
|
|
|
|
|
if self.InTreeMode() {
|
|
|
|
self.ExpandToPath(path)
|
|
|
|
} else if len(selectedNode.Children) > 0 {
|
|
|
|
path = selectedNode.GetLeaves()[0].Path
|
|
|
|
}
|
|
|
|
|
|
|
|
index, found := self.GetIndexForPath(path)
|
|
|
|
if found {
|
|
|
|
self.SetSelectedLineIdx(index)
|
|
|
|
}
|
2022-01-26 19:45:42 +11:00
|
|
|
}
|