package decor
import (
"fmt"
"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,
placeHolders: make([]*placeHolderDecorator, len(placeholders)),
}
md.wc = decorator.SetConfig(md.wc)
for i, wc := range placeholders {
wc.Init()
md.placeHolders[i] = &placeHolderDecorator{
WC: wc,
wsync: make(chan int),
}
}
return md
}
type mergeDecorator struct {
Decorator
wc WC
placeHolders []*placeHolderDecorator
}
func (d *mergeDecorator) MergeUnwrap() []Decorator {
decorators := make([]Decorator, len(d.placeHolders)+1)
decorators[0] = d.Decorator
for i, ph := range d.placeHolders {
decorators[i+1] = ph
}
return decorators
}
func (md *mergeDecorator) Sync() (chan int, bool) {
return md.wc.Sync()
}
func (d *mergeDecorator) Decor(st *Statistics) string {
msg := d.Decorator.Decor(st)
msgLen := utf8.RuneCountInString(msg)
var pWidth int
for _, ph := range d.placeHolders {
pWidth += <-ph.wsync
}
d.wc.wsync <- msgLen - pWidth
max := <-d.wc.wsync
if (d.wc.C & DextraSpace) != 0 {
max++
}
return fmt.Sprintf(fmt.Sprintf(d.wc.dynFormat, max+pWidth), msg)
}
type placeHolderDecorator struct {
WC
wsync chan int
}
func (d *placeHolderDecorator) Decor(st *Statistics) string {
go func() {
d.wsync <- utf8.RuneCountInString(d.FormatMsg(""))
}()
return ""
}