package presentation import ( "fmt" "strings" "github.com/jesseduffield/lazygit/pkg/commands/models" "github.com/jesseduffield/lazygit/pkg/commands/patch" "github.com/jesseduffield/lazygit/pkg/gui/filetree" "github.com/jesseduffield/lazygit/pkg/gui/style" "github.com/jesseduffield/lazygit/pkg/theme" "github.com/jesseduffield/lazygit/pkg/utils" ) const ( EXPANDED_ARROW = "▼" COLLAPSED_ARROW = "►" ) const ( INNER_ITEM = "├─ " LAST_ITEM = "└─ " NESTED = "│ " NOTHING = " " ) func RenderFileTree( tree filetree.IFileTree, diffName string, submoduleConfigs []*models.SubmoduleConfig, ) []string { return renderAux(tree.Tree(), tree.CollapsedPaths(), "", -1, func(n filetree.INode, depth int) string { castN := n.(*filetree.FileNode) return getFileLine(castN.GetHasUnstagedChanges(), castN.GetHasStagedChanges(), castN.NameAtDepth(depth), diffName, submoduleConfigs, castN.File) }) } func RenderCommitFileTree( tree *filetree.CommitFileTreeViewModel, diffName string, patchManager *patch.PatchManager, ) []string { return renderAux(tree.Tree(), tree.CollapsedPaths(), "", -1, func(n filetree.INode, depth int) string { castN := n.(*filetree.CommitFileNode) // 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, tree.GetRefName()) == patch.WHOLE }) { status = patch.WHOLE } else if castN.EveryFile(func(file *models.CommitFile) bool { return patchManager.GetFileStatus(file.Name, tree.GetRefName()) == patch.UNSELECTED }) { status = patch.UNSELECTED } else { status = patch.PART } return getCommitFileLine(castN.NameAtDepth(depth), diffName, castN.File, status) }) } func renderAux( s filetree.INode, collapsedPaths filetree.CollapsedPaths, prefix string, depth int, renderLine func(filetree.INode, int) string, ) []string { if s == nil || s.IsNil() { return []string{} } isRoot := depth == -1 renderLineWithPrefix := func() string { return prefix + renderLine(s, depth) } if s.IsLeaf() { if isRoot { return []string{} } return []string{renderLineWithPrefix()} } if collapsedPaths.IsCollapsed(s.GetPath()) { return []string{fmt.Sprintf("%s %s", renderLineWithPrefix(), COLLAPSED_ARROW)} } arr := []string{} if !isRoot { arr = append(arr, fmt.Sprintf("%s %s", renderLineWithPrefix(), EXPANDED_ARROW)) } newPrefix := prefix if strings.HasSuffix(prefix, LAST_ITEM) { newPrefix = strings.TrimSuffix(prefix, LAST_ITEM) + NOTHING } else if strings.HasSuffix(prefix, INNER_ITEM) { newPrefix = strings.TrimSuffix(prefix, INNER_ITEM) + NESTED } for i, child := range s.GetChildren() { isLast := i == len(s.GetChildren())-1 var childPrefix string if isRoot { childPrefix = newPrefix } else if isLast { childPrefix = newPrefix + LAST_ITEM } else { childPrefix = newPrefix + INNER_ITEM } arr = append(arr, renderAux(child, collapsedPaths, childPrefix, depth+1+s.GetCompressionLevel(), renderLine)...) } return arr } func getFileLine(hasUnstagedChanges bool, hasStagedChanges bool, name string, diffName string, submoduleConfigs []*models.SubmoduleConfig, file *models.File) string { // potentially inefficient to be instantiating these color // objects with each render partiallyModifiedColor := style.FgYellow restColor := style.FgGreen if name == diffName { restColor = theme.DiffTerminalColor } else if file == nil && hasStagedChanges && hasUnstagedChanges { restColor = partiallyModifiedColor } else if hasUnstagedChanges { restColor = theme.UnstagedChangesColor } output := "" if file != nil { // this is just making things look nice when the background attribute is 'reverse' firstChar := file.ShortStatus[0:1] firstCharCl := style.FgGreen if firstChar == "?" { firstCharCl = theme.UnstagedChangesColor } else if firstChar == " " { firstCharCl = restColor } secondChar := file.ShortStatus[1:2] secondCharCl := theme.UnstagedChangesColor if secondChar == " " { secondCharCl = restColor } output = firstCharCl.Sprint(firstChar) output += secondCharCl.Sprint(secondChar) output += restColor.Sprint(" ") } output += restColor.Sprint(utils.EscapeSpecialChars(name)) if file != nil && file.IsSubmodule(submoduleConfigs) { output += theme.DefaultTextColor.Sprint(" (submodule)") } return output } func getCommitFileLine(name string, diffName string, commitFile *models.CommitFile, status patch.PatchStatus) string { var colour style.TextStyle if diffName == name { colour = theme.DiffTerminalColor } else { switch status { case patch.WHOLE: colour = style.FgGreen case patch.PART: colour = style.FgYellow case patch.UNSELECTED: colour = theme.DefaultTextColor } } name = utils.EscapeSpecialChars(name) if commitFile == nil { return colour.Sprint(name) } return getColorForChangeStatus(commitFile.ChangeStatus).Sprint(commitFile.ChangeStatus) + " " + colour.Sprint(name) } func getColorForChangeStatus(changeStatus string) style.TextStyle { switch changeStatus { case "A": return style.FgGreen case "M", "R": return style.FgYellow case "D": return theme.UnstagedChangesColor case "C": return style.FgCyan case "T": return style.FgMagenta default: return theme.DefaultTextColor } }