2018-08-27 02:29:17 +00:00
|
|
|
define(["../core/Tone", "../core/Context"], function(Tone){
|
2017-08-27 21:18:19 +00:00
|
|
|
|
|
|
|
/**
|
2017-09-09 23:10:02 +00:00
|
|
|
* @class Tone.AudioNode is the base class for classes which process audio.
|
2017-08-27 21:18:19 +00:00
|
|
|
* AudioNodes have inputs and outputs.
|
|
|
|
* @param {AudioContext=} context The audio context to use with the class
|
|
|
|
* @extends {Tone}
|
|
|
|
*/
|
|
|
|
Tone.AudioNode = function(){
|
|
|
|
Tone.call(this);
|
|
|
|
|
|
|
|
//use the default context if one is not passed in
|
|
|
|
var options = Tone.defaults(arguments, ["context"], {
|
|
|
|
"context" : Tone.context
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The AudioContext of this instance
|
|
|
|
* @private
|
|
|
|
* @type {AudioContext}
|
|
|
|
*/
|
|
|
|
this._context = options.context;
|
|
|
|
};
|
|
|
|
|
|
|
|
Tone.extend(Tone.AudioNode);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the audio context belonging to this instance.
|
2017-08-30 16:52:33 +00:00
|
|
|
* @type {Tone.Context}
|
2017-08-27 21:18:19 +00:00
|
|
|
* @memberOf Tone.AudioNode#
|
|
|
|
* @name context
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.AudioNode.prototype, "context", {
|
|
|
|
get : function(){
|
|
|
|
return this._context;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create input and outputs for this object.
|
|
|
|
* @param {Number} [input=0] The number of inputs
|
|
|
|
* @param {Number} [outputs=0] The number of outputs
|
2017-08-28 15:08:47 +00:00
|
|
|
* @return {Tone.AudioNode} this
|
2017-08-27 21:18:19 +00:00
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.AudioNode.prototype.createInsOuts = function(inputs, outputs){
|
|
|
|
|
|
|
|
if (inputs === 1){
|
|
|
|
this.input = this.context.createGain();
|
|
|
|
} else if (inputs > 1){
|
|
|
|
this.input = new Array(inputs);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (outputs === 1){
|
|
|
|
this.output = this.context.createGain();
|
|
|
|
} else if (outputs > 1){
|
|
|
|
this.output = new Array(outputs);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-05 21:51:04 +00:00
|
|
|
/**
|
|
|
|
* channelCount is the number of channels used when up-mixing and down-mixing
|
|
|
|
* connections to any inputs to the node. The default value is 2 except for
|
|
|
|
* specific nodes where its value is specially determined.
|
|
|
|
*
|
|
|
|
* @memberof Tone.AudioNode#
|
|
|
|
* @type {Number}
|
|
|
|
* @name channelCount
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.AudioNode.prototype, "channelCount", {
|
|
|
|
get : function(){
|
|
|
|
return this.output.channelCount;
|
|
|
|
},
|
|
|
|
set : function(c){
|
|
|
|
return this.output.channelCount = c;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* channelCountMode determines how channels will be counted when up-mixing and
|
|
|
|
* down-mixing connections to any inputs to the node.
|
|
|
|
* The default value is "max". This attribute has no effect for nodes with no inputs.
|
|
|
|
* @memberof Tone.AudioNode#
|
|
|
|
* @type {String}
|
|
|
|
* @name channelCountMode
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.AudioNode.prototype, "channelCountMode", {
|
|
|
|
get : function(){
|
|
|
|
return this.output.channelCountMode;
|
|
|
|
},
|
|
|
|
set : function(m){
|
|
|
|
return this.output.channelCountMode = m;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* channelInterpretation determines how individual channels will be treated
|
|
|
|
* when up-mixing and down-mixing connections to any inputs to the node.
|
|
|
|
* The default value is "speakers".
|
|
|
|
* @memberof Tone.AudioNode#
|
|
|
|
* @type {String}
|
|
|
|
* @name channelInterpretation
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.AudioNode.prototype, "channelInterpretation", {
|
|
|
|
get : function(){
|
|
|
|
return this.output.channelInterpretation;
|
|
|
|
},
|
|
|
|
set : function(i){
|
|
|
|
return this.output.channelInterpretation = i;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-08-27 21:18:19 +00:00
|
|
|
/**
|
|
|
|
* The number of inputs feeding into the AudioNode.
|
|
|
|
* For source nodes, this will be 0.
|
|
|
|
* @type {Number}
|
|
|
|
* @name numberOfInputs
|
2018-02-05 21:51:04 +00:00
|
|
|
* @memberof Tone.AudioNode#
|
2017-08-27 21:18:19 +00:00
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.AudioNode.prototype, "numberOfInputs", {
|
|
|
|
get : function(){
|
|
|
|
if (this.input){
|
|
|
|
if (Tone.isArray(this.input)){
|
|
|
|
return this.input.length;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of outputs coming out of the AudioNode.
|
|
|
|
* @type {Number}
|
|
|
|
* @name numberOfOutputs
|
2018-02-05 21:51:04 +00:00
|
|
|
* @memberof Tone.AudioNode#
|
2017-08-27 21:18:19 +00:00
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.AudioNode.prototype, "numberOfOutputs", {
|
|
|
|
get : function(){
|
|
|
|
if (this.output){
|
|
|
|
if (Tone.isArray(this.output)){
|
|
|
|
return this.output.length;
|
|
|
|
} else {
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-10-21 23:02:46 +00:00
|
|
|
/**
|
2017-08-27 21:48:52 +00:00
|
|
|
* connect the output of a ToneNode to an AudioParam, AudioNode, or ToneNode
|
|
|
|
* @param {Tone | AudioParam | AudioNode} unit
|
|
|
|
* @param {number} [outputNum=0] optionally which output to connect from
|
|
|
|
* @param {number} [inputNum=0] optionally which input to connect to
|
2017-08-28 15:08:47 +00:00
|
|
|
* @returns {Tone.AudioNode} this
|
2017-08-27 21:48:52 +00:00
|
|
|
*/
|
|
|
|
Tone.AudioNode.prototype.connect = function(unit, outputNum, inputNum){
|
|
|
|
if (Tone.isArray(this.output)){
|
|
|
|
outputNum = Tone.defaultArg(outputNum, 0);
|
|
|
|
this.output[outputNum].connect(unit, 0, inputNum);
|
|
|
|
} else {
|
|
|
|
this.output.connect(unit, outputNum, inputNum);
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* disconnect the output
|
|
|
|
* @param {Number|AudioNode} output Either the output index to disconnect
|
|
|
|
* if the output is an array, or the
|
|
|
|
* node to disconnect from.
|
2017-08-28 15:08:47 +00:00
|
|
|
* @returns {Tone.AudioNode} this
|
2017-08-27 21:48:52 +00:00
|
|
|
*/
|
|
|
|
Tone.AudioNode.prototype.disconnect = function(destination, outputNum, inputNum){
|
|
|
|
if (Tone.isArray(this.output)){
|
|
|
|
if (Tone.isNumber(destination)){
|
|
|
|
this.output[destination].disconnect();
|
|
|
|
} else {
|
|
|
|
outputNum = Tone.defaultArg(outputNum, 0);
|
|
|
|
this.output[outputNum].disconnect(destination, 0, inputNum);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
this.output.disconnect.apply(this.output, arguments);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Connect the output of this node to the rest of the nodes in series.
|
|
|
|
* @example
|
|
|
|
* //connect a node to an effect, panVol and then to the master output
|
|
|
|
* node.chain(effect, panVol, Tone.Master);
|
|
|
|
* @param {...AudioParam|Tone|AudioNode} nodes
|
2017-08-28 15:08:47 +00:00
|
|
|
* @returns {Tone.AudioNode} this
|
2017-08-27 21:48:52 +00:00
|
|
|
*/
|
|
|
|
Tone.AudioNode.prototype.chain = function(){
|
|
|
|
var currentUnit = this;
|
|
|
|
for (var i = 0; i < arguments.length; i++){
|
|
|
|
var toUnit = arguments[i];
|
|
|
|
currentUnit.connect(toUnit);
|
|
|
|
currentUnit = toUnit;
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* connect the output of this node to the rest of the nodes in parallel.
|
|
|
|
* @param {...AudioParam|Tone|AudioNode} nodes
|
2017-08-28 15:08:47 +00:00
|
|
|
* @returns {Tone.AudioNode} this
|
2017-08-27 21:48:52 +00:00
|
|
|
*/
|
|
|
|
Tone.AudioNode.prototype.fan = function(){
|
|
|
|
for (var i = 0; i < arguments.length; i++){
|
|
|
|
this.connect(arguments[i]);
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
if (window.AudioNode){
|
|
|
|
//give native nodes chain and fan methods
|
|
|
|
AudioNode.prototype.chain = Tone.AudioNode.prototype.chain;
|
|
|
|
AudioNode.prototype.fan = Tone.AudioNode.prototype.fan;
|
|
|
|
}
|
|
|
|
|
2017-08-27 21:18:19 +00:00
|
|
|
/**
|
|
|
|
* Dispose and disconnect
|
|
|
|
* @return {Tone.AudioNode} this
|
|
|
|
*/
|
2018-03-05 17:25:33 +00:00
|
|
|
Tone.AudioNode.prototype.dispose = function(){
|
2018-03-05 16:32:08 +00:00
|
|
|
if (Tone.isDefined(this.input)){
|
2017-08-27 21:18:19 +00:00
|
|
|
if (this.input instanceof AudioNode){
|
|
|
|
this.input.disconnect();
|
|
|
|
}
|
|
|
|
this.input = null;
|
|
|
|
}
|
2018-03-05 16:32:08 +00:00
|
|
|
if (Tone.isDefined(this.output)){
|
2017-08-27 21:18:19 +00:00
|
|
|
if (this.output instanceof AudioNode){
|
|
|
|
this.output.disconnect();
|
|
|
|
}
|
|
|
|
this.output = null;
|
|
|
|
}
|
|
|
|
this._context = null;
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.AudioNode;
|
|
|
|
});
|