1
0
mirror of https://github.com/jesseduffield/lazygit.git synced 2025-04-04 22:34:39 +02:00

Add SpinnerConfig

This new config section allows to customize frames and rate of
thespinner
This commit is contained in:
Artem Belyakov 2024-04-06 13:35:50 +02:00 committed by Stefan Haller
parent 53f0c4aeff
commit f3dba743f0
15 changed files with 101 additions and 31 deletions

View File

@ -87,6 +87,9 @@ gui:
animateExplosion: true # shows an explosion animation when nuking the working tree animateExplosion: true # shows an explosion animation when nuking the working tree
portraitMode: 'auto' # one of 'auto' | 'never' | 'always' portraitMode: 'auto' # one of 'auto' | 'never' | 'always'
filterMode: 'substring' # one of 'substring' | 'fuzzy'; see 'Filtering' section below filterMode: 'substring' # one of 'substring' | 'fuzzy'; see 'Filtering' section below
spinner:
frames: ['|', '/', '-', '\\']
rate: 50 # spinner rate in milliseconds
git: git:
paging: paging:
colorArg: always colorArg: always

View File

@ -145,6 +145,8 @@ type GuiConfig struct {
// How things are filtered when typing '/'. // How things are filtered when typing '/'.
// One of 'substring' (default) | 'fuzzy' // One of 'substring' (default) | 'fuzzy'
FilterMode string `yaml:"filterMode" jsonschema:"enum=substring,enum=fuzzy"` FilterMode string `yaml:"filterMode" jsonschema:"enum=substring,enum=fuzzy"`
// Config relating to the spinner.
Spinner SpinnerConfig `yaml:"spinner"`
} }
func (c *GuiConfig) UseFuzzySearch() bool { func (c *GuiConfig) UseFuzzySearch() bool {
@ -182,6 +184,13 @@ type CommitLengthConfig struct {
Show bool `yaml:"show"` Show bool `yaml:"show"`
} }
type SpinnerConfig struct {
// The frames of the spinner animation.
Frames []string `yaml:"frames"`
// The "speed" of the spinner in milliseconds.
Rate int `yaml:"rate" jsonschema:"minimum=1"`
}
type GitConfig struct { type GitConfig struct {
// See https://github.com/jesseduffield/lazygit/blob/master/docs/Custom_Pagers.md // See https://github.com/jesseduffield/lazygit/blob/master/docs/Custom_Pagers.md
Paging PagingConfig `yaml:"paging"` Paging PagingConfig `yaml:"paging"`
@ -671,6 +680,10 @@ func GetDefaultConfig() *UserConfig {
AnimateExplosion: true, AnimateExplosion: true,
PortraitMode: "auto", PortraitMode: "auto",
FilterMode: "substring", FilterMode: "substring",
Spinner: SpinnerConfig{
Frames: []string{"|", "/", "-", "\\"},
Rate: 50,
},
}, },
Git: GitConfig{ Git: GitConfig{
Paging: PagingConfig{ Paging: PagingConfig{

View File

@ -26,7 +26,7 @@ func NewRemotesContext(c *ContextCommon) *RemotesContext {
getDisplayStrings := func(_ int, _ int) [][]string { getDisplayStrings := func(_ int, _ int) [][]string {
return presentation.GetRemoteListDisplayStrings( return presentation.GetRemoteListDisplayStrings(
viewModel.GetItems(), c.Modes().Diffing.Ref, c.State().GetItemOperation, c.Tr) viewModel.GetItems(), c.Modes().Diffing.Ref, c.State().GetItemOperation, c.Tr, c.UserConfig)
} }
return &RemotesContext{ return &RemotesContext{

View File

@ -30,7 +30,7 @@ func NewTagsContext(
return presentation.GetTagListDisplayStrings( return presentation.GetTagListDisplayStrings(
viewModel.GetItems(), viewModel.GetItems(),
c.State().GetItemOperation, c.State().GetItemOperation,
c.Modes().Diffing.Ref, c.Tr) c.Modes().Diffing.Ref, c.Tr, c.UserConfig)
} }
return &TagsContext{ return &TagsContext{

View File

@ -6,7 +6,6 @@ import (
"github.com/jesseduffield/gocui" "github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/gui/status" "github.com/jesseduffield/lazygit/pkg/gui/status"
"github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/gui/types"
"github.com/jesseduffield/lazygit/pkg/utils"
) )
type AppStatusHelper struct { type AppStatusHelper struct {
@ -88,16 +87,16 @@ func (self *AppStatusHelper) HasStatus() bool {
} }
func (self *AppStatusHelper) GetStatusString() string { func (self *AppStatusHelper) GetStatusString() string {
appStatus, _ := self.statusMgr().GetStatusString() appStatus, _ := self.statusMgr().GetStatusString(self.c.UserConfig)
return appStatus return appStatus
} }
func (self *AppStatusHelper) renderAppStatus() { func (self *AppStatusHelper) renderAppStatus() {
self.c.OnWorker(func(_ gocui.Task) { self.c.OnWorker(func(_ gocui.Task) {
ticker := time.NewTicker(time.Millisecond * utils.LoaderAnimationInterval) ticker := time.NewTicker(time.Millisecond * time.Duration(self.c.UserConfig.Gui.Spinner.Rate))
defer ticker.Stop() defer ticker.Stop()
for range ticker.C { for range ticker.C {
appStatus, color := self.statusMgr().GetStatusString() appStatus, color := self.statusMgr().GetStatusString(self.c.UserConfig)
self.c.Views().AppStatus.FgColor = color self.c.Views().AppStatus.FgColor = color
self.c.OnUIThread(func() error { self.c.OnUIThread(func() error {
self.c.SetViewContent(self.c.Views().AppStatus, appStatus) self.c.SetViewContent(self.c.Views().AppStatus, appStatus)
@ -130,7 +129,7 @@ func (self *AppStatusHelper) renderAppStatusSync(stop chan struct{}) {
for { for {
select { select {
case <-ticker.C: case <-ticker.C:
appStatus, color := self.statusMgr().GetStatusString() appStatus, color := self.statusMgr().GetStatusString(self.c.UserConfig)
self.c.Views().AppStatus.FgColor = color self.c.Views().AppStatus.FgColor = color
self.c.SetViewContent(self.c.Views().AppStatus, appStatus) self.c.SetViewContent(self.c.Views().AppStatus, appStatus)
// Redraw all views of the bottom line: // Redraw all views of the bottom line:

View File

@ -105,7 +105,7 @@ func (self *InlineStatusHelper) start(opts InlineStatusOpts) {
self.contextsWithInlineStatus[opts.ContextKey] = info self.contextsWithInlineStatus[opts.ContextKey] = info
go utils.Safe(func() { go utils.Safe(func() {
ticker := time.NewTicker(time.Millisecond * utils.LoaderAnimationInterval) ticker := time.NewTicker(time.Millisecond * time.Duration(self.c.UserConfig.Gui.Spinner.Rate))
defer ticker.Stop() defer ticker.Stop()
outer: outer:
for { for {

View File

@ -706,7 +706,7 @@ func (self *RefreshHelper) refreshStatus() {
repoName := self.c.Git().RepoPaths.RepoName() repoName := self.c.Git().RepoPaths.RepoName()
status := presentation.FormatStatus(repoName, currentBranch, types.ItemOperationNone, linkedWorktreeName, workingTreeState, self.c.Tr) status := presentation.FormatStatus(repoName, currentBranch, types.ItemOperationNone, linkedWorktreeName, workingTreeState, self.c.Tr, self.c.UserConfig)
self.c.SetViewContent(self.c.Views().Status, status) self.c.SetViewContent(self.c.Views().Status, status)
} }

View File

@ -136,7 +136,7 @@ func (self *StatusController) onClick() error {
} }
cx, _ := self.c.Views().Status.Cursor() cx, _ := self.c.Views().Status.Cursor()
upstreamStatus := presentation.BranchStatus(currentBranch, types.ItemOperationNone, self.c.Tr, time.Now()) upstreamStatus := presentation.BranchStatus(currentBranch, types.ItemOperationNone, self.c.Tr, time.Now(), self.c.UserConfig)
repoName := self.c.Git().RepoPaths.RepoName() repoName := self.c.Git().RepoPaths.RepoName()
workingTreeState := self.c.Git().Status.WorkingTreeState() workingTreeState := self.c.Git().Status.WorkingTreeState()
switch workingTreeState { switch workingTreeState {

View File

@ -50,7 +50,7 @@ func getBranchDisplayStrings(
) []string { ) []string {
checkedOutByWorkTree := git_commands.CheckedOutByOtherWorktree(b, worktrees) checkedOutByWorkTree := git_commands.CheckedOutByOtherWorktree(b, worktrees)
showCommitHash := fullDescription || userConfig.Gui.ShowBranchCommitHash showCommitHash := fullDescription || userConfig.Gui.ShowBranchCommitHash
branchStatus := BranchStatus(b, itemOperation, tr, now) branchStatus := BranchStatus(b, itemOperation, tr, now, userConfig)
worktreeIcon := lo.Ternary(icons.IsIconEnabled(), icons.LINKED_WORKTREE_ICON, fmt.Sprintf("(%s)", tr.LcWorktree)) worktreeIcon := lo.Ternary(icons.IsIconEnabled(), icons.LINKED_WORKTREE_ICON, fmt.Sprintf("(%s)", tr.LcWorktree))
// Recency is always three characters, plus one for the space // Recency is always three characters, plus one for the space
@ -159,14 +159,25 @@ func branchStatusColor(branch *models.Branch, itemOperation types.ItemOperation)
return colour return colour
} }
func ColoredBranchStatus(branch *models.Branch, itemOperation types.ItemOperation, tr *i18n.TranslationSet) string { func ColoredBranchStatus(
return branchStatusColor(branch, itemOperation).Sprint(BranchStatus(branch, itemOperation, tr, time.Now())) branch *models.Branch,
itemOperation types.ItemOperation,
tr *i18n.TranslationSet,
userConfig *config.UserConfig,
) string {
return branchStatusColor(branch, itemOperation).Sprint(BranchStatus(branch, itemOperation, tr, time.Now(), userConfig))
} }
func BranchStatus(branch *models.Branch, itemOperation types.ItemOperation, tr *i18n.TranslationSet, now time.Time) string { func BranchStatus(
branch *models.Branch,
itemOperation types.ItemOperation,
tr *i18n.TranslationSet,
now time.Time,
userConfig *config.UserConfig,
) string {
itemOperationStr := ItemOperationToString(itemOperation, tr) itemOperationStr := ItemOperationToString(itemOperation, tr)
if itemOperationStr != "" { if itemOperationStr != "" {
return itemOperationStr + " " + utils.Loader(now) return itemOperationStr + " " + utils.Loader(now, userConfig.Gui.Spinner)
} }
if !branch.IsTrackingRemote() { if !branch.IsTrackingRemote() {

View File

@ -4,6 +4,7 @@ import (
"time" "time"
"github.com/jesseduffield/lazygit/pkg/commands/models" "github.com/jesseduffield/lazygit/pkg/commands/models"
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/gui/presentation/icons" "github.com/jesseduffield/lazygit/pkg/gui/presentation/icons"
"github.com/jesseduffield/lazygit/pkg/gui/style" "github.com/jesseduffield/lazygit/pkg/gui/style"
"github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/gui/types"
@ -18,10 +19,11 @@ func GetRemoteListDisplayStrings(
diffName string, diffName string,
getItemOperation func(item types.HasUrn) types.ItemOperation, getItemOperation func(item types.HasUrn) types.ItemOperation,
tr *i18n.TranslationSet, tr *i18n.TranslationSet,
userConfig *config.UserConfig,
) [][]string { ) [][]string {
return lo.Map(remotes, func(remote *models.Remote, _ int) []string { return lo.Map(remotes, func(remote *models.Remote, _ int) []string {
diffed := remote.Name == diffName diffed := remote.Name == diffName
return getRemoteDisplayStrings(remote, diffed, getItemOperation(remote), tr) return getRemoteDisplayStrings(remote, diffed, getItemOperation(remote), tr, userConfig)
}) })
} }
@ -31,6 +33,7 @@ func getRemoteDisplayStrings(
diffed bool, diffed bool,
itemOperation types.ItemOperation, itemOperation types.ItemOperation,
tr *i18n.TranslationSet, tr *i18n.TranslationSet,
userConfig *config.UserConfig,
) []string { ) []string {
branchCount := len(r.Branches) branchCount := len(r.Branches)
@ -46,7 +49,7 @@ func getRemoteDisplayStrings(
descriptionStr := style.FgBlue.Sprintf("%d branches", branchCount) descriptionStr := style.FgBlue.Sprintf("%d branches", branchCount)
itemOperationStr := ItemOperationToString(itemOperation, tr) itemOperationStr := ItemOperationToString(itemOperation, tr)
if itemOperationStr != "" { if itemOperationStr != "" {
descriptionStr += " " + style.FgCyan.Sprint(itemOperationStr+" "+utils.Loader(time.Now())) descriptionStr += " " + style.FgCyan.Sprint(itemOperationStr+" "+utils.Loader(time.Now(), userConfig.Gui.Spinner))
} }
res = append(res, textStyle.Sprint(r.Name), descriptionStr) res = append(res, textStyle.Sprint(r.Name), descriptionStr)
return res return res

View File

@ -5,17 +5,26 @@ import (
"github.com/jesseduffield/lazygit/pkg/commands/models" "github.com/jesseduffield/lazygit/pkg/commands/models"
"github.com/jesseduffield/lazygit/pkg/commands/types/enums" "github.com/jesseduffield/lazygit/pkg/commands/types/enums"
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/gui/presentation/icons" "github.com/jesseduffield/lazygit/pkg/gui/presentation/icons"
"github.com/jesseduffield/lazygit/pkg/gui/style" "github.com/jesseduffield/lazygit/pkg/gui/style"
"github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/gui/types"
"github.com/jesseduffield/lazygit/pkg/i18n" "github.com/jesseduffield/lazygit/pkg/i18n"
) )
func FormatStatus(repoName string, currentBranch *models.Branch, itemOperation types.ItemOperation, linkedWorktreeName string, workingTreeState enums.RebaseMode, tr *i18n.TranslationSet) string { func FormatStatus(
repoName string,
currentBranch *models.Branch,
itemOperation types.ItemOperation,
linkedWorktreeName string,
workingTreeState enums.RebaseMode,
tr *i18n.TranslationSet,
userConfig *config.UserConfig,
) string {
status := "" status := ""
if currentBranch.IsRealBranch() { if currentBranch.IsRealBranch() {
status += ColoredBranchStatus(currentBranch, itemOperation, tr) + " " status += ColoredBranchStatus(currentBranch, itemOperation, tr, userConfig) + " "
} }
if workingTreeState != enums.REBASE_MODE_NONE { if workingTreeState != enums.REBASE_MODE_NONE {

View File

@ -4,6 +4,7 @@ import (
"time" "time"
"github.com/jesseduffield/lazygit/pkg/commands/models" "github.com/jesseduffield/lazygit/pkg/commands/models"
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/gui/presentation/icons" "github.com/jesseduffield/lazygit/pkg/gui/presentation/icons"
"github.com/jesseduffield/lazygit/pkg/gui/style" "github.com/jesseduffield/lazygit/pkg/gui/style"
"github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/gui/types"
@ -18,15 +19,22 @@ func GetTagListDisplayStrings(
getItemOperation func(item types.HasUrn) types.ItemOperation, getItemOperation func(item types.HasUrn) types.ItemOperation,
diffName string, diffName string,
tr *i18n.TranslationSet, tr *i18n.TranslationSet,
userConfig *config.UserConfig,
) [][]string { ) [][]string {
return lo.Map(tags, func(tag *models.Tag, _ int) []string { return lo.Map(tags, func(tag *models.Tag, _ int) []string {
diffed := tag.Name == diffName diffed := tag.Name == diffName
return getTagDisplayStrings(tag, getItemOperation(tag), diffed, tr) return getTagDisplayStrings(tag, getItemOperation(tag), diffed, tr, userConfig)
}) })
} }
// getTagDisplayStrings returns the display string of branch // getTagDisplayStrings returns the display string of branch
func getTagDisplayStrings(t *models.Tag, itemOperation types.ItemOperation, diffed bool, tr *i18n.TranslationSet) []string { func getTagDisplayStrings(
t *models.Tag,
itemOperation types.ItemOperation,
diffed bool,
tr *i18n.TranslationSet,
userConfig *config.UserConfig,
) []string {
textStyle := theme.DefaultTextColor textStyle := theme.DefaultTextColor
if diffed { if diffed {
textStyle = theme.DiffTerminalColor textStyle = theme.DiffTerminalColor
@ -39,7 +47,7 @@ func getTagDisplayStrings(t *models.Tag, itemOperation types.ItemOperation, diff
descriptionStr := descriptionColor.Sprint(t.Description()) descriptionStr := descriptionColor.Sprint(t.Description())
itemOperationStr := ItemOperationToString(itemOperation, tr) itemOperationStr := ItemOperationToString(itemOperation, tr)
if itemOperationStr != "" { if itemOperationStr != "" {
descriptionStr = style.FgCyan.Sprint(itemOperationStr+" "+utils.Loader(time.Now())) + " " + descriptionStr descriptionStr = style.FgCyan.Sprint(itemOperationStr+" "+utils.Loader(time.Now(), userConfig.Gui.Spinner)) + " " + descriptionStr
} }
res = append(res, textStyle.Sprint(t.Name), descriptionStr) res = append(res, textStyle.Sprint(t.Name), descriptionStr)
return res return res

View File

@ -4,6 +4,7 @@ import (
"time" "time"
"github.com/jesseduffield/gocui" "github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/config"
"github.com/jesseduffield/lazygit/pkg/gui/types" "github.com/jesseduffield/lazygit/pkg/gui/types"
"github.com/jesseduffield/lazygit/pkg/utils" "github.com/jesseduffield/lazygit/pkg/utils"
"github.com/samber/lo" "github.com/samber/lo"
@ -69,13 +70,13 @@ func (self *StatusManager) AddToastStatus(message string, kind types.ToastKind)
return id return id
} }
func (self *StatusManager) GetStatusString() (string, gocui.Attribute) { func (self *StatusManager) GetStatusString(userConfig *config.UserConfig) (string, gocui.Attribute) {
if len(self.statuses) == 0 { if len(self.statuses) == 0 {
return "", gocui.ColorDefault return "", gocui.ColorDefault
} }
topStatus := self.statuses[0] topStatus := self.statuses[0]
if topStatus.statusType == "waiting" { if topStatus.statusType == "waiting" {
return topStatus.message + " " + utils.Loader(time.Now()), topStatus.color return topStatus.message + " " + utils.Loader(time.Now(), userConfig.Gui.Spinner), topStatus.color
} }
return topStatus.message, topStatus.color return topStatus.message, topStatus.color
} }

View File

@ -11,6 +11,7 @@ import (
"time" "time"
"github.com/jesseduffield/gocui" "github.com/jesseduffield/gocui"
"github.com/jesseduffield/lazygit/pkg/config"
) )
// GetProjectRoot returns the path to the root of the project. Only to be used // GetProjectRoot returns the path to the root of the project. Only to be used
@ -24,15 +25,11 @@ func GetProjectRoot() string {
return strings.Split(dir, "lazygit")[0] + "lazygit" return strings.Split(dir, "lazygit")[0] + "lazygit"
} }
// The duration between two frames of the loader animation in milliseconds
const LoaderAnimationInterval = 50
// Loader dumps a string to be displayed as a loader // Loader dumps a string to be displayed as a loader
func Loader(now time.Time) string { func Loader(now time.Time, config config.SpinnerConfig) string {
characters := "|/-\\"
milliseconds := now.UnixMilli() milliseconds := now.UnixMilli()
index := milliseconds / LoaderAnimationInterval % int64(len(characters)) index := milliseconds / int64(config.Rate) % int64(len(config.Frames))
return characters[index : index+1] return config.Frames[index]
} }
// Min returns the minimum of two integers // Min returns the minimum of two integers

View File

@ -366,6 +366,32 @@
], ],
"description": "How things are filtered when typing '/'.\nOne of 'substring' (default) | 'fuzzy'", "description": "How things are filtered when typing '/'.\nOne of 'substring' (default) | 'fuzzy'",
"default": "substring" "default": "substring"
},
"spinner": {
"properties": {
"frames": {
"items": {
"type": "string"
},
"type": "array",
"description": "The frames of the spinner animation.",
"default": [
"|",
"/",
"-",
"\\"
]
},
"rate": {
"type": "integer",
"minimum": 1,
"description": "The \"speed\" of the spinner in milliseconds.",
"default": 50
}
},
"additionalProperties": false,
"type": "object",
"description": "Config relating to the spinner."
} }
}, },
"additionalProperties": false, "additionalProperties": false,