project-kyoku/include/precisionevaluator.h

67 lines
1.8 KiB
C
Raw Normal View History

#ifndef PRECISIONEVALUATOR_H
#define PRECISIONEVALUATOR_H
#include <numeric>
#include <type_traits>
#include <vector>
#include <cmath>
#include <SFML/System/Clock.hpp>
using microsec = sf::Int64;
2021-06-11 18:58:44 +02:00
template<typename Grade, typename = std::enable_if_t<std::is_enum<Grade>::value>>
class PrecisionEvaluator
{
public:
2021-06-07 20:19:58 +02:00
PrecisionEvaluator(const std::vector<microsec>& intervals, microsec offset) :
_offset(offset),
2021-06-07 20:19:58 +02:00
_intervals(intervals)
{
microsec&& handling_offset = std::accumulate(intervals.begin(), intervals.end(), 0);
_start_handling_offset = _offset - handling_offset;
_end_handling_offset = _offset + handling_offset;
}
inline microsec offset() const noexcept
{
return _offset;
}
inline bool isActive(microsec music_play_offset) const noexcept
{
return music_play_offset > _start_handling_offset
&& music_play_offset < _end_handling_offset;
}
2021-06-11 18:58:44 +02:00
inline Grade calculatePrecision(microsec odds) const noexcept
{
microsec shift_from_perfect = std::abs(odds - offset());
std::size_t raw_grade;
for (raw_grade = 0; raw_grade < _intervals.size(); ++raw_grade)
{
2021-06-07 20:19:58 +02:00
if (shift_from_perfect <= _intervals.at(raw_grade))
break;
}
2021-06-11 18:58:44 +02:00
return static_cast<Grade>(raw_grade);
}
private:
microsec _offset;
microsec _start_handling_offset;
microsec _end_handling_offset;
2021-06-11 18:58:44 +02:00
/* Amount of values in enum instanced as GradeS
* represents capacity of _intervals.
2021-06-11 18:58:44 +02:00
* So, for each V value in GradeS enum, _intervals[V]
* should return time shift from V - 1.
* V0 is PERFECT SCORE and the last V represents the worst
* grades which is death of note by expiration */
2021-06-07 20:19:58 +02:00
const std::vector<microsec>& _intervals;
};
#endif // PRECISIONEVALUATOR_H