mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-10 10:58:52 +00:00
213 lines
5 KiB
JavaScript
213 lines
5 KiB
JavaScript
define(["Tone/core/Tone", "Tone/signal/Signal"], function(Tone){
|
|
|
|
"use strict";
|
|
|
|
/**
|
|
* @class ADSR envelope generator attaches to an AudioParam or Signal
|
|
*
|
|
* @constructor
|
|
* @extends {Tone}
|
|
* @param {Tone.Time|Object=} attack
|
|
* @param {Tone.Time=} decay
|
|
* @param {number=} sustain a percentage (0-1) of the full amplitude
|
|
* @param {Tone.Time=} release
|
|
*/
|
|
Tone.Envelope = function(){
|
|
|
|
//get all of the defaults
|
|
var options = this.optionsObject(arguments, ["attack", "decay", "sustain", "release"], Tone.Envelope.defaults);
|
|
|
|
/**
|
|
* the output
|
|
* @type {GainNode}
|
|
*/
|
|
this.output = this.context.createGain();
|
|
|
|
/**
|
|
* the attack time in seconds
|
|
* @type {number}
|
|
*/
|
|
this.attack = options.attack;
|
|
|
|
/**
|
|
* the decay time in seconds
|
|
* @type {number}
|
|
*/
|
|
this.decay = options.decay;
|
|
|
|
/**
|
|
* the sustain is a value between 0-1
|
|
* @type {number}
|
|
*/
|
|
this.sustain = options.sustain;
|
|
|
|
/**
|
|
* the release time in seconds
|
|
* @type {number}
|
|
*/
|
|
this.release = options.release;
|
|
|
|
/**
|
|
* the minimum output of the envelope
|
|
* @type {number}
|
|
*/
|
|
this.min = options.min;
|
|
|
|
/**
|
|
* the maximum output of the envelope
|
|
* @type {number}
|
|
*/
|
|
this.max = options.max;
|
|
|
|
/**
|
|
* the control signal
|
|
* @type {Tone.Signal}
|
|
* @private
|
|
*/
|
|
this._control = new Tone.Signal(this.min);
|
|
|
|
//connections
|
|
this._control.connect(this.output);
|
|
};
|
|
|
|
Tone.extend(Tone.Envelope);
|
|
|
|
/**
|
|
* the default parameters
|
|
*
|
|
* @static
|
|
*/
|
|
Tone.Envelope.defaults = {
|
|
"attack" : 0.01,
|
|
"decay" : 0.1,
|
|
"sustain" : 0.5,
|
|
"release" : 1,
|
|
"min" : 0,
|
|
"max" : 1
|
|
};
|
|
|
|
// SETTERS //
|
|
|
|
/**
|
|
* set all of the parameters in bulk
|
|
* @param {Object} param the name of member as the key
|
|
* and the value as the value
|
|
*/
|
|
Tone.Envelope.prototype.set = function(params){
|
|
if (!this.isUndef(params.attack)) this.setAttack(params.attack);
|
|
if (!this.isUndef(params.decay)) this.setDecay(params.decay);
|
|
if (!this.isUndef(params.sustain)) this.setSustain(params.sustain);
|
|
if (!this.isUndef(params.release)) this.setRelease(params.release);
|
|
if (!this.isUndef(params.min)) this.setMin(params.min);
|
|
if (!this.isUndef(params.max)) this.setMax(params.max);
|
|
};
|
|
|
|
/**
|
|
* set the attack time
|
|
* @param {Tone.Time} time
|
|
*/
|
|
Tone.Envelope.prototype.setAttack = function(time){
|
|
this.attack = time;
|
|
};
|
|
|
|
/**
|
|
* set the decay time
|
|
* @param {Tone.Time} time
|
|
*/
|
|
Tone.Envelope.prototype.setDecay = function(time){
|
|
this.decay = time;
|
|
};
|
|
|
|
/**
|
|
* set the release time
|
|
* @param {Tone.Time} time
|
|
*/
|
|
Tone.Envelope.prototype.setRelease = function(time){
|
|
this.release = time;
|
|
};
|
|
|
|
/**
|
|
* set the sustain amount
|
|
* @param {number} sustain value between 0-1
|
|
*/
|
|
Tone.Envelope.prototype.setSustain = function(sustain){
|
|
this.sustain = sustain;
|
|
};
|
|
|
|
/**
|
|
* set the envelope max
|
|
* @param {number} max
|
|
*/
|
|
Tone.Envelope.prototype.setMax = function(max){
|
|
this.max = max;
|
|
};
|
|
|
|
/**
|
|
* set the envelope min
|
|
* @param {number} min
|
|
*/
|
|
Tone.Envelope.prototype.setMin = function(min){
|
|
this.min = min;
|
|
//should move the signal to the min
|
|
this._control.setValueAtTime(this.min, this.now());
|
|
};
|
|
|
|
/**
|
|
* attack->decay->sustain linear ramp
|
|
* @param {Tone.Time=} time
|
|
* @param {number=} [velocity=1] the velocity of the envelope scales the vales.
|
|
* number between 0-1
|
|
*/
|
|
Tone.Envelope.prototype.triggerAttack = function(time, velocity){
|
|
velocity = this.defaultArg(velocity, 1);
|
|
var attack = this.toSeconds(this.attack);
|
|
var decay = this.toSeconds(this.decay);
|
|
var scaledMax = this.max * velocity;
|
|
var sustainVal = (scaledMax - this.min) * this.sustain + this.min;
|
|
time = this.toSeconds(time);
|
|
this._control.cancelScheduledValues(time);
|
|
this._control.setTargetAtTime(scaledMax, time, attack / 4);
|
|
this._control.setTargetAtTime(sustainVal, time + attack, decay / 4);
|
|
};
|
|
|
|
/**
|
|
* triggers the release of the envelope with a linear ramp
|
|
* @param {Tone.Time=} time
|
|
*/
|
|
Tone.Envelope.prototype.triggerRelease = function(time){
|
|
time = this.toSeconds(time);
|
|
this._control.cancelScheduledValues(time);
|
|
var release = this.toSeconds(this.release);
|
|
this._control.setTargetAtTime(this.min, time, release / 4);
|
|
};
|
|
|
|
/**
|
|
* trigger the attack and release after a sustain time
|
|
* @param {Tone.Time} duration the duration of the note
|
|
* @param {Tone.Time=} time the time of the attack
|
|
* @param {number=} velocity the velocity of the note
|
|
*/
|
|
Tone.Envelope.prototype.triggerAttackRelease = function(duration, time, velocity) {
|
|
time = this.toSeconds(time);
|
|
this.triggerAttack(time, velocity);
|
|
this.triggerRelease(time + this.toSeconds(duration));
|
|
};
|
|
|
|
/**
|
|
* borrows the connect method from {@link Tone.Signal}
|
|
*
|
|
* @function
|
|
*/
|
|
Tone.Envelope.prototype.connect = Tone.Signal.prototype.connect;
|
|
|
|
/**
|
|
* disconnect and dispose
|
|
*/
|
|
Tone.Envelope.prototype.dispose = function(){
|
|
Tone.prototype.dispose.call(this);
|
|
this._control.dispose();
|
|
this._control = null;
|
|
};
|
|
|
|
return Tone.Envelope;
|
|
});
|