about summary refs log tree commit diff
path: root/src/nix/progress-bar.cc
blob: 746b891a78df4a746b8d02eb07b7da3fd63e78f4 (plain) (blame)
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
#include "progress-bar.hh"

#include <iostream>

namespace nix {

ProgressBar::ProgressBar()
{
    _writeToStderr = [&](const unsigned char * buf, size_t count) {
        auto state_(state.lock());
        assert(!state_->done);
        std::cerr << "\r\e[K" << std::string((const char *) buf, count);
        render(*state_);
    };
}

ProgressBar::~ProgressBar()
{
    done();
}

void ProgressBar::updateStatus(const std::string & s)
{
    auto state_(state.lock());
    assert(!state_->done);
    state_->status = s;
    render(*state_);
}

void ProgressBar::done()
{
    _writeToStderr = decltype(_writeToStderr)();
    auto state_(state.lock());
    assert(state_->activities.empty());
    state_->done = true;
    std::cerr << "\r\e[K";
    std::cerr.flush();
}

void ProgressBar::render(State & state_)
{
    std::cerr << '\r' << state_.status;
    if (!state_.activities.empty()) {
        if (!state_.status.empty()) std::cerr << ' ';
        std::cerr << *state_.activities.rbegin();
    }
    std::cerr << "\e[K";
    std::cerr.flush();
}


ProgressBar::Activity ProgressBar::startActivity(const FormatOrString & fs)
{
    return Activity(*this, fs);
}

ProgressBar::Activity::Activity(ProgressBar & pb, const FormatOrString & fs)
    : pb(pb)
{
    auto state_(pb.state.lock());
    state_->activities.push_back(fs.s);
    it = state_->activities.end(); --it;
    pb.render(*state_);
}

ProgressBar::Activity::~Activity()
{
    auto state_(pb.state.lock());
    state_->activities.erase(it);
}

}