b3ADSR.cpp 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109
  1. #include "b3ADSR.h"
  2. //ADSR mostly copied/reimplemented from Stk, see
  3. //http://github.com/thestk/stk
  4. //! ADSR envelope states.
  5. enum
  6. {
  7. ADSR_ATTACK, /*!< Attack */
  8. ADSR_DECAY, /*!< Decay */
  9. ADSR_SUSTAIN, /*!< Sustain */
  10. ADSR_RELEASE, /*!< Release */
  11. ADSR_IDLE /*!< Before attack / after release */
  12. };
  13. b3ADSR::b3ADSR()
  14. {
  15. m_target = 0.0;
  16. m_value = 0.0;
  17. m_attackRate = 0.001;
  18. m_decayRate = 0.00001;
  19. m_releaseRate = 0.0005;
  20. m_sustainLevel = 0.5;
  21. m_state = ADSR_IDLE;
  22. m_autoKeyOff = false;
  23. }
  24. b3ADSR::~b3ADSR()
  25. {
  26. }
  27. double b3ADSR::tick()
  28. {
  29. switch (m_state)
  30. {
  31. case ADSR_ATTACK:
  32. m_value += m_attackRate;
  33. if (m_value >= m_target)
  34. {
  35. m_value = m_target;
  36. m_target = m_sustainLevel;
  37. m_state = ADSR_DECAY;
  38. }
  39. break;
  40. case ADSR_DECAY:
  41. if (m_value > m_sustainLevel)
  42. {
  43. m_value -= m_decayRate;
  44. if (m_value <= m_sustainLevel)
  45. {
  46. m_value = m_sustainLevel;
  47. m_state = ADSR_SUSTAIN;
  48. if (m_autoKeyOff)
  49. {
  50. keyOff();
  51. }
  52. }
  53. }
  54. else
  55. {
  56. m_value += m_decayRate; // attack target < sustain level
  57. if (m_value >= m_sustainLevel)
  58. {
  59. m_value = m_sustainLevel;
  60. m_state = ADSR_SUSTAIN;
  61. if (m_autoKeyOff)
  62. {
  63. keyOff();
  64. }
  65. }
  66. }
  67. break;
  68. case ADSR_RELEASE:
  69. m_value -= m_releaseRate;
  70. if (m_value <= 0.0)
  71. {
  72. m_value = 0.0;
  73. m_state = ADSR_IDLE;
  74. }
  75. }
  76. return m_value;
  77. }
  78. bool b3ADSR::isIdle() const
  79. {
  80. return m_state == ADSR_IDLE;
  81. }
  82. void b3ADSR::keyOn(bool autoKeyOff)
  83. {
  84. m_autoKeyOff = autoKeyOff;
  85. if (m_target <= 0.0)
  86. m_target = 1.0;
  87. if (m_attackRate == 1)
  88. {
  89. m_value = 1.0;
  90. }
  91. m_state = ADSR_ATTACK;
  92. }
  93. void b3ADSR::keyOff()
  94. {
  95. m_autoKeyOff = false;
  96. m_target = 0.0;
  97. m_state = ADSR_RELEASE;
  98. }