summaryrefslogtreecommitdiff
path: root/vendor/github.com/vbauerster/mpb/v5/decor/merge.go
blob: 520f13a7f2ee7d048951dcfb2d69adcd565f5ce5 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
package decor

import (
	"fmt"
	"strings"
	"unicode/utf8"
)

// Merge wraps its decorator argument with intention to sync width
// with several decorators of another bar. Visual example:
//
//    +----+--------+---------+--------+
//    | B1 |      MERGE(D, P1, Pn)     |
//    +----+--------+---------+--------+
//    | B2 |   D0   |   D1    |   Dn   |
//    +----+--------+---------+--------+
//
func Merge(decorator Decorator, placeholders ...WC) Decorator {
	if _, ok := decorator.Sync(); !ok || len(placeholders) == 0 {
		return decorator
	}
	md := &mergeDecorator{
		Decorator:    decorator,
		wc:           decorator.GetConf(),
		placeHolders: make([]*placeHolderDecorator, len(placeholders)),
	}
	decorator.SetConf(WC{})
	for i, wc := range placeholders {
		if (wc.C & DSyncWidth) == 0 {
			return decorator
		}
		md.placeHolders[i] = &placeHolderDecorator{wc.Init()}
	}
	return md
}

type mergeDecorator struct {
	Decorator
	wc           WC
	placeHolders []*placeHolderDecorator
}

func (d *mergeDecorator) GetConf() WC {
	return d.wc
}

func (d *mergeDecorator) SetConf(conf WC) {
	d.wc = conf.Init()
}

func (d *mergeDecorator) MergeUnwrap() []Decorator {
	decorators := make([]Decorator, len(d.placeHolders))
	for i, ph := range d.placeHolders {
		decorators[i] = ph
	}
	return decorators
}

func (d *mergeDecorator) Sync() (chan int, bool) {
	return d.wc.Sync()
}

func (d *mergeDecorator) Base() Decorator {
	return d.Decorator
}

func (d *mergeDecorator) Decor(s *Statistics) string {
	msg := d.Decorator.Decor(s)
	msgLen := utf8.RuneCountInString(msg)
	if (d.wc.C & DextraSpace) != 0 {
		msgLen++
	}

	var total int
	max := utf8.RuneCountInString(d.placeHolders[0].FormatMsg(""))
	total += max
	pw := (msgLen - max) / len(d.placeHolders)
	rem := (msgLen - max) % len(d.placeHolders)

	var diff int
	for i := 1; i < len(d.placeHolders); i++ {
		ph := d.placeHolders[i]
		width := pw - diff
		if (ph.WC.C & DextraSpace) != 0 {
			width--
			if width < 0 {
				width = 0
			}
		}
		max = utf8.RuneCountInString(ph.FormatMsg(strings.Repeat(" ", width)))
		total += max
		diff = max - pw
	}

	d.wc.wsync <- pw + rem
	max = <-d.wc.wsync
	return fmt.Sprintf(fmt.Sprintf(d.wc.dynFormat, max+total), msg)
}

type placeHolderDecorator struct {
	WC
}

func (d *placeHolderDecorator) Decor(*Statistics) string {
	return ""
}