From d248ab44f3a40f26fb5e91d4882d8118c742c98b Mon Sep 17 00:00:00 2001 From: Vaibhav Awale Date: Thu, 27 Jun 2024 14:07:47 +0200 Subject: [PATCH] Fix progress bar display when terminal width is too small When the terminal width is too small, we run into problems as mentioned in https://github.com/p-ranav/indicators/issues/132. This PR fixes the problem by keeping track of number of wrapped lines printed by progress bar and erasing those extra wrapped lines before printing progress for next iteration. For multi_progress bar and dynamic_progress bar, we need to move cursor to start before each iteration by summing up number of wrapped lines for each bar. Since each bar's `print_progress` expects cursor to be on last printed line, we move cursor down by number of wrapped lines before printing progress for each bar. I have not yet handled case when `dynamic_progress` option `hide_bar_when_complete` is set. Fixes Issue #132 --- include/indicators/block_progress_bar.hpp | 38 ++++++++++++------- include/indicators/cursor_control.hpp | 9 +++++ include/indicators/details/stream_helper.hpp | 12 ++++++ include/indicators/dynamic_progress.hpp | 20 ++++++++-- .../indicators/indeterminate_progress_bar.hpp | 20 ++++++++-- include/indicators/multi_progress.hpp | 17 ++++++++- include/indicators/progress_bar.hpp | 20 ++++++++-- 7 files changed, 109 insertions(+), 27 deletions(-) diff --git a/include/indicators/block_progress_bar.hpp b/include/indicators/block_progress_bar.hpp index fcffdaf..aa7047a 100644 --- a/include/indicators/block_progress_bar.hpp +++ b/include/indicators/block_progress_bar.hpp @@ -4,6 +4,7 @@ #include #include +#include #include #include @@ -12,8 +13,8 @@ #include #include #include -#include #include +#include #include #include #include @@ -33,7 +34,7 @@ class BlockProgressBar { typename std::enable_if::type...>::value, void *>::type = nullptr> - explicit BlockProgressBar(Args &&... args) + explicit BlockProgressBar(Args &&...args) : settings_(details::get( option::ForegroundColor{Color::unspecified}, std::forward(args)...), details::get(option::BarWidth{100}, @@ -123,7 +124,7 @@ class BlockProgressBar { size_t current() { std::lock_guard lock{mutex_}; return (std::min)(static_cast(progress_), - size_t(get_value())); + size_t(get_value())); } bool is_completed() const { return get_value(); } @@ -133,6 +134,11 @@ class BlockProgressBar { print_progress(); } + size_t extra_wrapped_lines() { + std::lock_guard lock{mutex_}; + return extra_wrapped_lines_; + } + private: template auto get_value() -> decltype((details::get_value(std::declval()).value)) { @@ -140,8 +146,8 @@ class BlockProgressBar { } template - auto get_value() const - -> decltype((details::get_value(std::declval()).value)) { + auto + get_value() const -> decltype((details::get_value(std::declval()).value)) { return details::get_value(settings_).value; } @@ -149,6 +155,7 @@ class BlockProgressBar { float progress_{0.0}; std::chrono::time_point start_time_point_; std::mutex mutex_; + size_t extra_wrapped_lines_{0}; template friend class MultiProgress; template friend class DynamicProgress; @@ -201,10 +208,9 @@ class BlockProgressBar { if (saved_start_time) { auto eta = std::chrono::nanoseconds( - progress_ > 0 - ? static_cast(std::ceil(float(elapsed.count()) * - max_progress / progress_)) - : 0); + progress_ > 0 ? static_cast( + std::ceil(float(elapsed.count()) * max_progress / progress_)) + : 0); auto remaining = eta > elapsed ? (eta - elapsed) : (elapsed - eta); details::write_duration(os, remaining); } else { @@ -244,6 +250,10 @@ class BlockProgressBar { for (auto &style : get_value()) details::set_font_style(os, style); + // Need to erase previously written text across multiple lines to solve + // issue https://github.com/p-ranav/indicators/issues/132 + erase_lines(extra_wrapped_lines_); + const auto prefix_pair = get_prefix_text(); const auto prefix_text = prefix_pair.first; const auto prefix_length = prefix_pair.second; @@ -267,17 +277,19 @@ class BlockProgressBar { const auto bar_width = get_value(); const auto end_length = get_value().size(); const auto terminal_width = terminal_size().second; - // prefix + bar_width + postfix should be <= terminal_width - const int remaining = terminal_width - (prefix_length + start_length + bar_width + end_length + postfix_length); + const auto number_of_characters = + prefix_length + start_length + bar_width + end_length + postfix_length; + // If prefix + bar_width + postfix > terminal_width, lines will be wrapped + const int remaining = terminal_width - number_of_characters; if (prefix_length == -1 || postfix_length == -1) { os << "\r"; } else if (remaining > 0) { os << std::string(remaining, ' ') << "\r"; - } else if (remaining < 0) { - // Do nothing. Maybe in the future truncate postfix with ... } os.flush(); + extra_wrapped_lines_ = details::extra_wrapped_lines(number_of_characters); + if (progress_ > max_progress) { get_value() = true; } diff --git a/include/indicators/cursor_control.hpp b/include/indicators/cursor_control.hpp index 641156f..6cf13bb 100644 --- a/include/indicators/cursor_control.hpp +++ b/include/indicators/cursor_control.hpp @@ -2,6 +2,8 @@ #ifndef INDICATORS_CURSOR_CONTROL #define INDICATORS_CURSOR_CONTROL +#include + #if defined(_MSC_VER) #if !defined(NOMINMAX) #define NOMINMAX @@ -61,6 +63,13 @@ static inline void erase_line() { #endif +static inline void erase_lines(size_t lines) { + for (size_t i = 0; i < lines; ++i) { + erase_line(); + move_up(1); + } +} + } // namespace indicators #endif \ No newline at end of file diff --git a/include/indicators/details/stream_helper.hpp b/include/indicators/details/stream_helper.hpp index 48d06b8..62e3503 100644 --- a/include/indicators/details/stream_helper.hpp +++ b/include/indicators/details/stream_helper.hpp @@ -5,6 +5,7 @@ #include #include #include +#include #include #include @@ -217,6 +218,17 @@ class IndeterminateProgressScaleWriter { std::string lead; }; +inline size_t extra_wrapped_lines(size_t number_of_characters) +{ + const auto number_of_columns = indicators::terminal_width(); + const auto extra_lines = number_of_characters / number_of_columns; + // cursor does not wrap when writing to the last column + if ((extra_lines > 0) && (number_of_characters % number_of_columns) == 0) { + return extra_lines-1; + } + return extra_lines; +} + } // namespace details } // namespace indicators diff --git a/include/indicators/dynamic_progress.hpp b/include/indicators/dynamic_progress.hpp index 49d7d89..a062a90 100644 --- a/include/indicators/dynamic_progress.hpp +++ b/include/indicators/dynamic_progress.hpp @@ -2,13 +2,15 @@ #ifndef INDICATORS_DYNAMIC_PROGRESS #define INDICATORS_DYNAMIC_PROGRESS -#include -#include #include #include #include #include #include + +#include +#include +#include #include #include #include @@ -103,9 +105,19 @@ template class DynamicProgress { started_ = true; } else { // Don't hide any bars - if (started_) - move_up(static_cast(total_count_)); + if (started_) { + // move all the way up to start of first progress bar + const auto wrapped_lines = std::accumulate(begin(bars_), end(bars_), 0, [](size_t acc, auto &bar) { + return acc + bar.get().extra_wrapped_lines(); + }); + move_up(total_count_ + wrapped_lines); + } for (auto &bar : bars_) { + auto wrapped_line = bar.get().extra_wrapped_lines(); + if (wrapped_line > 0) { + // for each bar before calling `print_progress`, cursor needs to be on last line that bar printed + move_down(wrapped_line); + } bar.get().print_progress(true); std::cout << "\n"; } diff --git a/include/indicators/indeterminate_progress_bar.hpp b/include/indicators/indeterminate_progress_bar.hpp index f77dd5a..835204f 100644 --- a/include/indicators/indeterminate_progress_bar.hpp +++ b/include/indicators/indeterminate_progress_bar.hpp @@ -3,6 +3,7 @@ #define INDICATORS_INDETERMINATE_PROGRESS_BAR #include +#include #include #include @@ -138,6 +139,11 @@ class IndeterminateProgressBar { print_progress(); } + size_t extra_wrapped_lines() { + std::lock_guard lock{mutex_}; + return extra_wrapped_lines_; + } + private: template auto get_value() -> decltype((details::get_value(std::declval()).value)) { @@ -155,6 +161,7 @@ class IndeterminateProgressBar { Settings settings_; std::chrono::nanoseconds elapsed_; std::mutex mutex_; + size_t extra_wrapped_lines_{0}; template friend class MultiProgress; template friend class DynamicProgress; @@ -191,6 +198,10 @@ class IndeterminateProgressBar { for (auto &style : get_value()) details::set_font_style(os, style); + + // Need to erase previously written text across multiple lines to solve + // issue https://github.com/p-ranav/indicators/issues/132 + erase_lines(extra_wrapped_lines_); const auto prefix_pair = get_prefix_text(); const auto prefix_text = prefix_pair.first; @@ -217,17 +228,18 @@ class IndeterminateProgressBar { const auto bar_width = get_value(); const auto end_length = get_value().size(); const auto terminal_width = terminal_size().second; - // prefix + bar_width + postfix should be <= terminal_width - const int remaining = terminal_width - (prefix_length + start_length + bar_width + end_length + postfix_length); + const auto number_of_characters = prefix_length + start_length + bar_width + end_length + postfix_length; + // If prefix + bar_width + postfix > terminal_width, lines will be wrapped + const int remaining = terminal_width - number_of_characters; if (prefix_length == -1 || postfix_length == -1) { os << "\r"; } else if (remaining > 0) { os << std::string(remaining, ' ') << "\r"; - } else if (remaining < 0) { - // Do nothing. Maybe in the future truncate postfix with ... } os.flush(); + extra_wrapped_lines_ = details::extra_wrapped_lines(number_of_characters); + if (get_value() && !from_multi_progress) // Don't std::endl if calling from MultiProgress os << termcolor::reset << std::endl; diff --git a/include/indicators/multi_progress.hpp b/include/indicators/multi_progress.hpp index 16a9dc4..a10b550 100644 --- a/include/indicators/multi_progress.hpp +++ b/include/indicators/multi_progress.hpp @@ -6,6 +6,7 @@ #include #include #include +#include #include #include @@ -65,9 +66,21 @@ template class MultiProgress { public: void print_progress() { std::lock_guard lock{mutex_}; - if (started_) - move_up(count); + + if (started_) { + // move all the way up to start of first progress bar + const auto wrapped_lines = std::accumulate(begin(bars_), end(bars_), 0, [](size_t acc, auto &bar) { + return acc + bar.get().extra_wrapped_lines(); + }); + move_up(count + wrapped_lines); + } + for (auto &bar : bars_) { + auto wrapped_line = bar.get().extra_wrapped_lines(); + if (wrapped_line > 0) { + // for each bar before calling `print_progress`, cursor needs to be on last line that bar printed + move_down(wrapped_line); + } bar.get().print_progress(true); std::cout << "\n"; } diff --git a/include/indicators/progress_bar.hpp b/include/indicators/progress_bar.hpp index ce22957..4fe1aab 100644 --- a/include/indicators/progress_bar.hpp +++ b/include/indicators/progress_bar.hpp @@ -3,6 +3,7 @@ #define INDICATORS_PROGRESS_BAR #include +#include #include #include @@ -180,6 +181,11 @@ class ProgressBar { print_progress(); } + size_t extra_wrapped_lines() { + std::lock_guard lock{mutex_}; + return extra_wrapped_lines_; + } + private: template auto get_value() @@ -198,6 +204,7 @@ class ProgressBar { std::chrono::nanoseconds elapsed_; std::chrono::time_point start_time_point_; std::mutex mutex_; + size_t extra_wrapped_lines_{0}; template friend class MultiProgress; template friend class DynamicProgress; @@ -310,6 +317,10 @@ class ProgressBar { for (auto &style : get_value()) details::set_font_style(os, style); + // Need to erase previously written text across multiple lines to solve + // issue https://github.com/p-ranav/indicators/issues/132 + erase_lines(extra_wrapped_lines_); + const auto prefix_pair = get_prefix_text(); const auto prefix_text = prefix_pair.first; const auto prefix_length = prefix_pair.second; @@ -336,17 +347,18 @@ class ProgressBar { const auto bar_width = get_value(); const auto end_length = get_value().size(); const auto terminal_width = terminal_size().second; - // prefix + bar_width + postfix should be <= terminal_width - const int remaining = terminal_width - (prefix_length + start_length + bar_width + end_length + postfix_length); + const auto number_of_characters = prefix_length + start_length + bar_width + end_length + postfix_length; + // If prefix + bar_width + postfix > terminal_width, lines will be wrapped + const int remaining = terminal_width - number_of_characters; if (prefix_length == -1 || postfix_length == -1) { os << "\r"; } else if (remaining > 0) { os << std::string(remaining, ' ') << "\r"; - } else if (remaining < 0) { - // Do nothing. Maybe in the future truncate postfix with ... } os.flush(); + extra_wrapped_lines_ = details::extra_wrapped_lines(number_of_characters); + if ((type == ProgressType::incremental && progress_ >= max_progress) || (type == ProgressType::decremental && progress_ <= min_progress)) { get_value() = true;