Tone.js/Tone/signal/Signal.js

326 lines
9.1 KiB
JavaScript
Raw Normal View History

2014-07-30 17:56:32 +00:00
define(["Tone/core/Tone"], function(Tone){
"use strict";
2014-06-14 23:11:37 +00:00
/**
2014-07-04 17:47:56 +00:00
* @class Constant audio-rate signal.
* Tone.Signal is a core component which allows for sample-accurate
* synchronization of many components. Tone.Signal can be scheduled
* with all of the functions available to AudioParams
2014-06-17 16:15:10 +00:00
*
2014-06-14 23:11:37 +00:00
* @constructor
2014-06-19 17:38:21 +00:00
* @extends {Tone}
2014-06-14 23:11:37 +00:00
* @param {number=} value (optional) initial value
*/
Tone.Signal = function(value){
Tone.call(this);
/**
* scales the constant output to the desired output
* @type {GainNode}
2014-07-30 17:56:32 +00:00
* @private
*/
2014-08-23 18:22:18 +00:00
this._scalar = this.context.createGain();
/**
* the ratio of the this value to the control signal value
*
* @private
* @type {number}
*/
2014-06-14 23:11:37 +00:00
this._syncRatio = 1;
/**
* the value of the Signal
* @type {number}
*/
this.value = this.defaultArg(value, 0);
//connect the constant 1 output to the node output
2014-08-23 18:22:18 +00:00
this.chain(constant, this._scalar, this.output);
//signal passes through
this.input.connect(this.output);
2014-06-14 23:11:37 +00:00
};
Tone.extend(Tone.Signal);
2014-06-14 23:11:37 +00:00
/**
* @return {number} the current value of the signal
*/
Tone.Signal.prototype.getValue = function(){
2014-08-23 18:22:18 +00:00
return this._scalar.gain.value;
2014-06-14 23:11:37 +00:00
};
/**
* set the value of the signal right away
* will be overwritten if there are previously scheduled automation curves
*
* @param {number} value
*/
Tone.Signal.prototype.setValue = function(value){
if (this._syncRatio === 0){
value = 0;
} else {
value *= this._syncRatio;
}
2014-08-23 18:22:18 +00:00
this._scalar.gain.value = value;
2014-06-14 23:11:37 +00:00
};
/**
* Schedules a parameter value change at the given time.
*
* @param {number} value
* @param {Tone.Time} time
2014-06-14 23:11:37 +00:00
*/
Tone.Signal.prototype.setValueAtTime = function(value, time){
2014-06-14 23:11:37 +00:00
value *= this._syncRatio;
2014-08-23 18:22:18 +00:00
this._scalar.gain.setValueAtTime(value, this.toSeconds(time));
2014-06-14 23:11:37 +00:00
};
/**
* creates a schedule point with the current value at the current time
*
* @param {number=} now (optionally) pass the now value in
* @returns {number} the current value
*/
Tone.Signal.prototype.setCurrentValueNow = function(now){
now = this.defaultArg(now, this.now());
var currentVal = this.getValue();
this.cancelScheduledValues(now);
2014-08-23 18:22:18 +00:00
this._scalar.gain.setValueAtTime(currentVal, now);
return currentVal;
};
2014-06-14 23:11:37 +00:00
/**
* Schedules a linear continuous change in parameter value from the
* previous scheduled parameter value to the given value.
*
* @param {number} value
2014-06-15 21:38:59 +00:00
* @param {Tone.Time} endTime
2014-06-14 23:11:37 +00:00
*/
Tone.Signal.prototype.linearRampToValueAtTime = function(value, endTime){
2014-06-14 23:11:37 +00:00
value *= this._syncRatio;
2014-08-23 18:22:18 +00:00
this._scalar.gain.linearRampToValueAtTime(value, this.toSeconds(endTime));
2014-06-14 23:11:37 +00:00
};
/**
* Schedules an exponential continuous change in parameter value from
* the previous scheduled parameter value to the given value.
2014-06-21 17:09:28 +00:00
*
* NOTE: Chrome will throw an error if you try to exponentially ramp to a
* value 0 or less.
2014-06-14 23:11:37 +00:00
*
* @param {number} value
2014-06-15 21:38:59 +00:00
* @param {Tone.Time} endTime
2014-06-14 23:11:37 +00:00
*/
Tone.Signal.prototype.exponentialRampToValueAtTime = function(value, endTime){
2014-06-14 23:11:37 +00:00
value *= this._syncRatio;
try {
this._scalar.gain.exponentialRampToValueAtTime(value, this.toSeconds(endTime));
} catch(e){
//firefox won't let the signal ramp past 1, in these cases, revert to linear ramp
this._scalar.gain.linearRampToValueAtTime(value, this.toSeconds(endTime));
}
2014-06-14 23:11:37 +00:00
};
/**
* Schedules an exponential continuous change in parameter value from
* the current time and current value to the given value.
*
* @param {number} value
* @param {Tone.Time} endTime
*/
Tone.Signal.prototype.exponentialRampToValueNow = function(value, endTime){
var now = this.now();
this.setCurrentValueNow(now);
//make sure that the endTime doesn't start with +
if (endTime.toString().charAt(0) === "+"){
endTime = endTime.substr(1);
}
this.exponentialRampToValueAtTime(value, now + this.toSeconds(endTime));
};
/**
* Schedules an linear continuous change in parameter value from
* the current time and current value to the given value at the given time.
*
* @param {number} value
* @param {Tone.Time} endTime
*/
2014-06-18 16:50:00 +00:00
Tone.Signal.prototype.linearRampToValueNow = function(value, endTime){
var now = this.now();
this.setCurrentValueNow(now);
value *= this._syncRatio;
//make sure that the endTime doesn't start with +
if (endTime.toString().charAt(0) === "+"){
endTime = endTime.substr(1);
}
2014-08-23 18:22:18 +00:00
this._scalar.gain.linearRampToValueAtTime(value, now + this.toSeconds(endTime));
};
2014-06-14 23:11:37 +00:00
/**
* Start exponentially approaching the target value at the given time with
* a rate having the given time constant.
*
* @param {number} value
2014-06-15 21:38:59 +00:00
* @param {Tone.Time} startTime
2014-06-14 23:11:37 +00:00
* @param {number} timeConstant
*/
Tone.Signal.prototype.setTargetAtTime = function(value, startTime, timeConstant){
value *= this._syncRatio;
2014-09-03 21:31:51 +00:00
this._scalar.gain.setTargetAtTime(value, this.toSeconds(startTime), timeConstant);
2014-06-14 23:11:37 +00:00
};
/**
* Sets an array of arbitrary parameter values starting at the given time
* for the given duration.
*
* @param {Array<number>} values
2014-06-15 21:38:59 +00:00
* @param {Tone.Time} startTime
* @param {Tone.Time} duration
2014-06-14 23:11:37 +00:00
*/
Tone.Signal.prototype.setValueCurveAtTime = function(values, startTime, duration){
2014-06-14 23:11:37 +00:00
for (var i = 0; i < values.length; i++){
values[i] *= this._syncRatio;
}
2014-08-23 18:22:18 +00:00
this._scalar.gain.setValueCurveAtTime(values, this.toSeconds(startTime), this.toSeconds(duration));
2014-06-14 23:11:37 +00:00
};
/**
* Cancels all scheduled parameter changes with times greater than or
* equal to startTime.
*
2014-06-15 21:38:59 +00:00
* @param {Tone.Time} startTime
2014-06-14 23:11:37 +00:00
*/
Tone.Signal.prototype.cancelScheduledValues = function(startTime){
2014-08-23 18:22:18 +00:00
this._scalar.gain.cancelScheduledValues(this.toSeconds(startTime));
2014-06-14 23:11:37 +00:00
};
/**
* Sync this to another signal and it will always maintain the
* ratio between the two signals until it is unsynced
2014-06-20 05:23:47 +00:00
*
* Signals can only be synced to one other signal. while syncing,
* if a signal's value is changed, the new ratio between the signals
* is maintained as the syncing signal is changed.
2014-06-14 23:11:37 +00:00
*
* @param {Tone.Signal} signal to sync to
2014-06-20 05:23:47 +00:00
* @param {number=} ratio optionally pass in the ratio between
* the two signals, otherwise it will be computed
2014-06-14 23:11:37 +00:00
*/
2014-06-20 05:23:47 +00:00
Tone.Signal.prototype.sync = function(signal, ratio){
if (ratio){
this._syncRatio = ratio;
2014-06-14 23:11:37 +00:00
} else {
2014-06-20 05:23:47 +00:00
//get the sync ratio
if (signal.getValue() !== 0){
this._syncRatio = this.getValue() / signal.getValue();
} else {
this._syncRatio = 0;
}
2014-06-14 23:11:37 +00:00
}
//make a new scalar which is not connected to the constant signal
2014-08-23 18:22:18 +00:00
this._scalar.disconnect();
this._scalar = this.context.createGain();
this.chain(signal, this._scalar, this.output);
//set it ot the sync ratio
2014-08-23 18:22:18 +00:00
this._scalar.gain.value = this._syncRatio;
2014-06-14 23:11:37 +00:00
};
/**
* unbind the signal control
*
* will leave the signal value as it was without the influence of the control signal
*/
Tone.Signal.prototype.unsync = function(){
//make a new scalar so that it's disconnected from the control signal
//get the current gain
var currentGain = this.getValue();
2014-08-23 18:22:18 +00:00
this._scalar.disconnect();
this._scalar = this.context.createGain();
this._scalar.gain.value = currentGain / this._syncRatio;
2014-06-14 23:11:37 +00:00
this._syncRatio = 1;
//reconnect things up
2014-08-23 18:22:18 +00:00
this.chain(constant, this._scalar, this.output);
2014-06-14 23:11:37 +00:00
};
/**
* internal dispose method to tear down the node
*/
Tone.Signal.prototype.dispose = function(){
Tone.prototype.dispose.call(this);
2014-08-23 18:22:18 +00:00
this._scalar.disconnect();
this._scalar = null;
};
/**
* Signals can connect to other Signals
*
* @override
* @param {AudioParam|AudioNode|Tone.Signal|Tone} node
2014-07-22 16:48:04 +00:00
* @param {number=} outputNumber
* @param {number=} inputNumber
*/
2014-07-22 16:48:04 +00:00
Tone.Signal.prototype.connect = function(node, outputNumber, inputNumber){
//zero it out so that the signal can have full control
if (node instanceof Tone.Signal){
node.setValue(0);
} else if (node instanceof AudioParam){
node.value = 0;
}
2014-08-24 19:47:59 +00:00
Tone.prototype.connect.call(this, node, outputNumber, inputNumber);
};
//defines getter / setter for value
Object.defineProperty(Tone.Signal.prototype, "value", {
get : function(){
return this.getValue();
},
set : function(val){
this.setValue(val);
}
});
2014-07-30 17:56:32 +00:00
///////////////////////////////////////////////////////////////////////////
// STATIC
///////////////////////////////////////////////////////////////////////////
/**
* all signals share a common constant signal generator
*
* @static
* @private
* @type {OscillatorNode}
*/
var generator = null;
/**
* @static
* @private
* @type {WaveShaperNode}
*/
var constant = null;
/**
* initializer function
*/
Tone._initAudioContext(function(audioContext){
generator = audioContext.createOscillator();
constant = audioContext.createWaveShaper();
//generate the waveshaper table which outputs 1 for any input value
var len = 8;
var curve = new Float32Array(len);
for (var i = 0; i < len; i++){
//all inputs produce the output value
curve[i] = 1;
}
constant.curve = curve;
//connect it up
generator.connect(constant);
generator.start(0);
generator.noGC();
});
return Tone.Signal;
2014-06-14 23:11:37 +00:00
});