Files
MuditaOS/module-apps/apps-common/widgets/ProgressTimerImpl.cpp
mkamonMdt 44818b1287 [BH-880] Fix of missing minus in PowerNap
The bug could be noticed on the PowerNap's
progress window. The countdown there is supposed
to be increasing in value, that means the minus
sign should be visible at all non-zero values.
2021-09-21 20:10:28 +02:00

152 lines
4.4 KiB
C++

// Copyright (c) 2017-2021, Mudita Sp. z.o.o. All rights reserved.
// For licensing, see https://github.com/mudita/MuditaOS/LICENSE.md
#include "ProgressTimerImpl.hpp"
#include <Text.hpp>
#include <ProgressBar.hpp>
#include <ApplicationCommon.hpp>
#include <time/time_conversion.hpp>
#include <gsl/assert>
namespace
{
inline constexpr auto increasingModePrefix = "-";
}
namespace app
{
ProgressTimerImpl::ProgressTimerImpl(app::ApplicationCommon *app,
gui::Item *parent,
std::string timerName,
std::chrono::milliseconds baseTick,
ProgressCountdownMode countdownMode)
: app{app}, parent{parent}, name{std::move(timerName)}, baseTickInterval{baseTick}, countdownMode{countdownMode}
{}
void ProgressTimerImpl::resetProgress()
{
if (progress != nullptr) {
progress->setValue(0);
}
}
void ProgressTimerImpl::update()
{
updateText();
updateProgress();
app->refreshWindow(gui::RefreshModes::GUI_REFRESH_FAST);
}
void ProgressTimerImpl::updateText()
{
using utils::time::Duration;
if (text == nullptr) {
return;
}
const auto secondsRemaining = duration - elapsed;
const Duration remainingDuration{std::time_t{secondsRemaining.count()}};
UTF8 timerText;
if (countdownMode == ProgressCountdownMode::Increasing && secondsRemaining != std::chrono::seconds::zero()) {
timerText += increasingModePrefix;
}
timerText += remainingDuration.str(Duration::DisplayedFormat::Fixed0M0S);
text->setText(std::move(timerText));
}
void ProgressTimerImpl::updateProgress()
{
if (progress != nullptr) {
const auto percentage = static_cast<float>(elapsed.count()) / duration.count();
const auto currentStep = percentage * progress->getMaximum();
progress->setValue(std::ceil(currentStep));
}
}
void ProgressTimerImpl::reset(std::chrono::seconds _duration, std::chrono::seconds _interval)
{
Expects(_duration != std::chrono::seconds::zero());
duration = _duration;
elapsed = std::chrono::seconds::zero();
interval = _interval;
hasInterval = _interval != std::chrono::seconds::zero();
updateText();
resetProgress();
}
void ProgressTimerImpl::start()
{
startTimer();
isRunning = true;
}
void ProgressTimerImpl::startTimer()
{
Expects(app != nullptr);
parent->timerCallback = [this](gui::Item &it, sys::Timer &task) { return onTimerTimeout(it, task); };
timerTask = app::GuiTimerFactory::createPeriodicTimer(app, parent, name, baseTickInterval);
timerTask.start();
}
auto ProgressTimerImpl::onTimerTimeout(gui::Item &self, sys::Timer &task) -> bool
{
if (isStopped() || isFinished()) {
task.stop();
if (isFinished() && onFinishedCallback != nullptr) {
onFinishedCallback();
}
return true;
}
++elapsed;
if ((intervalReached() || isFinished()) && onIntervalCallback != nullptr) {
onIntervalCallback();
}
update();
return true;
}
auto ProgressTimerImpl::isFinished() const noexcept -> bool
{
return duration <= elapsed;
}
auto ProgressTimerImpl::isStopped() const noexcept -> bool
{
return !isRunning;
}
auto ProgressTimerImpl::intervalReached() const noexcept -> bool
{
return hasInterval && (elapsed.count() % interval.count()) == 0;
}
void ProgressTimerImpl::stop()
{
isRunning = false;
}
void ProgressTimerImpl::registerOnFinishedCallback(std::function<void()> cb)
{
onFinishedCallback = std::move(cb);
}
void ProgressTimerImpl::registerOnIntervalCallback(std::function<void()> cb)
{
onIntervalCallback = std::move(cb);
}
void ProgressTimerImpl::attach(gui::Progress *_progress)
{
Expects(_progress != nullptr);
progress = _progress;
}
void ProgressTimerImpl::attach(gui::Text *_text)
{
Expects(_text != nullptr);
text = _text;
}
} // namespace app