forked from NaiJi/project-kyoku
Improve BPM calculation (not really (it's bad...))
This commit is contained in:
parent
46baf6fdfb
commit
944ad6a5bd
|
@ -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);
|
||||||
|
|
|
@ -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();
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
{
|
{
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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.)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue