2015-09-19 00:18:44 +00:00
|
|
|
define(["Tone/core/Tone", "Tone/source/Source", "Tone/core/Gain"], function(Tone){
|
2015-07-14 23:26:16 +00:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
2015-09-19 00:18:44 +00:00
|
|
|
//polyfill for getUserMedia
|
|
|
|
navigator.getUserMedia = navigator.getUserMedia || navigator.webkitGetUserMedia ||
|
|
|
|
navigator.mozGetUserMedia || navigator.msGetUserMedia;
|
|
|
|
|
2015-07-14 23:26:16 +00:00
|
|
|
/**
|
|
|
|
* @class Tone.ExternalInput is a WebRTC Audio Input. Check
|
|
|
|
* [Media Stream API Support](https://developer.mozilla.org/en-US/docs/Web/API/MediaStream_API)
|
|
|
|
* to see which browsers are supported. As of
|
|
|
|
* writing this, Chrome, Firefox, and Opera
|
|
|
|
* support Media Stream. Chrome allows enumeration
|
|
|
|
* of the sources, and access to device name over a
|
|
|
|
* secure (HTTPS) connection. See [https://simpl.info](https://simpl.info/getusermedia/sources/index.html)
|
|
|
|
* vs [http://simple.info](https://simpl.info/getusermedia/sources/index.html)
|
|
|
|
* on a Chrome browser for the difference.
|
|
|
|
*
|
|
|
|
* @constructor
|
|
|
|
* @extends {Tone.Source}
|
|
|
|
* @param {number} [inputNum=0] If multiple inputs are present, select the input number. Chrome only.
|
|
|
|
* @example
|
|
|
|
* var motu = new Tone.ExternalInput(3);
|
|
|
|
*
|
2015-08-26 18:29:49 +00:00
|
|
|
* motu.open(function(){
|
|
|
|
* motu.start(10);
|
|
|
|
* });
|
2015-07-14 23:26:16 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
Tone.ExternalInput = function(){
|
|
|
|
|
2015-08-20 17:36:06 +00:00
|
|
|
var options = this.optionsObject(arguments, ["inputNum"], Tone.ExternalInput.defaults);
|
2015-07-14 23:26:16 +00:00
|
|
|
Tone.Source.call(this, options);
|
|
|
|
|
|
|
|
/**
|
2015-09-19 00:18:44 +00:00
|
|
|
* The MediaStreamNode
|
2015-07-14 23:26:16 +00:00
|
|
|
* @type {MediaStreamAudioSourceNode}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._mediaStream = null;
|
|
|
|
|
|
|
|
/**
|
2015-09-19 00:18:44 +00:00
|
|
|
* The media stream created by getUserMedia.
|
2015-07-14 23:26:16 +00:00
|
|
|
* @type {LocalMediaStream}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._stream = null;
|
|
|
|
|
|
|
|
/**
|
2015-09-19 00:18:44 +00:00
|
|
|
* The constraints argument for getUserMedia
|
2015-07-14 23:26:16 +00:00
|
|
|
* @type {Object}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._constraints = {"audio" : true};
|
|
|
|
|
|
|
|
/**
|
2015-08-28 18:53:05 +00:00
|
|
|
* The input source position in Tone.ExternalInput.sources.
|
|
|
|
* Set before ExternalInput.open().
|
2015-09-19 00:18:44 +00:00
|
|
|
* @type {Number}
|
|
|
|
* @private
|
2015-07-14 23:26:16 +00:00
|
|
|
*/
|
2015-09-19 00:18:44 +00:00
|
|
|
this._inputNum = options.inputNum;
|
2015-07-14 23:26:16 +00:00
|
|
|
|
2015-08-28 18:53:05 +00:00
|
|
|
/**
|
2015-09-19 00:18:44 +00:00
|
|
|
* Gates the input signal for start/stop.
|
|
|
|
* Initially closed.
|
2015-08-28 18:53:05 +00:00
|
|
|
* @type {GainNode}
|
|
|
|
* @private
|
|
|
|
*/
|
2015-09-19 00:18:44 +00:00
|
|
|
this._gate = new Tone.Gain(0).connect(this.output);
|
2015-07-14 23:26:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Tone.extend(Tone.ExternalInput, Tone.Source);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the default parameters
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Tone.ExternalInput.defaults = {
|
2015-08-20 17:36:06 +00:00
|
|
|
"inputNum" : 0
|
2015-07-14 23:26:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-08-26 18:29:49 +00:00
|
|
|
* wrapper for getUserMedia function
|
2015-09-19 00:18:44 +00:00
|
|
|
* @param {function} callback
|
2015-07-14 23:26:16 +00:00
|
|
|
* @private
|
|
|
|
*/
|
2015-08-26 18:29:49 +00:00
|
|
|
Tone.ExternalInput.prototype._getUserMedia = function(callback){
|
2015-09-19 00:18:44 +00:00
|
|
|
if (!Tone.ExternalInput.canGetUserMedia){
|
|
|
|
throw new Error("browser does not support 'getUserMedia'");
|
|
|
|
}
|
|
|
|
if (Tone.ExternalInput.sources[this._inputNum]){
|
|
|
|
this._constraints = {
|
|
|
|
audio : {
|
|
|
|
optional : [{sourceId: Tone.ExternalInput.sources[this._inputNum].id}]
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
2015-08-26 18:29:49 +00:00
|
|
|
navigator.getUserMedia(this._constraints, function(stream){
|
|
|
|
this._onStream(stream);
|
|
|
|
callback();
|
|
|
|
}.bind(this), function(err){
|
|
|
|
callback(err);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2015-09-19 00:18:44 +00:00
|
|
|
/**
|
|
|
|
* called when the stream is successfully setup
|
|
|
|
* @param {LocalMediaStream} stream
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.ExternalInput.prototype._onStream = function(stream){
|
|
|
|
if (!this.isFunction(this.context.createMediaStreamSource)){
|
|
|
|
throw new Error("browser does not support the 'MediaStreamSourceNode'");
|
|
|
|
}
|
|
|
|
//can only start a new source if the previous one is closed
|
|
|
|
if (!this._stream){
|
|
|
|
this._stream = stream;
|
|
|
|
//Wrap a MediaStreamSourceNode around the live input stream.
|
|
|
|
this._mediaStream = this.context.createMediaStreamSource(stream);
|
|
|
|
//Connect the MediaStreamSourceNode to a gate gain node
|
|
|
|
this._mediaStream.connect(this._gate);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2015-08-26 18:29:49 +00:00
|
|
|
/**
|
|
|
|
* Open the media stream
|
2015-09-19 00:18:44 +00:00
|
|
|
* @param {function=} callback The callback function to
|
2015-08-26 18:29:49 +00:00
|
|
|
* execute when the stream is open
|
|
|
|
* @return {Tone.ExternalInput} this
|
|
|
|
*/
|
|
|
|
Tone.ExternalInput.prototype.open = function(callback){
|
2015-09-19 00:18:44 +00:00
|
|
|
callback = this.defaultArg(callback, Tone.noOp);
|
2015-08-28 18:53:05 +00:00
|
|
|
Tone.ExternalInput.getSources(function(){
|
2015-08-26 18:29:49 +00:00
|
|
|
this._getUserMedia(callback);
|
2015-08-28 18:53:05 +00:00
|
|
|
}.bind(this));
|
|
|
|
return this;
|
2015-07-14 23:26:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-08-26 18:29:49 +00:00
|
|
|
* Close the media stream
|
|
|
|
* @return {Tone.ExternalInput} this
|
|
|
|
*/
|
|
|
|
Tone.ExternalInput.prototype.close = function(){
|
|
|
|
if(this._stream){
|
|
|
|
this._stream.stop();
|
2015-09-19 00:18:44 +00:00
|
|
|
this._stream = null;
|
2015-08-26 18:29:49 +00:00
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start the stream
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.ExternalInput.prototype._start = function(time){
|
|
|
|
time = this.toSeconds(time);
|
|
|
|
this._gate.gain.setValueAtTime(1, time);
|
2015-09-19 00:18:44 +00:00
|
|
|
return this;
|
2015-08-26 18:29:49 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-09-19 00:18:44 +00:00
|
|
|
* Stops the stream.
|
2015-08-26 18:29:49 +00:00
|
|
|
* @private
|
2015-07-14 23:26:16 +00:00
|
|
|
*/
|
2015-08-26 18:29:49 +00:00
|
|
|
Tone.ExternalInput.prototype._stop = function(time){
|
2015-09-19 00:18:44 +00:00
|
|
|
time = this.toSeconds(time);
|
2015-08-26 18:29:49 +00:00
|
|
|
this._gate.gain.setValueAtTime(0, time);
|
2015-07-14 23:26:16 +00:00
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up.
|
2015-08-26 18:29:49 +00:00
|
|
|
* @return {Tone.ExternalInput} this
|
2015-07-14 23:26:16 +00:00
|
|
|
*/
|
|
|
|
Tone.ExternalInput.prototype.dispose = function(){
|
|
|
|
Tone.Source.prototype.dispose.call(this);
|
2015-09-19 00:18:44 +00:00
|
|
|
this.close();
|
2015-07-14 23:26:16 +00:00
|
|
|
if (this._mediaStream){
|
|
|
|
this._mediaStream.disconnect();
|
|
|
|
this._mediaStream = null;
|
|
|
|
}
|
|
|
|
this._constraints = null;
|
2015-09-19 00:18:44 +00:00
|
|
|
this._gate.dispose();
|
|
|
|
this._gate = null;
|
2015-07-14 23:26:16 +00:00
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// STATIC METHODS
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The array of available sources, different depending on whether connection is secure
|
|
|
|
* @type {Array}
|
2015-09-19 00:18:44 +00:00
|
|
|
* @static
|
2015-07-14 23:26:16 +00:00
|
|
|
*/
|
2015-08-28 18:53:05 +00:00
|
|
|
Tone.ExternalInput.sources = [];
|
2015-07-14 23:26:16 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* indicates whether browser supports MediaStreamTrack.getSources (i.e. Chrome vs Firefox)
|
|
|
|
* @type {Boolean}
|
|
|
|
* @private
|
|
|
|
*/
|
2015-09-19 00:18:44 +00:00
|
|
|
Tone.ExternalInput._canGetSources = !Tone.prototype.isUndef(window.MediaStreamTrack) && Tone.prototype.isFunction(MediaStreamTrack.getSources);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Indicates if the browser supports 'getUserMedia'
|
|
|
|
* @type {Boolean}
|
|
|
|
* @static
|
|
|
|
*/
|
|
|
|
Tone.ExternalInput.canGetUserMedia = Tone.prototype.isFunction(navigator.getUserMedia);
|
2015-07-14 23:26:16 +00:00
|
|
|
|
|
|
|
/**
|
2015-09-19 00:18:44 +00:00
|
|
|
* Populates the source list. Invokes the callback with an array of
|
|
|
|
* possible audio sources.
|
2015-08-26 18:29:49 +00:00
|
|
|
* @param {function=} callback Callback to be executed after populating list
|
2015-09-19 00:18:44 +00:00
|
|
|
* @return {Tone.ExternalInput} this
|
|
|
|
* @static
|
2015-08-26 18:29:49 +00:00
|
|
|
* @example
|
2015-09-19 00:18:44 +00:00
|
|
|
* var soundflower = new Tone.ExternalInput();
|
|
|
|
* Tone.ExternalInput.getSources(selectSoundflower);
|
2015-08-26 18:29:49 +00:00
|
|
|
*
|
|
|
|
* function selectSoundflower(sources){
|
2015-09-19 00:18:44 +00:00
|
|
|
* for(var i = 0; i < sources.length; i++){
|
|
|
|
* if(sources[i].label === "soundflower"){
|
|
|
|
* soundflower.inputNum = i;
|
|
|
|
* soundflower.open(function(){
|
|
|
|
* soundflower.start();
|
|
|
|
* });
|
|
|
|
* break;
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* };
|
2015-07-14 23:26:16 +00:00
|
|
|
*/
|
2015-08-26 18:29:49 +00:00
|
|
|
Tone.ExternalInput.getSources = function(callback){
|
2015-08-28 18:53:05 +00:00
|
|
|
if(Tone.ExternalInput.sources.length === 0 && Tone.ExternalInput._canGetSources){
|
2015-08-26 18:29:49 +00:00
|
|
|
MediaStreamTrack.getSources(function (media_sources){
|
2015-09-19 00:18:44 +00:00
|
|
|
for(var i = 0; i < media_sources.length; i++) {
|
2015-08-26 18:29:49 +00:00
|
|
|
if(media_sources[i].kind === "audio"){
|
2015-08-28 18:53:05 +00:00
|
|
|
Tone.ExternalInput.sources[i] = media_sources[i];
|
2015-07-14 23:26:16 +00:00
|
|
|
}
|
2015-08-26 18:29:49 +00:00
|
|
|
}
|
2015-08-28 18:53:05 +00:00
|
|
|
callback(Tone.ExternalInput.sources);
|
2015-08-26 18:29:49 +00:00
|
|
|
});
|
2015-07-14 23:26:16 +00:00
|
|
|
} else {
|
2015-08-28 18:53:05 +00:00
|
|
|
callback(Tone.ExternalInput.sources);
|
2015-07-14 23:26:16 +00:00
|
|
|
}
|
2015-08-28 18:53:05 +00:00
|
|
|
return this;
|
2015-07-14 23:26:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.ExternalInput;
|
|
|
|
});
|