2018-08-27 02:29:17 +00:00
|
|
|
define(["../core/Tone", "../signal/Signal",
|
|
|
|
"../signal/Pow", "../type/Type", "../core/AudioNode"], function(Tone){
|
2014-03-11 23:27:46 +00:00
|
|
|
|
2014-09-04 04:41:40 +00:00
|
|
|
"use strict";
|
|
|
|
|
2014-06-15 23:35:00 +00:00
|
|
|
/**
|
2015-07-04 19:25:37 +00:00
|
|
|
* @class Tone.Envelope is an [ADSR](https://en.wikipedia.org/wiki/Synthesizer#ADSR_envelope)
|
2017-08-27 21:50:31 +00:00
|
|
|
* envelope generator. Tone.Envelope outputs a signal which
|
|
|
|
* can be connected to an AudioParam or Tone.Signal.
|
2015-07-04 16:32:18 +00:00
|
|
|
* <img src="https://upload.wikimedia.org/wikipedia/commons/e/ea/ADSR_parameter.svg">
|
2014-06-16 23:59:58 +00:00
|
|
|
*
|
|
|
|
* @constructor
|
2017-08-27 21:50:31 +00:00
|
|
|
* @extends {Tone.AudioNode}
|
|
|
|
* @param {Time} [attack] The amount of time it takes for the envelope to go from
|
|
|
|
* 0 to it's maximum value.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @param {Time} [decay] The period of time after the attack that it takes for the envelope
|
2018-12-19 16:21:15 +00:00
|
|
|
* to fall to the sustain value. Value must be greater than 0.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @param {NormalRange} [sustain] The percent of the maximum value that the envelope rests at until
|
2017-08-27 21:50:31 +00:00
|
|
|
* the release is triggered.
|
|
|
|
* @param {Time} [release] The amount of time after the release is triggered it takes to reach 0.
|
2018-12-19 16:21:15 +00:00
|
|
|
* Value must be greater than 0.
|
2015-02-27 21:53:10 +00:00
|
|
|
* @example
|
2015-06-20 23:25:49 +00:00
|
|
|
* //an amplitude envelope
|
2015-06-14 03:56:32 +00:00
|
|
|
* var gainNode = Tone.context.createGain();
|
|
|
|
* var env = new Tone.Envelope({
|
|
|
|
* "attack" : 0.1,
|
|
|
|
* "decay" : 0.2,
|
|
|
|
* "sustain" : 1,
|
|
|
|
* "release" : 0.8,
|
|
|
|
* });
|
|
|
|
* env.connect(gainNode.gain);
|
2014-06-15 23:35:00 +00:00
|
|
|
*/
|
2014-08-24 23:28:42 +00:00
|
|
|
Tone.Envelope = function(){
|
2014-08-23 20:38:06 +00:00
|
|
|
|
|
|
|
//get all of the defaults
|
2017-04-26 03:08:23 +00:00
|
|
|
var options = Tone.defaults(arguments, ["attack", "decay", "sustain", "release"], Tone.Envelope);
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.AudioNode.call(this);
|
|
|
|
|
|
|
|
/**
|
2015-06-20 23:25:49 +00:00
|
|
|
* When triggerAttack is called, the attack time is the amount of
|
2017-08-27 21:50:31 +00:00
|
|
|
* time it takes for the envelope to reach it's maximum value.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @type {Time}
|
2014-08-23 20:38:06 +00:00
|
|
|
*/
|
2014-10-02 17:23:04 +00:00
|
|
|
this.attack = options.attack;
|
2014-08-23 20:38:06 +00:00
|
|
|
|
|
|
|
/**
|
2015-06-20 23:25:49 +00:00
|
|
|
* After the attack portion of the envelope, the value will fall
|
2017-08-27 21:50:31 +00:00
|
|
|
* over the duration of the decay time to it's sustain value.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @type {Time}
|
2014-08-23 20:38:06 +00:00
|
|
|
*/
|
2014-10-02 17:23:04 +00:00
|
|
|
this.decay = options.decay;
|
2017-08-27 21:50:31 +00:00
|
|
|
|
2014-08-23 20:38:06 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* The sustain value is the value
|
2015-06-20 23:25:49 +00:00
|
|
|
* which the envelope rests at after triggerAttack is
|
2017-08-27 21:50:31 +00:00
|
|
|
* called, but before triggerRelease is invoked.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @type {NormalRange}
|
2014-08-23 20:38:06 +00:00
|
|
|
*/
|
2014-10-02 17:23:04 +00:00
|
|
|
this.sustain = options.sustain;
|
2014-08-23 20:38:06 +00:00
|
|
|
|
|
|
|
/**
|
2015-06-20 23:25:49 +00:00
|
|
|
* After triggerRelease is called, the envelope's
|
|
|
|
* value will fall to it's miminum value over the
|
2017-08-27 21:50:31 +00:00
|
|
|
* duration of the release time.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @type {Time}
|
2014-08-23 20:38:06 +00:00
|
|
|
*/
|
2014-10-02 17:23:04 +00:00
|
|
|
this.release = options.release;
|
2014-08-23 20:38:06 +00:00
|
|
|
|
2015-04-18 18:54:51 +00:00
|
|
|
/**
|
|
|
|
* the next time the envelope is at standby
|
|
|
|
* @type {number}
|
|
|
|
* @private
|
|
|
|
*/
|
2016-05-23 23:20:20 +00:00
|
|
|
this._attackCurve = "linear";
|
2015-04-18 18:54:51 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* the next time the envelope is at standby
|
|
|
|
* @type {number}
|
|
|
|
* @private
|
|
|
|
*/
|
2016-05-23 23:20:20 +00:00
|
|
|
this._releaseCurve = "exponential";
|
2015-04-18 18:54:51 +00:00
|
|
|
|
2014-08-23 20:38:06 +00:00
|
|
|
/**
|
2014-11-02 01:54:40 +00:00
|
|
|
* the signal
|
2017-12-30 01:42:01 +00:00
|
|
|
* @type {Tone.Signal}
|
2014-11-02 01:54:40 +00:00
|
|
|
* @private
|
2014-08-23 20:38:06 +00:00
|
|
|
*/
|
2017-10-26 04:51:31 +00:00
|
|
|
this._sig = this.output = new Tone.Signal(0);
|
2015-04-18 18:54:51 +00:00
|
|
|
|
2015-04-20 14:42:27 +00:00
|
|
|
//set the attackCurve initially
|
|
|
|
this.attackCurve = options.attackCurve;
|
2015-08-16 19:17:35 +00:00
|
|
|
this.releaseCurve = options.releaseCurve;
|
2018-05-28 22:51:44 +00:00
|
|
|
this.decayCurve = options.decayCurve;
|
2014-06-15 21:37:55 +00:00
|
|
|
};
|
2014-03-11 23:27:46 +00:00
|
|
|
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.extend(Tone.Envelope, Tone.AudioNode);
|
2014-04-06 00:47:59 +00:00
|
|
|
|
2014-08-23 20:38:06 +00:00
|
|
|
/**
|
|
|
|
* the default parameters
|
|
|
|
* @static
|
2015-02-06 22:49:04 +00:00
|
|
|
* @const
|
2014-08-23 20:38:06 +00:00
|
|
|
*/
|
2014-08-25 14:23:37 +00:00
|
|
|
Tone.Envelope.defaults = {
|
2014-08-23 20:38:06 +00:00
|
|
|
"attack" : 0.01,
|
|
|
|
"decay" : 0.1,
|
|
|
|
"sustain" : 0.5,
|
|
|
|
"release" : 1,
|
2015-08-16 19:17:35 +00:00
|
|
|
"attackCurve" : "linear",
|
2018-05-28 22:51:44 +00:00
|
|
|
"decayCurve" : "exponential",
|
2015-08-16 19:17:35 +00:00
|
|
|
"releaseCurve" : "exponential",
|
2014-08-23 20:38:06 +00:00
|
|
|
};
|
|
|
|
|
2015-07-04 17:36:38 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Read the current value of the envelope. Useful for
|
|
|
|
* syncronizing visual output to the envelope.
|
2015-07-04 17:36:38 +00:00
|
|
|
* @memberOf Tone.Envelope#
|
|
|
|
* @type {Number}
|
|
|
|
* @name value
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Envelope.prototype, "value", {
|
|
|
|
get : function(){
|
2016-03-03 18:04:02 +00:00
|
|
|
return this.getValueAtTime(this.now());
|
2015-07-04 17:36:38 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-05-28 22:51:44 +00:00
|
|
|
/**
|
|
|
|
* Get the curve
|
|
|
|
* @param {Array|String} curve
|
|
|
|
* @param {String} direction In/Out
|
|
|
|
* @return {String} The curve name
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Envelope.prototype._getCurve = function(curve, direction){
|
|
|
|
if (Tone.isString(curve)){
|
|
|
|
return curve;
|
|
|
|
} else if (Tone.isArray(curve)){
|
|
|
|
//look up the name in the curves array
|
|
|
|
for (var t in Tone.Envelope.Type){
|
|
|
|
if (Tone.Envelope.Type[t][direction] === curve){
|
|
|
|
return t;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//otherwise just return the array
|
|
|
|
return curve;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Assign a the curve to the given name using the direction
|
|
|
|
* @param {String} name
|
|
|
|
* @param {String} direction In/Out
|
|
|
|
* @param {Array} curve
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Envelope.prototype._setCurve = function(name, direction, curve){
|
|
|
|
//check if it's a valid type
|
|
|
|
if (Tone.Envelope.Type.hasOwnProperty(curve)){
|
|
|
|
var curveDef = Tone.Envelope.Type[curve];
|
|
|
|
if (Tone.isObject(curveDef)){
|
|
|
|
this[name] = curveDef[direction];
|
|
|
|
} else {
|
|
|
|
this[name] = curveDef;
|
|
|
|
}
|
|
|
|
} else if (Tone.isArray(curve)){
|
|
|
|
this[name] = curve;
|
|
|
|
} else {
|
|
|
|
throw new Error("Tone.Envelope: invalid curve: " + curve);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-04-18 18:54:51 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* The shape of the attack.
|
2016-05-23 23:20:20 +00:00
|
|
|
* Can be any of these strings:
|
|
|
|
* <ul>
|
|
|
|
* <li>linear</li>
|
|
|
|
* <li>exponential</li>
|
|
|
|
* <li>sine</li>
|
2016-11-23 06:38:58 +00:00
|
|
|
* <li>cosine</li>
|
2016-05-23 23:20:20 +00:00
|
|
|
* <li>bounce</li>
|
|
|
|
* <li>ripple</li>
|
|
|
|
* <li>step</li>
|
|
|
|
* </ul>
|
|
|
|
* Can also be an array which describes the curve. Values
|
|
|
|
* in the array are evenly subdivided and linearly
|
2017-08-27 21:50:31 +00:00
|
|
|
* interpolated over the duration of the attack.
|
2015-04-18 18:54:51 +00:00
|
|
|
* @memberOf Tone.Envelope#
|
2016-05-23 23:20:20 +00:00
|
|
|
* @type {String|Array}
|
2015-04-20 14:42:27 +00:00
|
|
|
* @name attackCurve
|
2015-04-18 18:54:51 +00:00
|
|
|
* @example
|
2015-04-20 14:42:27 +00:00
|
|
|
* env.attackCurve = "linear";
|
2016-05-23 23:20:20 +00:00
|
|
|
* @example
|
|
|
|
* //can also be an array
|
|
|
|
* env.attackCurve = [0, 0.2, 0.3, 0.4, 1]
|
2015-04-18 18:54:51 +00:00
|
|
|
*/
|
2015-04-20 14:42:27 +00:00
|
|
|
Object.defineProperty(Tone.Envelope.prototype, "attackCurve", {
|
2015-04-18 18:54:51 +00:00
|
|
|
get : function(){
|
2018-05-28 22:51:44 +00:00
|
|
|
return this._getCurve(this._attackCurve, "In");
|
2017-08-27 21:50:31 +00:00
|
|
|
},
|
2016-05-23 23:20:20 +00:00
|
|
|
set : function(curve){
|
2018-05-28 22:51:44 +00:00
|
|
|
this._setCurve("_attackCurve", "In", curve);
|
2015-04-18 18:54:51 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* The shape of the release. See the attack curve types.
|
2015-08-16 19:17:35 +00:00
|
|
|
* @memberOf Tone.Envelope#
|
2016-05-23 23:20:20 +00:00
|
|
|
* @type {String|Array}
|
2015-08-16 19:17:35 +00:00
|
|
|
* @name releaseCurve
|
|
|
|
* @example
|
|
|
|
* env.releaseCurve = "linear";
|
2015-04-18 18:54:51 +00:00
|
|
|
*/
|
2015-08-16 19:17:35 +00:00
|
|
|
Object.defineProperty(Tone.Envelope.prototype, "releaseCurve", {
|
|
|
|
get : function(){
|
2018-05-28 22:51:44 +00:00
|
|
|
return this._getCurve(this._releaseCurve, "Out");
|
2017-08-27 21:50:31 +00:00
|
|
|
},
|
2016-05-23 23:20:20 +00:00
|
|
|
set : function(curve){
|
2018-05-28 22:51:44 +00:00
|
|
|
this._setCurve("_releaseCurve", "Out", curve);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The shape of the decay either "linear" or "exponential"
|
|
|
|
* @memberOf Tone.Envelope#
|
|
|
|
* @type {String}
|
|
|
|
* @name decayCurve
|
|
|
|
* @example
|
|
|
|
* env.decayCurve = "linear";
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Envelope.prototype, "decayCurve", {
|
|
|
|
get : function(){
|
|
|
|
return this._decayCurve;
|
|
|
|
},
|
|
|
|
set : function(curve){
|
|
|
|
var curves = ["linear", "exponential"];
|
|
|
|
if (!curves.includes(curve)){
|
2016-05-26 00:50:46 +00:00
|
|
|
throw new Error("Tone.Envelope: invalid curve: " + curve);
|
2018-05-28 22:51:44 +00:00
|
|
|
} else {
|
|
|
|
this._decayCurve = curve;
|
2015-04-18 18:54:51 +00:00
|
|
|
}
|
|
|
|
}
|
2015-08-16 19:17:35 +00:00
|
|
|
});
|
2015-04-18 18:54:51 +00:00
|
|
|
|
2014-06-15 23:35:00 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Trigger the attack/decay portion of the ADSR envelope.
|
2015-06-20 23:25:49 +00:00
|
|
|
* @param {Time} [time=now] When the attack should start.
|
|
|
|
* @param {NormalRange} [velocity=1] The velocity of the envelope scales the vales.
|
2014-09-04 02:36:56 +00:00
|
|
|
* number between 0-1
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Envelope} this
|
2015-02-27 21:53:10 +00:00
|
|
|
* @example
|
|
|
|
* //trigger the attack 0.5 seconds from now with a velocity of 0.2
|
|
|
|
* env.triggerAttack("+0.5", 0.2);
|
2014-06-15 23:35:00 +00:00
|
|
|
*/
|
2014-09-04 02:36:56 +00:00
|
|
|
Tone.Envelope.prototype.triggerAttack = function(time, velocity){
|
2018-06-20 18:56:11 +00:00
|
|
|
this.log("triggerAttack", time, velocity);
|
2016-09-24 00:07:41 +00:00
|
|
|
time = this.toSeconds(time);
|
2016-05-23 23:20:20 +00:00
|
|
|
var originalAttack = this.toSeconds(this.attack);
|
|
|
|
var attack = originalAttack;
|
2014-10-02 17:23:04 +00:00
|
|
|
var decay = this.toSeconds(this.decay);
|
2017-04-30 19:03:49 +00:00
|
|
|
velocity = Tone.defaultArg(velocity, 1);
|
2016-02-01 04:40:50 +00:00
|
|
|
//check if it's not a complete attack
|
|
|
|
var currentValue = this.getValueAtTime(time);
|
|
|
|
if (currentValue > 0){
|
|
|
|
//subtract the current value from the attack time
|
|
|
|
var attackRate = 1 / attack;
|
|
|
|
var remainingDistance = 1 - currentValue;
|
|
|
|
//the attack is now the remaining time
|
|
|
|
attack = remainingDistance / attackRate;
|
|
|
|
}
|
2015-08-16 19:17:35 +00:00
|
|
|
//attack
|
2016-05-23 23:20:20 +00:00
|
|
|
if (this._attackCurve === "linear"){
|
2017-08-31 16:45:25 +00:00
|
|
|
this._sig.linearRampTo(velocity, attack, time);
|
2016-05-23 23:20:20 +00:00
|
|
|
} else if (this._attackCurve === "exponential"){
|
2017-09-13 15:04:13 +00:00
|
|
|
this._sig.targetRampTo(velocity, attack, time);
|
2016-05-23 23:20:20 +00:00
|
|
|
} else if (attack > 0){
|
2017-10-26 04:51:31 +00:00
|
|
|
this._sig.cancelAndHoldAtTime(time);
|
2016-05-23 23:20:20 +00:00
|
|
|
var curve = this._attackCurve;
|
2018-05-28 21:59:31 +00:00
|
|
|
//find the starting position in the curve
|
|
|
|
for (var i = 1; i < curve.length; i++){
|
|
|
|
//the starting index is between the two values
|
|
|
|
if (curve[i-1] <= currentValue && currentValue <= curve[i]){
|
|
|
|
curve = this._attackCurve.slice(i);
|
|
|
|
//the first index is the current value
|
|
|
|
curve[0] = currentValue;
|
|
|
|
break;
|
|
|
|
}
|
2016-05-23 23:20:20 +00:00
|
|
|
}
|
|
|
|
this._sig.setValueCurveAtTime(curve, time, attack, velocity);
|
2015-04-18 18:54:51 +00:00
|
|
|
}
|
2015-08-16 19:17:35 +00:00
|
|
|
//decay
|
2017-10-26 04:51:31 +00:00
|
|
|
if (decay){
|
2018-05-28 22:51:44 +00:00
|
|
|
var decayValue = velocity * this.sustain;
|
|
|
|
var decayStart = time + attack;
|
2018-06-13 03:48:51 +00:00
|
|
|
this.log("decay", decayStart);
|
2018-05-28 22:51:44 +00:00
|
|
|
if (this._decayCurve === "linear"){
|
2018-12-21 20:17:39 +00:00
|
|
|
this._sig.linearRampTo(decayValue, decay, decayStart+this.sampleTime);
|
2018-05-28 22:51:44 +00:00
|
|
|
} else if (this._decayCurve === "exponential"){
|
2018-06-13 03:48:51 +00:00
|
|
|
this._sig.exponentialApproachValueAtTime(decayValue, decayStart, decay);
|
2018-05-28 22:51:44 +00:00
|
|
|
}
|
2017-10-26 04:51:31 +00:00
|
|
|
}
|
2015-02-02 17:49:13 +00:00
|
|
|
return this;
|
2014-06-15 21:37:55 +00:00
|
|
|
};
|
2017-08-27 21:50:31 +00:00
|
|
|
|
2014-06-15 23:35:00 +00:00
|
|
|
/**
|
2015-02-27 21:53:10 +00:00
|
|
|
* Triggers the release of the envelope.
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {Time} [time=now] When the release portion of the envelope should start.
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Envelope} this
|
2015-02-27 21:53:10 +00:00
|
|
|
* @example
|
|
|
|
* //trigger release immediately
|
|
|
|
* env.triggerRelease();
|
2014-06-15 23:35:00 +00:00
|
|
|
*/
|
2014-04-06 00:47:59 +00:00
|
|
|
Tone.Envelope.prototype.triggerRelease = function(time){
|
2018-06-20 18:56:11 +00:00
|
|
|
this.log("triggerRelease", time);
|
2016-09-24 00:07:41 +00:00
|
|
|
time = this.toSeconds(time);
|
2016-05-23 23:20:20 +00:00
|
|
|
var currentValue = this.getValueAtTime(time);
|
|
|
|
if (currentValue > 0){
|
2016-01-30 20:56:56 +00:00
|
|
|
var release = this.toSeconds(this.release);
|
2016-05-23 23:20:20 +00:00
|
|
|
if (this._releaseCurve === "linear"){
|
2017-08-31 16:45:25 +00:00
|
|
|
this._sig.linearRampTo(0, release, time);
|
2016-05-23 23:20:20 +00:00
|
|
|
} else if (this._releaseCurve === "exponential"){
|
2017-09-13 15:04:13 +00:00
|
|
|
this._sig.targetRampTo(0, release, time);
|
2018-01-02 15:37:27 +00:00
|
|
|
} else {
|
2016-05-23 23:20:20 +00:00
|
|
|
var curve = this._releaseCurve;
|
2017-04-26 04:24:19 +00:00
|
|
|
if (Tone.isArray(curve)){
|
2017-10-26 04:51:31 +00:00
|
|
|
this._sig.cancelAndHoldAtTime(time);
|
2016-06-17 17:13:20 +00:00
|
|
|
this._sig.setValueCurveAtTime(curve, time, release, currentValue);
|
2016-05-23 23:20:20 +00:00
|
|
|
}
|
2016-01-30 20:56:56 +00:00
|
|
|
}
|
2015-04-18 18:54:51 +00:00
|
|
|
}
|
2015-02-02 17:49:13 +00:00
|
|
|
return this;
|
2014-06-15 21:37:55 +00:00
|
|
|
};
|
2014-04-11 23:17:01 +00:00
|
|
|
|
2016-01-30 20:56:56 +00:00
|
|
|
/**
|
|
|
|
* Get the scheduled value at the given time. This will
|
|
|
|
* return the unconverted (raw) value.
|
|
|
|
* @param {Number} time The time in seconds.
|
|
|
|
* @return {Number} The scheduled value at the given time.
|
|
|
|
*/
|
|
|
|
Tone.Envelope.prototype.getValueAtTime = function(time){
|
|
|
|
return this._sig.getValueAtTime(time);
|
|
|
|
};
|
|
|
|
|
2014-09-20 19:18:36 +00:00
|
|
|
/**
|
2015-06-20 23:25:49 +00:00
|
|
|
* triggerAttackRelease is shorthand for triggerAttack, then waiting
|
2017-08-27 21:50:31 +00:00
|
|
|
* some duration, then triggerRelease.
|
2015-06-20 23:25:49 +00:00
|
|
|
* @param {Time} duration The duration of the sustain.
|
|
|
|
* @param {Time} [time=now] When the attack should be triggered.
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {number} [velocity=1] The velocity of the envelope.
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Envelope} this
|
2015-02-27 21:53:10 +00:00
|
|
|
* @example
|
2015-06-20 23:25:49 +00:00
|
|
|
* //trigger the attack and then the release after 0.6 seconds.
|
|
|
|
* env.triggerAttackRelease(0.6);
|
2014-09-20 19:18:36 +00:00
|
|
|
*/
|
2018-03-05 17:25:33 +00:00
|
|
|
Tone.Envelope.prototype.triggerAttackRelease = function(duration, time, velocity){
|
2014-09-20 19:18:36 +00:00
|
|
|
time = this.toSeconds(time);
|
|
|
|
this.triggerAttack(time, velocity);
|
|
|
|
this.triggerRelease(time + this.toSeconds(duration));
|
2015-02-02 17:49:13 +00:00
|
|
|
return this;
|
2014-09-20 19:18:36 +00:00
|
|
|
};
|
|
|
|
|
2016-03-04 20:54:24 +00:00
|
|
|
/**
|
|
|
|
* Cancels all scheduled envelope changes after the given time.
|
|
|
|
* @param {Time} after
|
|
|
|
* @returns {Tone.Envelope} this
|
|
|
|
*/
|
2018-03-05 17:25:33 +00:00
|
|
|
Tone.Envelope.prototype.cancel = function(after){
|
2016-03-04 20:54:24 +00:00
|
|
|
this._sig.cancelScheduledValues(after);
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
2014-06-15 23:35:00 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Borrows the connect method from Tone.Signal.
|
2014-08-23 20:38:06 +00:00
|
|
|
* @function
|
2015-06-20 23:25:49 +00:00
|
|
|
* @private
|
2014-06-15 23:35:00 +00:00
|
|
|
*/
|
2018-02-04 16:40:47 +00:00
|
|
|
Tone.Envelope.prototype.connect = Tone.SignalBase.prototype.connect;
|
2014-04-06 00:47:59 +00:00
|
|
|
|
2017-10-21 23:02:46 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Generate some complex envelope curves.
|
2016-05-23 23:20:20 +00:00
|
|
|
*/
|
|
|
|
(function _createCurves(){
|
|
|
|
|
|
|
|
var curveLen = 128;
|
|
|
|
|
|
|
|
var i, k;
|
|
|
|
|
2016-06-12 13:56:18 +00:00
|
|
|
//cosine curve
|
|
|
|
var cosineCurve = [];
|
2016-05-23 23:20:20 +00:00
|
|
|
for (i = 0; i < curveLen; i++){
|
2016-06-12 13:56:18 +00:00
|
|
|
cosineCurve[i] = Math.sin((i / (curveLen - 1)) * (Math.PI / 2));
|
2016-05-23 23:20:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//ripple curve
|
|
|
|
var rippleCurve = [];
|
|
|
|
var rippleCurveFreq = 6.4;
|
|
|
|
for (i = 0; i < curveLen - 1; i++){
|
|
|
|
k = (i / (curveLen - 1));
|
2017-10-26 19:26:04 +00:00
|
|
|
var sineWave = Math.sin(k * (Math.PI * 2) * rippleCurveFreq - Math.PI / 2) + 1;
|
2016-05-23 23:20:20 +00:00
|
|
|
rippleCurve[i] = sineWave/10 + k * 0.83;
|
|
|
|
}
|
|
|
|
rippleCurve[curveLen - 1] = 1;
|
|
|
|
|
|
|
|
//stairs curve
|
|
|
|
var stairsCurve = [];
|
|
|
|
var steps = 5;
|
|
|
|
for (i = 0; i < curveLen; i++){
|
|
|
|
stairsCurve[i] = Math.ceil((i / (curveLen - 1)) * steps) / steps;
|
2017-08-27 21:50:31 +00:00
|
|
|
}
|
2016-05-23 23:20:20 +00:00
|
|
|
|
|
|
|
//in-out easing curve
|
2016-06-12 13:56:18 +00:00
|
|
|
var sineCurve = [];
|
2016-05-23 23:20:20 +00:00
|
|
|
for (i = 0; i < curveLen; i++){
|
|
|
|
k = i / (curveLen - 1);
|
2016-06-12 13:56:18 +00:00
|
|
|
sineCurve[i] = 0.5 * (1 - Math.cos(Math.PI * k));
|
2016-05-23 23:20:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
//a bounce curve
|
|
|
|
var bounceCurve = [];
|
|
|
|
for (i = 0; i < curveLen; i++){
|
|
|
|
k = i / (curveLen - 1);
|
|
|
|
var freq = Math.pow(k, 3) * 4 + 0.2;
|
|
|
|
var val = Math.cos(freq * Math.PI * 2 * k);
|
|
|
|
bounceCurve[i] = Math.abs(val * (1 - k));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invert a value curve to make it work for the release
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
function invertCurve(curve){
|
|
|
|
var out = new Array(curve.length);
|
|
|
|
for (var j = 0; j < curve.length; j++){
|
|
|
|
out[j] = 1 - curve[j];
|
|
|
|
}
|
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* reverse the curve
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
function reverseCurve(curve){
|
|
|
|
return curve.slice(0).reverse();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* attack and release curve arrays
|
|
|
|
* @type {Object}
|
|
|
|
* @private
|
|
|
|
*/
|
2017-10-21 23:02:46 +00:00
|
|
|
Tone.Envelope.Type = {
|
|
|
|
"linear" : "linear",
|
|
|
|
"exponential" : "exponential",
|
2016-05-23 23:20:20 +00:00
|
|
|
"bounce" : {
|
|
|
|
In : invertCurve(bounceCurve),
|
|
|
|
Out : bounceCurve
|
|
|
|
},
|
2016-06-12 13:56:18 +00:00
|
|
|
"cosine" : {
|
|
|
|
In : cosineCurve,
|
|
|
|
Out : reverseCurve(cosineCurve)
|
2016-05-23 23:20:20 +00:00
|
|
|
},
|
|
|
|
"step" : {
|
|
|
|
In : stairsCurve,
|
|
|
|
Out : invertCurve(stairsCurve)
|
|
|
|
},
|
|
|
|
"ripple" : {
|
|
|
|
In : rippleCurve,
|
|
|
|
Out : invertCurve(rippleCurve)
|
|
|
|
},
|
2016-06-12 13:56:18 +00:00
|
|
|
"sine" : {
|
|
|
|
In : sineCurve,
|
|
|
|
Out : invertCurve(sineCurve)
|
2016-05-23 23:20:20 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
})();
|
|
|
|
|
2014-06-20 05:23:35 +00:00
|
|
|
/**
|
2015-06-20 23:25:49 +00:00
|
|
|
* Disconnect and dispose.
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Envelope} this
|
2014-06-20 05:23:35 +00:00
|
|
|
*/
|
|
|
|
Tone.Envelope.prototype.dispose = function(){
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.AudioNode.prototype.dispose.call(this);
|
2014-11-02 01:54:40 +00:00
|
|
|
this._sig.dispose();
|
|
|
|
this._sig = null;
|
2016-05-23 23:20:20 +00:00
|
|
|
this._attackCurve = null;
|
|
|
|
this._releaseCurve = null;
|
2015-02-02 17:49:13 +00:00
|
|
|
return this;
|
2014-06-20 05:23:35 +00:00
|
|
|
};
|
|
|
|
|
2014-04-06 00:47:59 +00:00
|
|
|
return Tone.Envelope;
|
|
|
|
});
|