2016-05-11 21:30:45 +00:00
|
|
|
#include "amuse/Envelope.hpp"
|
|
|
|
|
|
|
|
namespace amuse
|
|
|
|
{
|
|
|
|
|
|
|
|
void Envelope::reset(const ADSR* adsr)
|
|
|
|
{
|
|
|
|
m_phase = State::Attack;
|
2016-05-16 20:20:29 +00:00
|
|
|
m_curTime = 0.0;
|
|
|
|
m_attackTime = adsr->getAttack();
|
|
|
|
m_decayTime = adsr->getDecay();
|
|
|
|
m_sustainFactor = adsr->getSustain();
|
|
|
|
m_releaseTime = adsr->getRelease();
|
|
|
|
m_releaseStartFactor = 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Envelope::reset(const ADSRDLS* adsr, int8_t note, int8_t vel)
|
|
|
|
{
|
|
|
|
m_phase = State::Attack;
|
|
|
|
m_curTime = 0.0;
|
|
|
|
m_attackTime = adsr->getVelToAttack(vel);
|
|
|
|
m_decayTime = adsr->getKeyToDecay(note);
|
|
|
|
m_sustainFactor = adsr->getSustain();
|
|
|
|
m_releaseTime = adsr->getRelease();
|
2016-05-11 21:30:45 +00:00
|
|
|
m_releaseStartFactor = 0.0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void Envelope::keyOff()
|
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
m_phase = (m_releaseTime != 0.0) ? State::Release : State::Complete;
|
|
|
|
m_curTime = 0.0;
|
2016-05-11 21:30:45 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
float Envelope::nextSample(double sampleRate)
|
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
m_curTime += 1.0 / sampleRate;
|
2016-05-11 21:30:45 +00:00
|
|
|
|
|
|
|
switch (m_phase)
|
|
|
|
{
|
|
|
|
case State::Attack:
|
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
if (m_attackTime == 0.0)
|
2016-05-14 22:38:37 +00:00
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
m_phase = State::Decay;
|
|
|
|
m_curTime = 0.0;
|
2016-05-16 02:40:18 +00:00
|
|
|
m_releaseStartFactor = 1.f;
|
2016-05-14 22:38:37 +00:00
|
|
|
return 1.f;
|
|
|
|
}
|
2016-05-16 20:20:29 +00:00
|
|
|
double attackFac = m_curTime / m_attackTime;
|
2016-05-11 21:30:45 +00:00
|
|
|
if (attackFac >= 1.0)
|
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
m_phase = State::Decay;
|
|
|
|
m_curTime = 0.0;
|
2016-05-16 02:40:18 +00:00
|
|
|
m_releaseStartFactor = 1.f;
|
2016-05-11 21:30:45 +00:00
|
|
|
return 1.f;
|
|
|
|
}
|
|
|
|
m_releaseStartFactor = attackFac;
|
|
|
|
return attackFac;
|
|
|
|
}
|
|
|
|
case State::Decay:
|
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
if (m_decayTime == 0.0)
|
2016-05-14 22:38:37 +00:00
|
|
|
{
|
|
|
|
m_phase = State::Sustain;
|
2016-05-16 20:20:29 +00:00
|
|
|
m_curTime = 0.0;
|
2016-05-14 22:38:37 +00:00
|
|
|
m_releaseStartFactor = m_sustainFactor;
|
|
|
|
return m_sustainFactor;
|
|
|
|
}
|
2016-05-16 20:20:29 +00:00
|
|
|
double decayFac = m_curTime / m_decayTime;
|
2016-05-11 21:30:45 +00:00
|
|
|
if (decayFac >= 1.0)
|
|
|
|
{
|
|
|
|
m_phase = State::Sustain;
|
2016-05-16 20:20:29 +00:00
|
|
|
m_curTime = 0.0;
|
2016-05-11 21:30:45 +00:00
|
|
|
m_releaseStartFactor = m_sustainFactor;
|
|
|
|
return m_sustainFactor;
|
|
|
|
}
|
|
|
|
m_releaseStartFactor = (1.0 - decayFac) + decayFac * m_sustainFactor;
|
|
|
|
return m_releaseStartFactor;
|
|
|
|
}
|
|
|
|
case State::Sustain:
|
|
|
|
{
|
|
|
|
return m_sustainFactor;
|
|
|
|
}
|
|
|
|
case State::Release:
|
|
|
|
{
|
2016-05-16 20:20:29 +00:00
|
|
|
if (m_releaseTime == 0.0)
|
2016-05-15 21:56:23 +00:00
|
|
|
{
|
|
|
|
m_phase = State::Complete;
|
2016-05-14 22:38:37 +00:00
|
|
|
return 0.f;
|
2016-05-15 21:56:23 +00:00
|
|
|
}
|
2016-05-16 20:20:29 +00:00
|
|
|
double releaseFac = m_curTime / m_releaseTime;
|
2016-05-11 21:30:45 +00:00
|
|
|
if (releaseFac >= 1.0)
|
2016-05-15 21:56:23 +00:00
|
|
|
{
|
|
|
|
m_phase = State::Complete;
|
2016-05-11 21:30:45 +00:00
|
|
|
return 0.f;
|
2016-05-15 21:56:23 +00:00
|
|
|
}
|
2016-05-16 20:20:29 +00:00
|
|
|
return std::min(m_releaseStartFactor, 1.0 - releaseFac);
|
2016-05-11 21:30:45 +00:00
|
|
|
}
|
2016-05-15 21:56:23 +00:00
|
|
|
case State::Complete:
|
|
|
|
return 0.f;
|
2016-05-11 21:30:45 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|