mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-06-15 00:15:32 +02:00
file tree for commit files
This commit is contained in:
@ -38,7 +38,6 @@ func (c *GitCommand) getCommitFilesFromFilenames(filenames string, parent string
|
|||||||
}
|
}
|
||||||
|
|
||||||
commitFiles = append(commitFiles, &models.CommitFile{
|
commitFiles = append(commitFiles, &models.CommitFile{
|
||||||
Parent: parent,
|
|
||||||
Name: name,
|
Name: name,
|
||||||
ChangeStatus: changeStatus,
|
ChangeStatus: changeStatus,
|
||||||
PatchStatus: status,
|
PatchStatus: status,
|
||||||
|
@ -2,8 +2,6 @@ package models
|
|||||||
|
|
||||||
// CommitFile : A git commit file
|
// CommitFile : A git commit file
|
||||||
type CommitFile struct {
|
type CommitFile struct {
|
||||||
// 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
|
|
||||||
Name string
|
Name string
|
||||||
|
|
||||||
// PatchStatus tells us whether the file has been wholly or partially added to a patch. We might want to pull this logic up into the gui package and make it a map like we do with cherry picked commits
|
// PatchStatus tells us whether the file has been wholly or partially added to a patch. We might want to pull this logic up into the gui package and make it a map like we do with cherry picked commits
|
||||||
|
@ -3,30 +3,49 @@ package gui
|
|||||||
import (
|
import (
|
||||||
"github.com/jesseduffield/gocui"
|
"github.com/jesseduffield/gocui"
|
||||||
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
||||||
|
"github.com/jesseduffield/lazygit/pkg/gui/filetree"
|
||||||
)
|
)
|
||||||
|
|
||||||
func (gui *Gui) getSelectedCommitFile() *models.CommitFile {
|
// todo: rename to getSelectedCommitFileChangeNode, or decide to remove the change part in the context of files
|
||||||
|
func (gui *Gui) getSelectedCommitFileNode() *filetree.CommitFileChangeNode {
|
||||||
selectedLine := gui.State.Panels.CommitFiles.SelectedLineIdx
|
selectedLine := gui.State.Panels.CommitFiles.SelectedLineIdx
|
||||||
if selectedLine == -1 || selectedLine > len(gui.State.CommitFiles)-1 {
|
if selectedLine == -1 || selectedLine > gui.State.CommitFileChangeManager.GetItemsLength()-1 {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return gui.State.CommitFiles[selectedLine]
|
return gui.State.CommitFileChangeManager.GetItemAtIndex(selectedLine)
|
||||||
|
}
|
||||||
|
|
||||||
|
// todo: rename to getSelectedCommitFileChange
|
||||||
|
func (gui *Gui) getSelectedCommitFile() *models.CommitFile {
|
||||||
|
node := gui.getSelectedCommitFileNode()
|
||||||
|
if node == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return node.File
|
||||||
|
}
|
||||||
|
|
||||||
|
func (gui *Gui) getSelectedCommitFilePath() string {
|
||||||
|
node := gui.getSelectedCommitFileNode()
|
||||||
|
if node == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
return node.GetPath()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) handleCommitFileSelect() error {
|
func (gui *Gui) handleCommitFileSelect() error {
|
||||||
gui.escapeLineByLinePanel()
|
gui.escapeLineByLinePanel()
|
||||||
|
|
||||||
commitFile := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if commitFile == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
to := commitFile.Parent
|
to := gui.State.CommitFileChangeManager.GetParent()
|
||||||
from, reverse := gui.getFromAndReverseArgsForDiff(to)
|
from, reverse := gui.getFromAndReverseArgsForDiff(to)
|
||||||
|
|
||||||
cmd := gui.OSCommand.ExecutableFromString(
|
cmd := gui.OSCommand.ExecutableFromString(
|
||||||
gui.GitCommand.ShowFileDiffCmdStr(from, to, reverse, commitFile.Name, false),
|
gui.GitCommand.ShowFileDiffCmdStr(from, to, reverse, node.GetPath(), false),
|
||||||
)
|
)
|
||||||
task := gui.createRunPtyTask(cmd)
|
task := gui.createRunPtyTask(cmd)
|
||||||
|
|
||||||
@ -40,12 +59,13 @@ func (gui *Gui) handleCommitFileSelect() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) handleCheckoutCommitFile(g *gocui.Gui, v *gocui.View) error {
|
func (gui *Gui) handleCheckoutCommitFile(g *gocui.Gui, v *gocui.View) error {
|
||||||
file := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if file == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := gui.GitCommand.CheckoutFile(file.Parent, file.Name); err != nil {
|
// TODO: verify this works for directories
|
||||||
|
if err := gui.GitCommand.CheckoutFile(gui.State.CommitFileChangeManager.GetParent(), node.GetPath()); err != nil {
|
||||||
return gui.surfaceError(err)
|
return gui.surfaceError(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -57,7 +77,7 @@ func (gui *Gui) handleDiscardOldFileChange(g *gocui.Gui, v *gocui.View) error {
|
|||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
fileName := gui.State.CommitFiles[gui.State.Panels.CommitFiles.SelectedLineIdx].Name
|
fileName := gui.getSelectedCommitFileName()
|
||||||
|
|
||||||
return gui.ask(askOpts{
|
return gui.ask(askOpts{
|
||||||
title: gui.Tr.DiscardFileChangesTitle,
|
title: gui.Tr.DiscardFileChangesTitle,
|
||||||
@ -88,35 +108,37 @@ func (gui *Gui) refreshCommitFilesView() error {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return gui.surfaceError(err)
|
return gui.surfaceError(err)
|
||||||
}
|
}
|
||||||
gui.State.CommitFiles = files
|
gui.State.CommitFileChangeManager.SetFiles(files, to)
|
||||||
|
|
||||||
return gui.postRefreshUpdate(gui.Contexts.CommitFiles.Context)
|
return gui.postRefreshUpdate(gui.Contexts.CommitFiles.Context)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) handleOpenOldCommitFile(g *gocui.Gui, v *gocui.View) error {
|
func (gui *Gui) handleOpenOldCommitFile(g *gocui.Gui, v *gocui.View) error {
|
||||||
file := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if file == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return gui.openFile(file.Name)
|
return gui.openFile(node.GetPath())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) handleEditCommitFile(g *gocui.Gui, v *gocui.View) error {
|
func (gui *Gui) handleEditCommitFile(g *gocui.Gui, v *gocui.View) error {
|
||||||
file := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if file == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
return gui.editFile(file.Name)
|
return gui.editFile(node.GetPath())
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) handleToggleFileForPatch(g *gocui.Gui, v *gocui.View) error {
|
func (gui *Gui) handleToggleFileForPatch(g *gocui.Gui, v *gocui.View) error {
|
||||||
commitFile := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if commitFile == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: if file is nil, toggle all leaves underneath on/off
|
||||||
|
|
||||||
toggleTheFile := func() error {
|
toggleTheFile := func() error {
|
||||||
if !gui.GitCommand.PatchManager.Active() {
|
if !gui.GitCommand.PatchManager.Active() {
|
||||||
if err := gui.startPatchManager(); err != nil {
|
if err := gui.startPatchManager(); err != nil {
|
||||||
@ -124,7 +146,7 @@ func (gui *Gui) handleToggleFileForPatch(g *gocui.Gui, v *gocui.View) error {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := gui.GitCommand.PatchManager.ToggleFileWhole(commitFile.Name); err != nil {
|
if err := gui.GitCommand.PatchManager.ToggleFileWhole(node.GetPath()); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -135,7 +157,7 @@ func (gui *Gui) handleToggleFileForPatch(g *gocui.Gui, v *gocui.View) error {
|
|||||||
return gui.refreshCommitFilesView()
|
return gui.refreshCommitFilesView()
|
||||||
}
|
}
|
||||||
|
|
||||||
if gui.GitCommand.PatchManager.Active() && gui.GitCommand.PatchManager.To != commitFile.Parent {
|
if gui.GitCommand.PatchManager.Active() && gui.GitCommand.PatchManager.To != gui.State.CommitFileChangeManager.GetParent() {
|
||||||
return gui.ask(askOpts{
|
return gui.ask(askOpts{
|
||||||
title: gui.Tr.DiscardPatch,
|
title: gui.Tr.DiscardPatch,
|
||||||
prompt: gui.Tr.DiscardPatchConfirm,
|
prompt: gui.Tr.DiscardPatchConfirm,
|
||||||
@ -164,11 +186,15 @@ func (gui *Gui) handleEnterCommitFile(g *gocui.Gui, v *gocui.View) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) enterCommitFile(selectedLineIdx int) error {
|
func (gui *Gui) enterCommitFile(selectedLineIdx int) error {
|
||||||
commitFile := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if commitFile == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if node.File == nil {
|
||||||
|
return gui.handleToggleCommitFileDirCollapsed()
|
||||||
|
}
|
||||||
|
|
||||||
enterTheFile := func(selectedLineIdx int) error {
|
enterTheFile := func(selectedLineIdx int) error {
|
||||||
if !gui.GitCommand.PatchManager.Active() {
|
if !gui.GitCommand.PatchManager.Active() {
|
||||||
if err := gui.startPatchManager(); err != nil {
|
if err := gui.startPatchManager(); err != nil {
|
||||||
@ -182,7 +208,7 @@ func (gui *Gui) enterCommitFile(selectedLineIdx int) error {
|
|||||||
return gui.handleRefreshPatchBuildingPanel(selectedLineIdx)
|
return gui.handleRefreshPatchBuildingPanel(selectedLineIdx)
|
||||||
}
|
}
|
||||||
|
|
||||||
if gui.GitCommand.PatchManager.Active() && gui.GitCommand.PatchManager.To != commitFile.Parent {
|
if gui.GitCommand.PatchManager.Active() && gui.GitCommand.PatchManager.To != gui.State.CommitFileChangeManager.GetParent() {
|
||||||
return gui.ask(askOpts{
|
return gui.ask(askOpts{
|
||||||
title: gui.Tr.DiscardPatch,
|
title: gui.Tr.DiscardPatch,
|
||||||
prompt: gui.Tr.DiscardPatchConfirm,
|
prompt: gui.Tr.DiscardPatchConfirm,
|
||||||
@ -200,6 +226,21 @@ func (gui *Gui) enterCommitFile(selectedLineIdx int) error {
|
|||||||
return enterTheFile(selectedLineIdx)
|
return enterTheFile(selectedLineIdx)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (gui *Gui) handleToggleCommitFileDirCollapsed() error {
|
||||||
|
node := gui.getSelectedCommitFileNode()
|
||||||
|
if node == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
gui.State.CommitFileChangeManager.ToggleCollapsed(node.GetPath())
|
||||||
|
|
||||||
|
if err := gui.postRefreshUpdate(gui.Contexts.CommitFiles.Context); err != nil {
|
||||||
|
gui.Log.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func (gui *Gui) switchToCommitFilesContext(refName string, canRebase bool, context Context, windowName string) error {
|
func (gui *Gui) switchToCommitFilesContext(refName string, canRebase bool, context Context, windowName string) error {
|
||||||
// sometimes the commitFiles view is already shown in another window, so we need to ensure that window
|
// sometimes the commitFiles view is already shown in another window, so we need to ensure that window
|
||||||
// no longer considers the commitFiles view as its main view.
|
// no longer considers the commitFiles view as its main view.
|
||||||
|
@ -23,6 +23,7 @@ type CustomCommandObjects struct {
|
|||||||
SelectedTag *models.Tag
|
SelectedTag *models.Tag
|
||||||
SelectedStashEntry *models.StashEntry
|
SelectedStashEntry *models.StashEntry
|
||||||
SelectedCommitFile *models.CommitFile
|
SelectedCommitFile *models.CommitFile
|
||||||
|
SelectedCommitFilePath string
|
||||||
CheckedOutBranch *models.Branch
|
CheckedOutBranch *models.Branch
|
||||||
PromptResponses []string
|
PromptResponses []string
|
||||||
}
|
}
|
||||||
@ -39,6 +40,7 @@ func (gui *Gui) resolveTemplate(templateStr string, promptResponses []string) (s
|
|||||||
SelectedTag: gui.getSelectedTag(),
|
SelectedTag: gui.getSelectedTag(),
|
||||||
SelectedStashEntry: gui.getSelectedStashEntry(),
|
SelectedStashEntry: gui.getSelectedStashEntry(),
|
||||||
SelectedCommitFile: gui.getSelectedCommitFile(),
|
SelectedCommitFile: gui.getSelectedCommitFile(),
|
||||||
|
SelectedCommitFilePath: gui.getSelectedCommitFilePath(),
|
||||||
SelectedSubCommit: gui.getSelectedSubCommit(),
|
SelectedSubCommit: gui.getSelectedSubCommit(),
|
||||||
CheckedOutBranch: gui.currentBranch(),
|
CheckedOutBranch: gui.currentBranch(),
|
||||||
PromptResponses: promptResponses,
|
PromptResponses: promptResponses,
|
||||||
|
@ -32,12 +32,7 @@ func (gui *Gui) getSelectedFileChangeNode() *filetree.FileChangeNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) getSelectedFile() *models.File {
|
func (gui *Gui) getSelectedFile() *models.File {
|
||||||
selectedLine := gui.State.Panels.Files.SelectedLineIdx
|
node := gui.getSelectedFileChangeNode()
|
||||||
if selectedLine == -1 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
node := gui.State.FileChangeManager.GetItemAtIndex(selectedLine)
|
|
||||||
if node == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
@ -844,7 +839,7 @@ func (gui *Gui) handleToggleDirCollapsed() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
gui.State.FileChangeManager.ToggleCollapsed(node)
|
gui.State.FileChangeManager.ToggleCollapsed(node.GetPath())
|
||||||
|
|
||||||
if err := gui.postRefreshUpdate(gui.Contexts.Files.Context); err != nil {
|
if err := gui.postRefreshUpdate(gui.Contexts.Files.Context); err != nil {
|
||||||
gui.Log.Error(err)
|
gui.Log.Error(err)
|
||||||
|
@ -49,6 +49,53 @@ func BuildTreeFromFiles(files []*models.File) *FileChangeNode {
|
|||||||
return root
|
return root
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func BuildFlatTreeFromCommitFiles(files []*models.CommitFile) *CommitFileChangeNode {
|
||||||
|
rootAux := BuildTreeFromCommitFiles(files)
|
||||||
|
sortedFiles := rootAux.GetLeaves()
|
||||||
|
|
||||||
|
return &CommitFileChangeNode{Children: sortedFiles}
|
||||||
|
}
|
||||||
|
|
||||||
|
func BuildTreeFromCommitFiles(files []*models.CommitFile) *CommitFileChangeNode {
|
||||||
|
root := &CommitFileChangeNode{}
|
||||||
|
|
||||||
|
var curr *CommitFileChangeNode
|
||||||
|
for _, file := range files {
|
||||||
|
split := strings.Split(file.Name, string(os.PathSeparator))
|
||||||
|
curr = root
|
||||||
|
outer:
|
||||||
|
for i := range split {
|
||||||
|
var setFile *models.CommitFile
|
||||||
|
isFile := i == len(split)-1
|
||||||
|
if isFile {
|
||||||
|
setFile = file
|
||||||
|
}
|
||||||
|
|
||||||
|
path := filepath.Join(split[:i+1]...)
|
||||||
|
|
||||||
|
for _, existingChild := range curr.Children {
|
||||||
|
if existingChild.Path == path {
|
||||||
|
curr = existingChild
|
||||||
|
continue outer
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
newChild := &CommitFileChangeNode{
|
||||||
|
Path: path,
|
||||||
|
File: setFile,
|
||||||
|
}
|
||||||
|
curr.Children = append(curr.Children, newChild)
|
||||||
|
|
||||||
|
curr = newChild
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
root.Sort()
|
||||||
|
root.Compress()
|
||||||
|
|
||||||
|
return root
|
||||||
|
}
|
||||||
|
|
||||||
func BuildFlatTreeFromFiles(files []*models.File) *FileChangeNode {
|
func BuildFlatTreeFromFiles(files []*models.File) *FileChangeNode {
|
||||||
rootAux := BuildTreeFromFiles(files)
|
rootAux := BuildTreeFromFiles(files)
|
||||||
sortedFiles := rootAux.GetLeaves()
|
sortedFiles := rootAux.GetLeaves()
|
||||||
|
25
pkg/gui/filetree/collapsed_paths.go
Normal file
25
pkg/gui/filetree/collapsed_paths.go
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
package filetree
|
||||||
|
|
||||||
|
import (
|
||||||
|
"os"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CollapsedPaths map[string]bool
|
||||||
|
|
||||||
|
func (cp CollapsedPaths) ExpandToPath(path string) {
|
||||||
|
// need every directory along the way
|
||||||
|
split := strings.Split(path, string(os.PathSeparator))
|
||||||
|
for i := range split {
|
||||||
|
dir := strings.Join(split[0:i+1], string(os.PathSeparator))
|
||||||
|
cp[dir] = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cp CollapsedPaths) IsCollapsed(path string) bool {
|
||||||
|
return cp[path]
|
||||||
|
}
|
||||||
|
|
||||||
|
func (cp CollapsedPaths) ToggleCollapsed(path string) {
|
||||||
|
cp[path] = !cp[path]
|
||||||
|
}
|
95
pkg/gui/filetree/commit_file_change_manager.go
Normal file
95
pkg/gui/filetree/commit_file_change_manager.go
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
package filetree
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
||||||
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
|
)
|
||||||
|
|
||||||
|
type CommitFileChangeManager struct {
|
||||||
|
files []*models.CommitFile
|
||||||
|
tree *CommitFileChangeNode
|
||||||
|
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
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) GetParent() string {
|
||||||
|
return m.parent
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewCommitFileChangeManager(files []*models.CommitFile, log *logrus.Entry, showTree bool) *CommitFileChangeManager {
|
||||||
|
return &CommitFileChangeManager{
|
||||||
|
files: files,
|
||||||
|
log: log,
|
||||||
|
showTree: showTree,
|
||||||
|
collapsedPaths: CollapsedPaths{},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) ExpandToPath(path string) {
|
||||||
|
m.collapsedPaths.ExpandToPath(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) ToggleShowTree() {
|
||||||
|
m.showTree = !m.showTree
|
||||||
|
m.SetTree()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) GetItemAtIndex(index int) *CommitFileChangeNode {
|
||||||
|
// need to traverse the three depth first until we get to the index.
|
||||||
|
return m.tree.GetNodeAtIndex(index+1, m.collapsedPaths) // ignoring root
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) GetIndexForPath(path string) (int, bool) {
|
||||||
|
index, found := m.tree.GetIndexForPath(path, m.collapsedPaths)
|
||||||
|
return index - 1, found
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) GetAllItems() []*CommitFileChangeNode {
|
||||||
|
if m.tree == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
return m.tree.Flatten(m.collapsedPaths)[1:] // ignoring root
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) GetItemsLength() int {
|
||||||
|
return m.tree.Size(m.collapsedPaths) - 1 // ignoring root
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) GetAllFiles() []*models.CommitFile {
|
||||||
|
return m.files
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) SetFiles(files []*models.CommitFile, parent string) {
|
||||||
|
m.files = files
|
||||||
|
m.parent = parent
|
||||||
|
|
||||||
|
m.SetTree()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) SetTree() {
|
||||||
|
if m.showTree {
|
||||||
|
m.tree = BuildTreeFromCommitFiles(m.files)
|
||||||
|
} else {
|
||||||
|
m.tree = BuildFlatTreeFromCommitFiles(m.files)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) IsCollapsed(path string) bool {
|
||||||
|
return m.collapsedPaths.IsCollapsed(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) ToggleCollapsed(path string) {
|
||||||
|
m.collapsedPaths.ToggleCollapsed(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *CommitFileChangeManager) Render(diffName string) []string {
|
||||||
|
return renderAux(m.tree, m.collapsedPaths, "", -1, func(n INode, depth int) string {
|
||||||
|
castN := n.(*CommitFileChangeNode)
|
||||||
|
return presentation.GetCommitFileLine(castN.NameAtDepth(depth), diffName, castN.File)
|
||||||
|
})
|
||||||
|
}
|
9
pkg/gui/filetree/constants.go
Normal file
9
pkg/gui/filetree/constants.go
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
package filetree
|
||||||
|
|
||||||
|
const EXPANDED_ARROW = "▼"
|
||||||
|
const COLLAPSED_ARROW = "►"
|
||||||
|
|
||||||
|
const INNER_ITEM = "├─ "
|
||||||
|
const LAST_ITEM = "└─ "
|
||||||
|
const NESTED = "│ "
|
||||||
|
const NOTHING = " "
|
@ -1,29 +1,17 @@
|
|||||||
package filetree
|
package filetree
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
||||||
"github.com/jesseduffield/lazygit/pkg/gui/presentation"
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation"
|
||||||
"github.com/sirupsen/logrus"
|
"github.com/sirupsen/logrus"
|
||||||
)
|
)
|
||||||
|
|
||||||
const EXPANDED_ARROW = "▼"
|
|
||||||
const COLLAPSED_ARROW = "►"
|
|
||||||
|
|
||||||
const INNER_ITEM = "├─ "
|
|
||||||
const LAST_ITEM = "└─ "
|
|
||||||
const NESTED = "│ "
|
|
||||||
const NOTHING = " "
|
|
||||||
|
|
||||||
type FileChangeManager struct {
|
type FileChangeManager struct {
|
||||||
files []*models.File
|
files []*models.File
|
||||||
tree *FileChangeNode
|
tree *FileChangeNode
|
||||||
showTree bool
|
showTree bool
|
||||||
log *logrus.Entry
|
log *logrus.Entry
|
||||||
collapsedPaths map[string]bool
|
collapsedPaths CollapsedPaths
|
||||||
}
|
}
|
||||||
|
|
||||||
func NewFileChangeManager(files []*models.File, log *logrus.Entry, showTree bool) *FileChangeManager {
|
func NewFileChangeManager(files []*models.File, log *logrus.Entry, showTree bool) *FileChangeManager {
|
||||||
@ -31,10 +19,14 @@ func NewFileChangeManager(files []*models.File, log *logrus.Entry, showTree bool
|
|||||||
files: files,
|
files: files,
|
||||||
log: log,
|
log: log,
|
||||||
showTree: showTree,
|
showTree: showTree,
|
||||||
collapsedPaths: map[string]bool{},
|
collapsedPaths: CollapsedPaths{},
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *FileChangeManager) ExpandToPath(path string) {
|
||||||
|
m.collapsedPaths.ExpandToPath(path)
|
||||||
|
}
|
||||||
|
|
||||||
func (m *FileChangeManager) ToggleShowTree() {
|
func (m *FileChangeManager) ToggleShowTree() {
|
||||||
m.showTree = !m.showTree
|
m.showTree = !m.showTree
|
||||||
m.SetTree()
|
m.SetTree()
|
||||||
@ -80,74 +72,17 @@ func (m *FileChangeManager) SetTree() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (m *FileChangeManager) IsCollapsed(path string) bool {
|
||||||
|
return m.collapsedPaths.IsCollapsed(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *FileChangeManager) ToggleCollapsed(path string) {
|
||||||
|
m.collapsedPaths.ToggleCollapsed(path)
|
||||||
|
}
|
||||||
|
|
||||||
func (m *FileChangeManager) Render(diffName string, submoduleConfigs []*models.SubmoduleConfig) []string {
|
func (m *FileChangeManager) Render(diffName string, submoduleConfigs []*models.SubmoduleConfig) []string {
|
||||||
return m.renderAux(m.tree, "", -1, diffName, submoduleConfigs)
|
return renderAux(m.tree, m.collapsedPaths, "", -1, func(n INode, depth int) string {
|
||||||
}
|
castN := n.(*FileChangeNode)
|
||||||
|
return presentation.GetFileLine(castN.GetHasUnstagedChanges(), castN.GetHasStagedChanges(), castN.NameAtDepth(depth), diffName, submoduleConfigs, castN.File)
|
||||||
func (m *FileChangeManager) IsCollapsed(s *FileChangeNode) bool {
|
})
|
||||||
return m.collapsedPaths[s.GetPath()]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *FileChangeManager) ToggleCollapsed(s *FileChangeNode) {
|
|
||||||
m.collapsedPaths[s.GetPath()] = !m.collapsedPaths[s.GetPath()]
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *FileChangeManager) renderAux(s *FileChangeNode, prefix string, depth int, diffName string, submoduleConfigs []*models.SubmoduleConfig) []string {
|
|
||||||
isRoot := depth == -1
|
|
||||||
if s == nil {
|
|
||||||
return []string{}
|
|
||||||
}
|
|
||||||
|
|
||||||
getLine := func() string {
|
|
||||||
return prefix + presentation.GetFileLine(s.GetHasUnstagedChanges(), s.GetHasStagedChanges(), s.NameAtDepth(depth), diffName, submoduleConfigs, s.File)
|
|
||||||
}
|
|
||||||
|
|
||||||
if s.IsLeaf() {
|
|
||||||
if isRoot {
|
|
||||||
return []string{}
|
|
||||||
}
|
|
||||||
return []string{getLine()}
|
|
||||||
}
|
|
||||||
|
|
||||||
if m.IsCollapsed(s) {
|
|
||||||
return []string{fmt.Sprintf("%s %s", getLine(), COLLAPSED_ARROW)}
|
|
||||||
}
|
|
||||||
|
|
||||||
arr := []string{}
|
|
||||||
if !isRoot {
|
|
||||||
arr = append(arr, fmt.Sprintf("%s %s", getLine(), 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.Children {
|
|
||||||
isLast := i == len(s.Children)-1
|
|
||||||
|
|
||||||
var childPrefix string
|
|
||||||
if isRoot {
|
|
||||||
childPrefix = newPrefix
|
|
||||||
} else if isLast {
|
|
||||||
childPrefix = newPrefix + LAST_ITEM
|
|
||||||
} else {
|
|
||||||
childPrefix = newPrefix + INNER_ITEM
|
|
||||||
}
|
|
||||||
|
|
||||||
arr = append(arr, m.renderAux(child, childPrefix, depth+1+s.CompressionLevel, diffName, submoduleConfigs)...)
|
|
||||||
}
|
|
||||||
|
|
||||||
return arr
|
|
||||||
}
|
|
||||||
|
|
||||||
func (m *FileChangeManager) ExpandToPath(path string) {
|
|
||||||
// need every directory along the way
|
|
||||||
split := strings.Split(path, string(os.PathSeparator))
|
|
||||||
for i := range split {
|
|
||||||
dir := strings.Join(split[0:i+1], string(os.PathSeparator))
|
|
||||||
m.collapsedPaths[dir] = false
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,10 @@
|
|||||||
package filetree
|
package filetree
|
||||||
|
|
||||||
import "sort"
|
import (
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
)
|
||||||
|
|
||||||
type INode interface {
|
type INode interface {
|
||||||
IsLeaf() bool
|
IsLeaf() bool
|
||||||
@ -194,3 +198,54 @@ func getLeaves(node INode) []INode {
|
|||||||
|
|
||||||
return output
|
return output
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func renderAux(s INode, collapsedPaths CollapsedPaths, prefix string, depth int, renderLine func(INode, int) string) []string {
|
||||||
|
isRoot := depth == -1
|
||||||
|
if s == nil {
|
||||||
|
return []string{}
|
||||||
|
}
|
||||||
|
|
||||||
|
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
|
||||||
|
}
|
||||||
|
@ -17,12 +17,12 @@ func (gui *Gui) handleCreateFilteringMenuPanel(g *gocui.Gui, v *gocui.View) erro
|
|||||||
case "files":
|
case "files":
|
||||||
node := gui.getSelectedFileChangeNode()
|
node := gui.getSelectedFileChangeNode()
|
||||||
if node != nil {
|
if node != nil {
|
||||||
fileName = node.Path
|
fileName = node.GetPath()
|
||||||
}
|
}
|
||||||
case "commitFiles":
|
case "commitFiles":
|
||||||
file := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if file != nil {
|
if node != nil {
|
||||||
fileName = file.Name
|
fileName = node.GetPath()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -305,11 +305,11 @@ type guiStateMutexes struct {
|
|||||||
|
|
||||||
type guiState struct {
|
type guiState struct {
|
||||||
FileChangeManager *filetree.FileChangeManager
|
FileChangeManager *filetree.FileChangeManager
|
||||||
|
CommitFileChangeManager *filetree.CommitFileChangeManager
|
||||||
Submodules []*models.SubmoduleConfig
|
Submodules []*models.SubmoduleConfig
|
||||||
Branches []*models.Branch
|
Branches []*models.Branch
|
||||||
Commits []*models.Commit
|
Commits []*models.Commit
|
||||||
StashEntries []*models.StashEntry
|
StashEntries []*models.StashEntry
|
||||||
CommitFiles []*models.CommitFile
|
|
||||||
// Suggestions will sometimes appear when typing into a prompt
|
// Suggestions will sometimes appear when typing into a prompt
|
||||||
Suggestions []*types.Suggestion
|
Suggestions []*types.Suggestion
|
||||||
// FilteredReflogCommits are the ones that appear in the reflog panel.
|
// FilteredReflogCommits are the ones that appear in the reflog panel.
|
||||||
@ -382,6 +382,7 @@ func (gui *Gui) resetState() {
|
|||||||
|
|
||||||
gui.State = &guiState{
|
gui.State = &guiState{
|
||||||
FileChangeManager: filetree.NewFileChangeManager(make([]*models.File, 0), gui.Log, showTree),
|
FileChangeManager: filetree.NewFileChangeManager(make([]*models.File, 0), gui.Log, showTree),
|
||||||
|
CommitFileChangeManager: filetree.NewCommitFileChangeManager(make([]*models.CommitFile, 0), gui.Log, showTree),
|
||||||
Commits: make([]*models.Commit, 0),
|
Commits: make([]*models.Commit, 0),
|
||||||
FilteredReflogCommits: make([]*models.Commit, 0),
|
FilteredReflogCommits: make([]*models.Commit, 0),
|
||||||
ReflogCommits: make([]*models.Commit, 0),
|
ReflogCommits: make([]*models.Commit, 0),
|
||||||
|
@ -227,7 +227,9 @@ func (gui *Gui) handleMouseScrollDown(g *gocui.Gui, v *gocui.View) error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) getSelectedCommitFileName() string {
|
func (gui *Gui) getSelectedCommitFileName() string {
|
||||||
return gui.State.CommitFiles[gui.State.Panels.CommitFiles.SelectedLineIdx].Name
|
idx := gui.State.Panels.CommitFiles.SelectedLineIdx
|
||||||
|
|
||||||
|
return gui.State.CommitFileChangeManager.GetItemAtIndex(idx).GetPath()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (gui *Gui) refreshMainViewForLineByLine(state *lBlPanelState) error {
|
func (gui *Gui) refreshMainViewForLineByLine(state *lBlPanelState) error {
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
package gui
|
package gui
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"github.com/fatih/color"
|
||||||
"github.com/jesseduffield/gocui"
|
"github.com/jesseduffield/gocui"
|
||||||
"github.com/jesseduffield/lazygit/pkg/gui/presentation"
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation"
|
||||||
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
type ListContext struct {
|
type ListContext struct {
|
||||||
@ -452,17 +454,27 @@ func (gui *Gui) commitFilesListContext() *ListContext {
|
|||||||
ViewName: "commitFiles",
|
ViewName: "commitFiles",
|
||||||
WindowName: "commits",
|
WindowName: "commits",
|
||||||
ContextKey: COMMIT_FILES_CONTEXT_KEY,
|
ContextKey: COMMIT_FILES_CONTEXT_KEY,
|
||||||
GetItemsLength: func() int { return len(gui.State.CommitFiles) },
|
GetItemsLength: func() int { return gui.State.CommitFileChangeManager.GetItemsLength() },
|
||||||
GetPanelState: func() IListPanelState { return gui.State.Panels.CommitFiles },
|
GetPanelState: func() IListPanelState { return gui.State.Panels.CommitFiles },
|
||||||
OnFocus: gui.handleCommitFileSelect,
|
OnFocus: gui.handleCommitFileSelect,
|
||||||
Gui: gui,
|
Gui: gui,
|
||||||
ResetMainViewOriginOnFocus: true,
|
ResetMainViewOriginOnFocus: true,
|
||||||
Kind: SIDE_CONTEXT,
|
Kind: SIDE_CONTEXT,
|
||||||
GetDisplayStrings: func() [][]string {
|
GetDisplayStrings: func() [][]string {
|
||||||
return presentation.GetCommitFileListDisplayStrings(gui.State.CommitFiles, gui.State.Modes.Diffing.Ref)
|
if gui.State.CommitFileChangeManager.GetItemsLength() == 0 {
|
||||||
|
return [][]string{{utils.ColoredString("(none)", color.FgRed)}}
|
||||||
|
}
|
||||||
|
|
||||||
|
lines := gui.State.CommitFileChangeManager.Render(gui.State.Modes.Diffing.Ref)
|
||||||
|
mappedLines := make([][]string, len(lines))
|
||||||
|
for i, line := range lines {
|
||||||
|
mappedLines[i] = []string{line}
|
||||||
|
}
|
||||||
|
|
||||||
|
return mappedLines
|
||||||
},
|
},
|
||||||
SelectedItem: func() (ListItem, bool) {
|
SelectedItem: func() (ListItem, bool) {
|
||||||
item := gui.getSelectedCommitFile()
|
item := gui.getSelectedCommitFileNode()
|
||||||
return item, item != nil
|
return item, item != nil
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -28,19 +28,19 @@ func (gui *Gui) refreshPatchBuildingPanel(selectedLineIdx int, state *lBlPanelSt
|
|||||||
gui.getSecondaryView().Title = "Custom Patch"
|
gui.getSecondaryView().Title = "Custom Patch"
|
||||||
|
|
||||||
// get diff from commit file that's currently selected
|
// get diff from commit file that's currently selected
|
||||||
commitFile := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if commitFile == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
to := commitFile.Parent
|
to := gui.State.CommitFileChangeManager.GetParent()
|
||||||
from, reverse := gui.getFromAndReverseArgsForDiff(to)
|
from, reverse := gui.getFromAndReverseArgsForDiff(to)
|
||||||
diff, err := gui.GitCommand.ShowFileDiff(from, to, reverse, commitFile.Name, true)
|
diff, err := gui.GitCommand.ShowFileDiff(from, to, reverse, node.GetPath(), true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
secondaryDiff := gui.GitCommand.PatchManager.RenderPatchForFile(commitFile.Name, true, false, true)
|
secondaryDiff := gui.GitCommand.PatchManager.RenderPatchForFile(node.GetPath(), true, false, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
@ -78,12 +78,12 @@ func (gui *Gui) handleToggleSelectionForPatch() error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add range of lines to those set for the file
|
// add range of lines to those set for the file
|
||||||
commitFile := gui.getSelectedCommitFile()
|
node := gui.getSelectedCommitFileNode()
|
||||||
if commitFile == nil {
|
if node == nil {
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
if err := toggleFunc(commitFile.Name, state.FirstLineIdx, state.LastLineIdx); err != nil {
|
if err := toggleFunc(node.GetPath(), state.FirstLineIdx, state.LastLineIdx); err != nil {
|
||||||
// might actually want to return an error here
|
// might actually want to return an error here
|
||||||
gui.Log.Error(err)
|
gui.Log.Error(err)
|
||||||
}
|
}
|
||||||
|
@ -8,30 +8,21 @@ import (
|
|||||||
"github.com/jesseduffield/lazygit/pkg/utils"
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
||||||
)
|
)
|
||||||
|
|
||||||
func GetCommitFileListDisplayStrings(commitFiles []*models.CommitFile, diffName string) [][]string {
|
func GetCommitFileLine(name string, diffName string, commitFile *models.CommitFile) string {
|
||||||
if len(commitFiles) == 0 {
|
|
||||||
return [][]string{{utils.ColoredString("(none)", color.FgRed)}}
|
|
||||||
}
|
|
||||||
|
|
||||||
lines := make([][]string, len(commitFiles))
|
|
||||||
|
|
||||||
for i := range commitFiles {
|
|
||||||
diffed := commitFiles[i].Name == diffName
|
|
||||||
lines[i] = getCommitFileDisplayStrings(commitFiles[i], diffed)
|
|
||||||
}
|
|
||||||
|
|
||||||
return lines
|
|
||||||
}
|
|
||||||
|
|
||||||
// getCommitFileDisplayStrings returns the display string of branch
|
|
||||||
func getCommitFileDisplayStrings(f *models.CommitFile, diffed bool) []string {
|
|
||||||
yellow := color.New(color.FgYellow)
|
yellow := color.New(color.FgYellow)
|
||||||
green := color.New(color.FgGreen)
|
green := color.New(color.FgGreen)
|
||||||
defaultColor := color.New(theme.DefaultTextColor)
|
defaultColor := color.New(theme.DefaultTextColor)
|
||||||
diffTerminalColor := color.New(theme.DiffTerminalColor)
|
diffTerminalColor := color.New(theme.DiffTerminalColor)
|
||||||
|
|
||||||
var colour *color.Color
|
if commitFile == nil {
|
||||||
switch f.PatchStatus {
|
return name
|
||||||
|
}
|
||||||
|
|
||||||
|
colour := defaultColor
|
||||||
|
if diffName == name {
|
||||||
|
colour = diffTerminalColor
|
||||||
|
} else if commitFile != nil {
|
||||||
|
switch commitFile.PatchStatus {
|
||||||
case patch.UNSELECTED:
|
case patch.UNSELECTED:
|
||||||
colour = defaultColor
|
colour = defaultColor
|
||||||
case patch.WHOLE:
|
case patch.WHOLE:
|
||||||
@ -39,10 +30,9 @@ func getCommitFileDisplayStrings(f *models.CommitFile, diffed bool) []string {
|
|||||||
case patch.PART:
|
case patch.PART:
|
||||||
colour = yellow
|
colour = yellow
|
||||||
}
|
}
|
||||||
if diffed {
|
|
||||||
colour = diffTerminalColor
|
|
||||||
}
|
}
|
||||||
return []string{utils.ColoredString(f.ChangeStatus, getColorForChangeStatus(f.ChangeStatus)), colour.Sprint(f.Name)}
|
|
||||||
|
return utils.ColoredString(commitFile.ChangeStatus, getColorForChangeStatus(commitFile.ChangeStatus)) + " " + colour.Sprint(name)
|
||||||
}
|
}
|
||||||
|
|
||||||
func getColorForChangeStatus(changeStatus string) color.Attribute {
|
func getColorForChangeStatus(changeStatus string) color.Attribute {
|
||||||
|
Reference in New Issue
Block a user