2021-10-17 20:12:33 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_MISC_PROGRESSREPORTER_H
|
|
|
|
#define OPENMW_COMPONENTS_MISC_PROGRESSREPORTER_H
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <chrono>
|
|
|
|
#include <mutex>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
namespace Misc
|
|
|
|
{
|
|
|
|
template <class Report>
|
|
|
|
class ProgressReporter
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit ProgressReporter(Report&& report = Report{})
|
|
|
|
: mReport(std::forward<Report>(report))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
explicit ProgressReporter(std::chrono::steady_clock::duration interval, Report&& report = Report{})
|
|
|
|
: mInterval(interval)
|
|
|
|
, mReport(std::forward<Report>(report))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
void operator()(std::size_t provided, std::size_t expected)
|
|
|
|
{
|
|
|
|
expected = std::max(expected, provided);
|
|
|
|
const bool shouldReport = [&] {
|
|
|
|
const std::lock_guard lock(mMutex);
|
|
|
|
const auto now = std::chrono::steady_clock::now();
|
2022-01-19 00:07:17 +00:00
|
|
|
if (mNextReport > now || provided == expected)
|
2021-10-17 20:12:33 +00:00
|
|
|
return false;
|
2022-01-19 00:07:17 +00:00
|
|
|
if (mInterval.count() > 0)
|
|
|
|
mNextReport
|
|
|
|
= mNextReport + mInterval * ((now - mNextReport + mInterval).count() / mInterval.count());
|
2021-10-17 20:12:33 +00:00
|
|
|
return true;
|
|
|
|
}();
|
|
|
|
if (shouldReport)
|
|
|
|
mReport(provided, expected);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
const std::chrono::steady_clock::duration mInterval = std::chrono::seconds(1);
|
|
|
|
Report mReport;
|
|
|
|
std::mutex mMutex;
|
|
|
|
std::chrono::steady_clock::time_point mNextReport{ std::chrono::steady_clock::now() + mInterval };
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|