Merge pull request #9079 from jeremyschlatter/main

cmd: fix flickering in progress bar
This commit is contained in:
Michael Yang 2025-02-18 22:59:29 +00:00 committed by GitHub
commit e13e7c8d94
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194

View File

@ -1,6 +1,7 @@
package progress package progress
import ( import (
"bufio"
"fmt" "fmt"
"io" "io"
"sync" "sync"
@ -13,7 +14,8 @@ type State interface {
type Progress struct { type Progress struct {
mu sync.Mutex mu sync.Mutex
w io.Writer // buffer output to minimize flickering on all terminals
w *bufio.Writer
pos int pos int
@ -22,7 +24,7 @@ type Progress struct {
} }
func NewProgress(w io.Writer) *Progress { func NewProgress(w io.Writer) *Progress {
p := &Progress{w: w} p := &Progress{w: bufio.NewWriter(w)}
go p.start() go p.start()
return p return p
} }
@ -48,11 +50,14 @@ func (p *Progress) Stop() bool {
stopped := p.stop() stopped := p.stop()
if stopped { if stopped {
fmt.Fprint(p.w, "\n") fmt.Fprint(p.w, "\n")
p.w.Flush()
} }
return stopped return stopped
} }
func (p *Progress) StopAndClear() bool { func (p *Progress) StopAndClear() bool {
defer p.w.Flush()
fmt.Fprint(p.w, "\033[?25l") fmt.Fprint(p.w, "\033[?25l")
defer fmt.Fprint(p.w, "\033[?25h") defer fmt.Fprint(p.w, "\033[?25h")
@ -81,20 +86,24 @@ func (p *Progress) render() {
p.mu.Lock() p.mu.Lock()
defer p.mu.Unlock() defer p.mu.Unlock()
defer p.w.Flush()
// eliminate flickering on terminals that support synchronized output
fmt.Fprint(p.w, "\033[?2026h")
defer fmt.Fprint(p.w, "\033[?2026l")
fmt.Fprint(p.w, "\033[?25l") fmt.Fprint(p.w, "\033[?25l")
defer fmt.Fprint(p.w, "\033[?25h") defer fmt.Fprint(p.w, "\033[?25h")
// clear already rendered progress lines // move the cursor back to the beginning
for i := range p.pos { for range p.pos - 1 {
if i > 0 {
fmt.Fprint(p.w, "\033[A") fmt.Fprint(p.w, "\033[A")
} }
fmt.Fprint(p.w, "\033[2K\033[1G") fmt.Fprint(p.w, "\033[1G")
}
// render progress lines // render progress lines
for i, state := range p.states { for i, state := range p.states {
fmt.Fprint(p.w, state.String()) fmt.Fprint(p.w, state.String(), "\033[K")
if i < len(p.states)-1 { if i < len(p.states)-1 {
fmt.Fprint(p.w, "\n") fmt.Fprint(p.w, "\n")
} }