-
Notifications
You must be signed in to change notification settings - Fork 185
/
BlowBotl.cpp
122 lines (99 loc) · 2.96 KB
/
BlowBotl.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/***************************************************/
/*! \class BlowBotl
\brief STK blown bottle instrument class.
This class implements a helmholtz resonator
(biquad filter) with a polynomial jet
excitation (a la Cook).
Control Change Numbers:
- Noise Gain = 4
- Vibrato Frequency = 11
- Vibrato Gain = 1
- Volume = 128
by Perry R. Cook and Gary P. Scavone, 1995--2023.
*/
/***************************************************/
#include "BlowBotl.h"
#include "SKINImsg.h"
namespace stk {
#define __BOTTLE_RADIUS_ 0.999
BlowBotl :: BlowBotl( void )
{
dcBlock_.setBlockZero();
vibrato_.setFrequency( 5.925 );
vibratoGain_ = 0.0;
resonator_.setResonance( 500.0, __BOTTLE_RADIUS_, true );
adsr_.setAllTimes( 0.005, 0.01, 0.8, 0.010 );
noiseGain_ = 20.0;
maxPressure_ = 0.0;
}
BlowBotl :: ~BlowBotl( void )
{
}
void BlowBotl :: clear( void )
{
resonator_.clear();
}
void BlowBotl :: setFrequency( StkFloat frequency )
{
#if defined(_STK_DEBUG_)
if ( frequency <= 0.0 ) {
oStream_ << "BlowBotl::setFrequency: argument is less than or equal to zero!";
handleError( StkError::WARNING ); return;
}
#endif
resonator_.setResonance( frequency, __BOTTLE_RADIUS_, true );
}
void BlowBotl :: startBlowing( StkFloat amplitude, StkFloat rate )
{
if ( amplitude <= 0.0 || rate <= 0.0 ) {
oStream_ << "BlowBotl::startBowing: one or more arguments is less than or equal to zero!";
handleError( StkError::WARNING ); return;
}
adsr_.setAttackRate( rate );
maxPressure_ = amplitude;
adsr_.keyOn();
}
void BlowBotl :: stopBlowing( StkFloat rate )
{
if ( rate <= 0.0 ) {
oStream_ << "BlowBotl::stopBowing: argument is less than or equal to zero!";
handleError( StkError::WARNING ); return;
}
adsr_.setReleaseRate( rate );
adsr_.keyOff();
}
void BlowBotl :: noteOn( StkFloat frequency, StkFloat amplitude )
{
this->setFrequency( frequency );
startBlowing( 1.1 + (amplitude * 0.20), amplitude * 0.02);
outputGain_ = amplitude + 0.001;
}
void BlowBotl :: noteOff( StkFloat amplitude )
{
this->stopBlowing( amplitude * 0.02 );
}
void BlowBotl :: controlChange( int number, StkFloat value )
{
#if defined(_STK_DEBUG_)
if ( value < 0 || ( number != 101 && value > 128.0 ) ) {
oStream_ << "BlowBotl::controlChange: value (" << value << ") is out of range!";
handleError( StkError::WARNING ); return;
}
#endif
StkFloat normalizedValue = value * ONE_OVER_128;
if (number == __SK_NoiseLevel_) // 4
noiseGain_ = normalizedValue * 30.0;
else if (number == __SK_ModFrequency_) // 11
vibrato_.setFrequency( normalizedValue * 12.0 );
else if (number == __SK_ModWheel_) // 1
vibratoGain_ = normalizedValue * 0.4;
else if (number == __SK_AfterTouch_Cont_) // 128
adsr_.setTarget( normalizedValue );
#if defined(_STK_DEBUG_)
else {
oStream_ << "BlowBotl::controlChange: undefined control number (" << number << ")!";
handleError( StkError::WARNING );
}
#endif
}
} // stk namespace