mirror of https://github.com/mamba-org/mamba.git
141 lines
4.7 KiB
C++
141 lines
4.7 KiB
C++
/*
|
|
Activity Indicators for Modern C++
|
|
https://github.com/p-ranav/indicators
|
|
|
|
Licensed under the MIT License <http://opensource.org/licenses/MIT>.
|
|
SPDX-License-Identifier: MIT
|
|
Copyright (c) 2019 Pranav Srinivas Kumar <pranav.srinivas.kumar@gmail.com>.
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
SOFTWARE.
|
|
*/
|
|
#pragma once
|
|
#include <atomic>
|
|
#include <functional>
|
|
#include <indicators/color.hpp>
|
|
#include <indicators/setting.hpp>
|
|
#include <iostream>
|
|
#include <mutex>
|
|
#include <vector>
|
|
|
|
namespace indicators {
|
|
|
|
template <typename Indicator> class DynamicProgress {
|
|
using Settings = std::tuple<option::HideBarWhenComplete>;
|
|
|
|
public:
|
|
template <typename... Indicators> explicit DynamicProgress(Indicators &... bars) {
|
|
bars_ = {bars...};
|
|
for (auto &bar : bars_) {
|
|
bar.get().multi_progress_mode_ = true;
|
|
++total_count_;
|
|
++incomplete_count_;
|
|
}
|
|
}
|
|
|
|
Indicator &operator[](size_t index) {
|
|
print_progress();
|
|
std::lock_guard<std::mutex> lock{mutex_};
|
|
return bars_[index].get();
|
|
}
|
|
|
|
size_t push_back(Indicator &bar) {
|
|
std::lock_guard<std::mutex> lock{mutex_};
|
|
bar.multi_progress_mode_ = true;
|
|
bars_.push_back(bar);
|
|
++total_count_;
|
|
++incomplete_count_;
|
|
return bars_.size() - 1;
|
|
}
|
|
|
|
template <typename T, details::ProgressBarOption id>
|
|
void set_option(details::Setting<T, id> &&setting) {
|
|
static_assert(!std::is_same<T, typename std::decay<decltype(details::get_value<id>(
|
|
std::declval<Settings>()))>::type>::value,
|
|
"Setting has wrong type!");
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
|
get_value<id>() = std::move(setting).value;
|
|
}
|
|
|
|
template <typename T, details::ProgressBarOption id>
|
|
void set_option(const details::Setting<T, id> &setting) {
|
|
static_assert(!std::is_same<T, typename std::decay<decltype(details::get_value<id>(
|
|
std::declval<Settings>()))>::type>::value,
|
|
"Setting has wrong type!");
|
|
std::lock_guard<std::mutex> lock(mutex_);
|
|
get_value<id>() = setting.value;
|
|
}
|
|
|
|
private:
|
|
Settings settings_;
|
|
std::atomic<bool> started_{false};
|
|
std::mutex mutex_;
|
|
std::vector<std::reference_wrapper<Indicator>> bars_;
|
|
std::atomic<size_t> total_count_{0};
|
|
std::atomic<size_t> incomplete_count_{0};
|
|
|
|
template <details::ProgressBarOption id>
|
|
auto get_value() -> decltype((details::get_value<id>(std::declval<Settings &>()).value)) {
|
|
return details::get_value<id>(settings_).value;
|
|
}
|
|
|
|
template <details::ProgressBarOption id>
|
|
auto get_value() const
|
|
-> decltype((details::get_value<id>(std::declval<const Settings &>()).value)) {
|
|
return details::get_value<id>(settings_).value;
|
|
}
|
|
|
|
void print_progress() {
|
|
std::lock_guard<std::mutex> lock{mutex_};
|
|
auto &hide_bar_when_complete = get_value<details::ProgressBarOption::hide_bar_when_complete>();
|
|
if (hide_bar_when_complete) {
|
|
// Hide completed bars
|
|
if (started_) {
|
|
for (size_t i = 0; i < incomplete_count_; ++i)
|
|
std::cout << "\033[A\r\033[K" << std::flush;
|
|
}
|
|
incomplete_count_ = 0;
|
|
for (auto &bar : bars_) {
|
|
if (!bar.get().is_completed()) {
|
|
bar.get().print_progress(true);
|
|
std::cout << "\n";
|
|
++incomplete_count_;
|
|
}
|
|
}
|
|
if (!started_)
|
|
started_ = true;
|
|
} else {
|
|
// Don't hide any bars
|
|
if (started_) {
|
|
for (size_t i = 0; i < total_count_; ++i)
|
|
std::cout << "\x1b[A";
|
|
}
|
|
for (auto &bar : bars_) {
|
|
bar.get().print_progress(true);
|
|
std::cout << "\n";
|
|
}
|
|
if (!started_)
|
|
started_ = true;
|
|
}
|
|
total_count_ = bars_.size();
|
|
std::cout << termcolor::reset;
|
|
}
|
|
};
|
|
|
|
} // namespace indicators
|