Tone.js/Tone/signal/Signal.js

270 lines
7.6 KiB
JavaScript
Raw Normal View History

define(["Tone/core/Tone", "Tone/signal/WaveShaper", "Tone/type/Type", "Tone/core/Param",
"Tone/shim/ConstantSourceNode", "Tone/core/Gain"], function(Tone){
"use strict";
/**
* @class A signal is an audio-rate value. Tone.Signal is a core component of the library.
* Unlike a number, Signals can be scheduled with sample-level accuracy. Tone.Signal
* has all of the methods available to native Web Audio
* [AudioParam](http://webaudio.github.io/web-audio-api/#the-audioparam-interface)
* as well as additional conveniences. Read more about working with signals
* [here](https://github.com/Tonejs/Tone.js/wiki/Signals).
*
* @constructor
2015-11-01 22:49:14 +00:00
* @extends {Tone.Param}
* @param {Number|AudioParam} [value] Initial value of the signal. If an AudioParam
* is passed in, that parameter will be wrapped
* and controlled by the Signal.
* @param {string} [units=Number] unit The units the signal is in.
* @example
* var signal = new Tone.Signal(10);
*/
Tone.Signal = function(){
var options = Tone.defaults(arguments, ["value", "units"], Tone.Signal);
Tone.Param.call(this, options);
/**
* When a signal is connected to another signal or audio param,
* this signal becomes a proxy for it
* @type {Array}
* @private
*/
this._proxies = [];
/**
* Indicates if the constant source was started or not
* @private
* @type {Boolean}
*/
this._sourceStarted = false;
/**
* The constant source node which generates the signal
* @type {ConstantSourceNode}
* @private
*/
this._constantSource = this.context.createConstantSource();
this._param = this._constantSource.offset;
this.value = options.value;
/**
* The node where the constant signal value is scaled.
* @type {GainNode}
* @private
*/
this.output = this._constantSource;
/**
* The node where the value is set.
* @type {Tone.Param}
* @private
*/
this.input = this._param = this.output.offset;
};
Tone.extend(Tone.Signal, Tone.Param);
/**
* The default values
* @type {Object}
* @static
* @const
*/
Tone.Signal.defaults = {
"value" : 0,
"units" : Tone.Type.Default,
"convert" : true,
};
/**
* When signals connect to other signals or AudioParams,
* they take over the output value of that signal or AudioParam.
* For all other nodes, the behavior is the same as a default <code>connect</code>.
*
* @override
* @param {AudioParam|AudioNode|Tone.Signal|Tone} node
* @param {number} [outputNumber=0] The output number to connect from.
* @param {number} [inputNumber=0] The input number to connect to.
* @returns {Tone.Signal} this
* @method
*/
Tone.Signal.prototype.connect = function(node){
//this is an optimization where this node will forward automations
//to connected nodes without any signal if possible.
2018-06-06 01:58:00 +00:00
if (this._canProxy(node) && this.proxy){
this._proxies.push(node);
node.overridden = true;
2018-06-06 01:58:00 +00:00
//apply all the automations initially
this._applyAutomations(node);
} else {
Tone.SignalBase.prototype.connect.apply(this, arguments);
2018-06-06 01:58:00 +00:00
this._connectProxies();
}
return this;
};
/**
2018-06-06 01:58:00 +00:00
* Takes a node as an argument and returns if the node can be proxied
* or if the signal should be started when it's connected.
* @param {*} node The node to test
* @return {Boolean}
* @private
*/
2018-06-06 01:58:00 +00:00
Tone.Signal.prototype._canProxy = function(node){
return node.constructor === Tone.Param ||
(node.constructor === Tone.Signal && node.proxy) ||
node instanceof AudioParam;
};
/**
* Discard the optimization and connect all of the proxies
* @private
*/
Tone.Signal.prototype._connectProxies = function(){
if (!this._sourceStarted){
this._sourceStarted = true;
this._constantSource.start(0);
}
this._proxies.forEach(function(proxy){
Tone.SignalBase.prototype.connect.call(this, proxy);
2018-06-06 01:58:00 +00:00
if (proxy.proxy){
proxy._connectProxies();
}
}.bind(this));
};
/**
* Invoked when a node is connected to this
* @param {AudioNode} from
* @private
*/
2018-02-08 16:02:30 +00:00
Tone.Signal.prototype._onConnect = function(from){
2018-06-06 01:58:00 +00:00
if (!this._canProxy(from)){
//connect all the proxies
this._connectProxies();
}
};
/**
* Apply all the current automations to the given parameter
* @param {AudioParam} param
* @private
*/
Tone.Signal.prototype._applyAutomations = function(param){
var now = this.context.currentTime;
param.cancelScheduledValues(now);
var currentVal = this.getValueAtTime(now);
param.setValueAtTime(currentVal, now);
this._events.forEachFrom(now, function(event){
param[event.type](event.value, event.time, event.constant);
});
};
/**
* Disconnect from the given node or all nodes if no param is given.
* @param {AudioNode|AudioParam} node
* @return {Tone.Signal} this
*/
Tone.Signal.prototype.disconnect = function(node){
if (this._proxies.includes(node)){
var index = this._proxies.indexOf(node);
this._proxies.splice(index, 1);
} else if (!node){
//no argument, disconnect everything
this._proxies = [];
}
if (this._sourceStarted){
Tone.SignalBase.prototype.disconnect.apply(this, arguments);
}
return this;
};
/**
2018-06-06 01:58:00 +00:00
* True if the source is not actually outputting any signal, but merely
* acting as a proxy signal which forwards scheduled events to all of
* the connected signals. Can force the signal to unproxy by
* setting `sig.proxy = false` though once unproxied, cannot force it back.
* @type {Boolean}
* @memberOf Tone.Signal#
* @name proxy
*/
2018-06-06 01:58:00 +00:00
Object.defineProperty(Tone.Signal.prototype, "proxy", {
get : function(){
return !this._sourceStarted;
},
set : function(proxy){
if (!proxy && !this._sourceStarted){
this._connectProxies();
}
2017-12-30 02:15:30 +00:00
}
2018-06-06 01:58:00 +00:00
});
2017-12-30 02:15:30 +00:00
/**
* Return the current signal value at the given time.
* @param {Time} time When to get the signal value
* @return {Number}
*/
Tone.Signal.prototype.getValueAtTime = function(time){
if (this._param.getValueAtTime){
return this._param.getValueAtTime(time);
} else {
return Tone.Param.prototype.getValueAtTime.call(this, time);
}
};
2018-06-06 01:58:00 +00:00
/**
* Wrap the native scheduling methods with a method
* that also applies the automation to all the proxied connections.
* @param {Function} method The original method to wrap
* @param {Number} argCount How many arguments the method takes
* @private
*/
function _wrapMethod(method, argCount){
var previousMethod = Tone.Signal.prototype[method];
Tone.Signal.prototype[method] = function(){
var args = arguments;
//call the original method
previousMethod.apply(this, arguments);
//if it's proxied, call the same method on all of the proxied methods
if (this.proxy){
if (argCount === 2){
args[0] = this._fromUnits(args[0]);
args[1] = this.toSeconds(args[1]);
} else {
args[0] = this.toSeconds(args[0]);
}
//apply it to the proxies
this._proxies.forEach(function(signal){
signal[method].apply(signal, args);
});
2018-06-06 01:58:00 +00:00
}
};
}
//wrap all of the automation methods
["setValueAtTime", "linearRampToValueAtTime", "exponentialRampToValueAtTime", "setTargetAtTime"]
.forEach(function(method){
_wrapMethod(method, 2);
});
["cancelScheduledValues", "cancelAndHoldAtTime"]
.forEach(function(method){
2018-06-06 01:58:00 +00:00
_wrapMethod(method, 1);
});
/**
* dispose and disconnect
* @returns {Tone.Signal} this
*/
Tone.Signal.prototype.dispose = function(){
Tone.Param.prototype.dispose.call(this);
this._constantSource.disconnect();
this._constantSource = null;
this._proxies = null;
return this;
};
return Tone.Signal;
});