2021-03-31 13:08:55 +02:00
|
|
|
package filetree
|
|
|
|
|
|
|
|
import (
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
2021-03-31 13:39:55 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/patch"
|
2021-03-31 13:08:55 +02:00
|
|
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation"
|
|
|
|
"github.com/sirupsen/logrus"
|
|
|
|
)
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
type CommitFileManager struct {
|
2021-03-31 13:08:55 +02:00
|
|
|
files []*models.CommitFile
|
2021-03-31 14:26:53 +02:00
|
|
|
tree *CommitFileNode
|
2021-03-31 13:08:55 +02:00
|
|
|
showTree bool
|
|
|
|
log *logrus.Entry
|
|
|
|
collapsedPaths CollapsedPaths
|
|
|
|
// parent is the identifier of the parent object e.g. a commit SHA if this commit file is for a commit, or a stash entry ref like 'stash@{1}'
|
|
|
|
parent string
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) GetParent() string {
|
2021-03-31 13:08:55 +02:00
|
|
|
return m.parent
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func NewCommitFileManager(files []*models.CommitFile, log *logrus.Entry, showTree bool) *CommitFileManager {
|
|
|
|
return &CommitFileManager{
|
2021-03-31 13:08:55 +02:00
|
|
|
files: files,
|
|
|
|
log: log,
|
|
|
|
showTree: showTree,
|
|
|
|
collapsedPaths: CollapsedPaths{},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) ExpandToPath(path string) {
|
2021-03-31 13:08:55 +02:00
|
|
|
m.collapsedPaths.ExpandToPath(path)
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) ToggleShowTree() {
|
2021-03-31 13:08:55 +02:00
|
|
|
m.showTree = !m.showTree
|
|
|
|
m.SetTree()
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) GetItemAtIndex(index int) *CommitFileNode {
|
2021-03-31 13:08:55 +02:00
|
|
|
// need to traverse the three depth first until we get to the index.
|
|
|
|
return m.tree.GetNodeAtIndex(index+1, m.collapsedPaths) // ignoring root
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) GetIndexForPath(path string) (int, bool) {
|
2021-03-31 13:08:55 +02:00
|
|
|
index, found := m.tree.GetIndexForPath(path, m.collapsedPaths)
|
|
|
|
return index - 1, found
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) GetAllItems() []*CommitFileNode {
|
2021-03-31 13:08:55 +02:00
|
|
|
if m.tree == nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.tree.Flatten(m.collapsedPaths)[1:] // ignoring root
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) GetItemsLength() int {
|
2021-03-31 13:08:55 +02:00
|
|
|
return m.tree.Size(m.collapsedPaths) - 1 // ignoring root
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) GetAllFiles() []*models.CommitFile {
|
2021-03-31 13:08:55 +02:00
|
|
|
return m.files
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) SetFiles(files []*models.CommitFile, parent string) {
|
2021-03-31 13:08:55 +02:00
|
|
|
m.files = files
|
|
|
|
m.parent = parent
|
|
|
|
|
|
|
|
m.SetTree()
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) SetTree() {
|
2021-03-31 13:08:55 +02:00
|
|
|
if m.showTree {
|
|
|
|
m.tree = BuildTreeFromCommitFiles(m.files)
|
|
|
|
} else {
|
|
|
|
m.tree = BuildFlatTreeFromCommitFiles(m.files)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) IsCollapsed(path string) bool {
|
2021-03-31 13:08:55 +02:00
|
|
|
return m.collapsedPaths.IsCollapsed(path)
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) ToggleCollapsed(path string) {
|
2021-03-31 13:08:55 +02:00
|
|
|
m.collapsedPaths.ToggleCollapsed(path)
|
|
|
|
}
|
|
|
|
|
2021-03-31 14:26:53 +02:00
|
|
|
func (m *CommitFileManager) Render(diffName string, patchManager *patch.PatchManager) []string {
|
2021-04-01 00:25:11 +02:00
|
|
|
// can't rely on renderAux to check for nil because an interface won't be nil if its concrete value is nil
|
|
|
|
if m.tree == nil {
|
|
|
|
return []string{}
|
|
|
|
}
|
|
|
|
|
2021-03-31 13:08:55 +02:00
|
|
|
return renderAux(m.tree, m.collapsedPaths, "", -1, func(n INode, depth int) string {
|
2021-03-31 14:26:53 +02:00
|
|
|
castN := n.(*CommitFileNode)
|
2021-03-31 14:00:24 +02:00
|
|
|
|
|
|
|
// This is a little convoluted because we're dealing with either a leaf or a non-leaf.
|
|
|
|
// But this code actually applies to both. If it's a leaf, the status will just
|
|
|
|
// be whatever status it is, but if it's a non-leaf it will determine its status
|
|
|
|
// based on the leaves of that subtree
|
|
|
|
var status patch.PatchStatus
|
|
|
|
if castN.EveryFile(func(file *models.CommitFile) bool {
|
|
|
|
return patchManager.GetFileStatus(file.Name, m.parent) == patch.WHOLE
|
|
|
|
}) {
|
|
|
|
status = patch.WHOLE
|
|
|
|
} else if castN.EveryFile(func(file *models.CommitFile) bool {
|
|
|
|
return patchManager.GetFileStatus(file.Name, m.parent) == patch.UNSELECTED
|
|
|
|
}) {
|
|
|
|
status = patch.UNSELECTED
|
|
|
|
} else {
|
|
|
|
status = patch.PART
|
|
|
|
}
|
|
|
|
|
|
|
|
return presentation.GetCommitFileLine(castN.NameAtDepth(depth), diffName, castN.File, status)
|
2021-03-31 13:08:55 +02:00
|
|
|
})
|
|
|
|
}
|