Improve BPM calculation (not really (it's bad...))

This commit is contained in:
NaiJi ✨ 2021-09-13 21:50:39 +03:00
parent 46baf6fdfb
commit 944ad6a5bd
7 changed files with 54 additions and 14 deletions

View File

@ -7,8 +7,8 @@ namespace beat_utils
{ {
struct BeatInfo struct BeatInfo
{ {
long double rate_per_microsecond = 0.; int BPM = 0;
microsec average_interval = 0; microsec interval = 0;
}; };
BeatInfo calculateBeatRateInfo(const std::vector<microsec>& approximate_deltas); BeatInfo calculateBeatRateInfo(const std::vector<microsec>& approximate_deltas);

View File

@ -16,6 +16,8 @@ public:
void stop(); void stop();
void click(); void click();
bool calculating() const;
const beat_utils::BeatInfo& fetchApproximatedInfo(); const beat_utils::BeatInfo& fetchApproximatedInfo();
microsec fetchTimeUntilNextBeat(); microsec fetchTimeUntilNextBeat();

View File

@ -3,6 +3,7 @@
#include <SFML/System/Clock.hpp> #include <SFML/System/Clock.hpp>
using microsec = sf::Int64; using microsec = sf::Int64;
using minute = int;
struct Coordinates struct Coordinates
{ {

View File

@ -41,12 +41,12 @@ void BPMCalculatorWidget::update(const sf::Time& dt)
Window::update(dt); Window::update(dt);
const auto beat_info = _bpm_calculator->fetchApproximatedInfo(); const auto beat_info = _bpm_calculator->fetchApproximatedInfo();
if (beat_info.rate_per_microsecond != 0) if (beat_info.BPM != 0)
{ {
_bpm_value.setString(std::to_string(static_cast<int>(beat_info.rate_per_microsecond))); _bpm_value.setString(std::to_string(static_cast<int>(beat_info.BPM)));
const microsec until_beat = _bpm_calculator->fetchTimeUntilNextBeat(); const microsec until_beat = _bpm_calculator->fetchTimeUntilNextBeat();
const auto time_relation = static_cast<long double>(beat_info.average_interval) / static_cast<long double>(until_beat); const auto time_relation = static_cast<long double>(beat_info.interval) / static_cast<long double>(until_beat);
const auto slider_path_left = _slider->rect().width / time_relation; const auto slider_path_left = _slider->rect().width / time_relation;
if (slider_path_left < 50) if (slider_path_left < 50)
{ {
@ -69,6 +69,7 @@ void BPMCalculatorWidget::draw(sf::RenderTarget& target, sf::RenderStates states
{ {
_slider->draw(target, states); _slider->draw(target, states);
_button_start->draw(target, states); _button_start->draw(target, states);
_button_stop->draw(target, states);
target.draw(_bpm_value, states); target.draw(_bpm_value, states);
} }
} }
@ -84,6 +85,10 @@ void BPMCalculatorWidget::setRect(const sf::FloatRect& rect)
_button_start->setPosition({_window_content.getGlobalBounds().left + rect.width / 7, _button_start->setPosition({_window_content.getGlobalBounds().left + rect.width / 7,
_window_content.getGlobalBounds().top + _window_content.getGlobalBounds().height - 40}); _window_content.getGlobalBounds().top + _window_content.getGlobalBounds().height - 40});
_button_stop->setRect(sf::FloatRect{0, 0, rect.width / 10 * 3, 30});
_button_stop->setPosition({_window_content.getGlobalBounds().left + rect.width / 7,
_window_content.getGlobalBounds().top + _window_content.getGlobalBounds().height - 40});
_bpm_value.setPosition({_window_content.getGlobalBounds().left + rect.width / 8, _bpm_value.setPosition({_window_content.getGlobalBounds().left + rect.width / 8,
_window_content.getGlobalBounds().top + rect.height / 8 }); _window_content.getGlobalBounds().top + rect.height / 8 });
} }
@ -105,12 +110,35 @@ void BPMCalculatorWidget::init()
auto& bpm_calculator = _bpm_calculator; auto& bpm_calculator = _bpm_calculator;
_button_start = std::make_shared<PushButton>("Start", _font); _button_start = std::make_shared<PushButton>("Start", _font);
_button_start->setCallback([bpm_calculator, button_start=_button_start]() _button_stop = std::make_shared<PushButton>("Stop", _font);
_button_start->setCallback([bpm_calculator, button_start=_button_start, button_stop=_button_stop]()
{ {
bpm_calculator->music()->play(); bpm_calculator->music()->play(); // Remove when global play/stop available
bpm_calculator->start(); bpm_calculator->start();
button_start->setVisibility(false); button_start->setVisibility(false);
button_stop->setVisibility(true);
});
_button_stop->setCallback([bpm_calculator, button_start=_button_start, button_stop=_button_stop]()
{
bpm_calculator->music()->stop(); // Remove when global play/stop available
bpm_calculator->stop();
button_start->setVisibility(true);
button_stop->setVisibility(false);
}); });
addChild(_button_start); addChild(_button_start);
addChild(_button_stop);
_button_stop->setVisibility(false);
}
void BPMCalculatorWidget::setVisibility(bool is_visible)
{
Window::setVisibility(is_visible);
bool can_stop = _bpm_calculator->calculating();
_button_stop->setVisibility(can_stop);
_button_start->setVisibility(!can_stop);
} }

View File

@ -23,10 +23,13 @@ public:
virtual void setPosition(const sf::Vector2f& position) override; virtual void setPosition(const sf::Vector2f& position) override;
virtual void move(const sf::Vector2f& delta) override; virtual void move(const sf::Vector2f& delta) override;
virtual void setVisibility(bool is_visible = true) override;
void init(); void init();
private: private:
std::shared_ptr<PushButton> _button_start; std::shared_ptr<PushButton> _button_start;
std::shared_ptr<PushButton> _button_stop;
std::shared_ptr<BPMCalculator> _bpm_calculator; std::shared_ptr<BPMCalculator> _bpm_calculator;
std::shared_ptr<BPMSlider> _slider; std::shared_ptr<BPMSlider> _slider;

View File

@ -13,11 +13,12 @@ auto beat_utils::calculateBeatRateInfo(const std::vector<microsec>& approximate_
long double average = static_cast<long double>(sum) long double average = static_cast<long double>(sum)
/ static_cast<long double>(amount); / static_cast<long double>(amount);
const int bpm = static_cast<int>(60000000. / average);
return BeatInfo return BeatInfo
{ {
60000000. / average, bpm,
static_cast<microsec>(average) static_cast<microsec>(1. / static_cast<long double>(bpm) * 60000000.)
}; };
} }

View File

@ -41,6 +41,11 @@ void BPMCalculator::stop()
_calculating = false; _calculating = false;
} }
bool BPMCalculator::calculating() const
{
return _calculating;
}
void BPMCalculator::click() void BPMCalculator::click()
{ {
if (!_calculating) if (!_calculating)
@ -64,10 +69,10 @@ void BPMCalculator::click()
const beat_utils::BeatInfo& BPMCalculator::fetchApproximatedInfo() const beat_utils::BeatInfo& BPMCalculator::fetchApproximatedInfo()
{ {
//if (!_need_recalculate) if (!_need_recalculate)
//return _approximated_info; return _approximated_info;
//_need_recalculate = false; _need_recalculate = false;
bool hasEnoughDeltas = _deltas.size() >= 8; bool hasEnoughDeltas = _deltas.size() >= 8;
@ -95,7 +100,7 @@ void BPMCalculator::moveStartingOffsetBy(microsec shift)
microsec BPMCalculator::fetchTimeUntilNextBeat() microsec BPMCalculator::fetchTimeUntilNextBeat()
{ {
const microsec actual_offset = _music->fetchOffset() - _first_click_offset; const microsec actual_offset = _music->fetchOffset() - getStartingOffset();
return actual_offset % fetchApproximatedInfo().average_interval; return actual_offset % fetchApproximatedInfo().interval;
} }