mirror of
https://github.com/jesseduffield/lazygit.git
synced 2025-01-24 05:36:19 +02:00
178 lines
4.2 KiB
Go
178 lines
4.2 KiB
Go
package presentation
|
|
|
|
import (
|
|
"strings"
|
|
"sync"
|
|
|
|
"github.com/jesseduffield/lazygit/pkg/commands/models"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation/authors"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/presentation/graph"
|
|
"github.com/jesseduffield/lazygit/pkg/gui/style"
|
|
"github.com/jesseduffield/lazygit/pkg/theme"
|
|
"github.com/jesseduffield/lazygit/pkg/utils"
|
|
"github.com/kyokomi/emoji/v2"
|
|
)
|
|
|
|
type pipeSetCacheKey struct {
|
|
commitSha string
|
|
commitCount int
|
|
}
|
|
|
|
var pipeSetCache = make(map[pipeSetCacheKey][][]*graph.Pipe)
|
|
var mutex sync.Mutex
|
|
|
|
func GetCommitListDisplayStrings(
|
|
commits []*models.Commit,
|
|
fullDescription bool,
|
|
cherryPickedCommitShaMap map[string]bool,
|
|
diffName string,
|
|
parseEmoji bool,
|
|
selectedCommitSha string,
|
|
startIdx int,
|
|
length int,
|
|
showGraph bool,
|
|
) [][]string {
|
|
mutex.Lock()
|
|
defer mutex.Unlock()
|
|
|
|
if len(commits) == 0 {
|
|
return nil
|
|
}
|
|
|
|
// given that our cache key is a commit sha and a commit count, it's very important that we don't actually try to render pipes
|
|
// when dealing with things like filtered commits.
|
|
cacheKey := pipeSetCacheKey{
|
|
commitSha: commits[0].Sha,
|
|
commitCount: len(commits),
|
|
}
|
|
|
|
pipeSets, ok := pipeSetCache[cacheKey]
|
|
if !ok {
|
|
// pipe sets are unique to a commit head. and a commit count. Sometimes we haven't loaded everything for that.
|
|
// so let's just cache it based on that.
|
|
getStyle := func(commit *models.Commit) style.TextStyle {
|
|
return authors.AuthorStyle(commit.Author)
|
|
}
|
|
pipeSets = graph.GetPipeSets(commits, getStyle)
|
|
pipeSetCache[cacheKey] = pipeSets
|
|
}
|
|
|
|
if startIdx > len(commits) {
|
|
return nil
|
|
}
|
|
end := startIdx + length
|
|
if end > len(commits)-1 {
|
|
end = len(commits) - 1
|
|
}
|
|
|
|
filteredCommits := commits[startIdx : end+1]
|
|
|
|
var getGraphLine func(int) string
|
|
if showGraph {
|
|
filteredPipeSets := pipeSets[startIdx : end+1]
|
|
graphLines := graph.RenderAux(filteredPipeSets, filteredCommits, selectedCommitSha)
|
|
getGraphLine = func(idx int) string { return graphLines[idx] }
|
|
} else {
|
|
getGraphLine = func(idx int) string { return "" }
|
|
}
|
|
|
|
lines := make([][]string, 0, len(filteredCommits))
|
|
for i, commit := range filteredCommits {
|
|
lines = append(lines, displayCommit(commit, cherryPickedCommitShaMap, diffName, parseEmoji, getGraphLine(i), fullDescription))
|
|
}
|
|
return lines
|
|
}
|
|
|
|
func displayCommit(
|
|
commit *models.Commit,
|
|
cherryPickedCommitShaMap map[string]bool,
|
|
diffName string,
|
|
parseEmoji bool,
|
|
graphLine string,
|
|
fullDescription bool,
|
|
) []string {
|
|
|
|
shaColor := getShaColor(commit, diffName, cherryPickedCommitShaMap)
|
|
|
|
actionString := ""
|
|
if commit.Action != "" {
|
|
actionString = actionColorMap(commit.Action).Sprint(commit.Action) + " "
|
|
}
|
|
|
|
tagString := ""
|
|
if fullDescription {
|
|
if commit.ExtraInfo != "" {
|
|
tagString = style.FgMagenta.SetBold().Sprint(commit.ExtraInfo) + " "
|
|
}
|
|
} else {
|
|
if len(commit.Tags) > 0 {
|
|
tagString = theme.DiffTerminalColor.SetBold().Sprint(strings.Join(commit.Tags, " ")) + " "
|
|
}
|
|
}
|
|
|
|
name := commit.Name
|
|
if parseEmoji {
|
|
name = emoji.Sprint(name)
|
|
}
|
|
|
|
authorFunc := authors.ShortAuthor
|
|
if fullDescription {
|
|
authorFunc = authors.LongAuthor
|
|
}
|
|
|
|
cols := make([]string, 0, 5)
|
|
cols = append(cols, shaColor.Sprint(commit.ShortSha()))
|
|
if fullDescription {
|
|
cols = append(cols, style.FgBlue.Sprint(utils.UnixToDate(commit.UnixTimestamp)))
|
|
}
|
|
cols = append(
|
|
cols,
|
|
actionString,
|
|
authorFunc(commit.Author),
|
|
graphLine+tagString+theme.DefaultTextColor.Sprint(name),
|
|
)
|
|
|
|
return cols
|
|
|
|
}
|
|
|
|
func getShaColor(commit *models.Commit, diffName string, cherryPickedCommitShaMap map[string]bool) style.TextStyle {
|
|
diffed := commit.Sha == diffName
|
|
shaColor := theme.DefaultTextColor
|
|
switch commit.Status {
|
|
case "unpushed":
|
|
shaColor = style.FgRed
|
|
case "pushed":
|
|
shaColor = style.FgYellow
|
|
case "merged":
|
|
shaColor = style.FgGreen
|
|
case "rebasing":
|
|
shaColor = style.FgBlue
|
|
case "reflog":
|
|
shaColor = style.FgBlue
|
|
}
|
|
|
|
if diffed {
|
|
shaColor = theme.DiffTerminalColor
|
|
} else if cherryPickedCommitShaMap[commit.Sha] {
|
|
shaColor = theme.CherryPickedCommitTextStyle
|
|
}
|
|
|
|
return shaColor
|
|
}
|
|
|
|
func actionColorMap(str string) style.TextStyle {
|
|
switch str {
|
|
case "pick":
|
|
return style.FgCyan
|
|
case "drop":
|
|
return style.FgRed
|
|
case "edit":
|
|
return style.FgGreen
|
|
case "fixup":
|
|
return style.FgMagenta
|
|
default:
|
|
return style.FgYellow
|
|
}
|
|
}
|