2018-08-27 02:29:17 +00:00
|
|
|
define(["../core/Tone", "../component/Volume", "../core/AudioNode"], function(Tone){
|
2016-12-17 21:26:27 +00:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @class Tone.UserMedia uses MediaDevices.getUserMedia to open up
|
2017-08-27 21:50:31 +00:00
|
|
|
* and external microphone or audio input. Check
|
2016-12-17 21:26:27 +00:00
|
|
|
* [MediaDevices API Support](https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getUserMedia)
|
|
|
|
* to see which browsers are supported. Access to an external input
|
|
|
|
* is limited to secure (HTTPS) connections.
|
2017-08-27 21:50:31 +00:00
|
|
|
*
|
2016-12-17 21:26:27 +00:00
|
|
|
* @constructor
|
2017-08-27 21:50:31 +00:00
|
|
|
* @extends {Tone.AudioNode}
|
2016-12-17 21:26:27 +00:00
|
|
|
* @param {Decibels=} volume The level of the input
|
|
|
|
* @example
|
|
|
|
* //list the inputs and open the third one
|
|
|
|
* var motu = new Tone.UserMedia();
|
2017-08-27 21:50:31 +00:00
|
|
|
*
|
2016-12-17 21:26:27 +00:00
|
|
|
* //opening the input asks the user to activate their mic
|
2017-01-08 22:20:07 +00:00
|
|
|
* motu.open().then(function(){
|
2018-03-05 15:31:32 +00:00
|
|
|
* //promise resolves when input is available
|
2016-12-17 21:26:27 +00:00
|
|
|
* });
|
|
|
|
*/
|
|
|
|
|
|
|
|
Tone.UserMedia = function(){
|
|
|
|
|
2017-04-26 04:00:01 +00:00
|
|
|
var options = Tone.defaults(arguments, ["volume"], Tone.UserMedia);
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.AudioNode.call(this);
|
2016-12-17 21:26:27 +00:00
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* The MediaStreamNode
|
2016-12-17 21:26:27 +00:00
|
|
|
* @type {MediaStreamAudioSourceNode}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._mediaStream = null;
|
2017-08-27 21:50:31 +00:00
|
|
|
|
2016-12-17 21:26:27 +00:00
|
|
|
/**
|
|
|
|
* The media stream created by getUserMedia.
|
|
|
|
* @type {LocalMediaStream}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._stream = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The open device
|
|
|
|
* @type {MediaDeviceInfo}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._device = null;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The output volume node
|
|
|
|
* @type {Tone.Volume}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._volume = this.output = new Tone.Volume(options.volume);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The volume of the output in decibels.
|
|
|
|
* @type {Decibels}
|
|
|
|
* @signal
|
|
|
|
* @example
|
|
|
|
* input.volume.value = -6;
|
|
|
|
*/
|
|
|
|
this.volume = this._volume.volume;
|
|
|
|
this._readOnly("volume");
|
|
|
|
|
|
|
|
this.mute = options.mute;
|
|
|
|
};
|
|
|
|
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.extend(Tone.UserMedia, Tone.AudioNode);
|
2016-12-17 21:26:27 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* the default parameters
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Tone.UserMedia.defaults = {
|
|
|
|
"volume" : 0,
|
|
|
|
"mute" : false
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Open the media stream. If a string is passed in, it is assumed
|
|
|
|
* to be the label or id of the stream, if a number is passed in,
|
|
|
|
* it is the input number of the stream.
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {String|Number} [labelOrId="default"] The label or id of the audio input media device.
|
2016-12-17 21:26:27 +00:00
|
|
|
* With no argument, the default stream is opened.
|
|
|
|
* @return {Promise} The promise is resolved when the stream is open.
|
|
|
|
*/
|
|
|
|
Tone.UserMedia.prototype.open = function(labelOrId){
|
2017-05-03 00:58:14 +00:00
|
|
|
return Tone.UserMedia.enumerateDevices().then(function(devices){
|
2016-12-17 21:26:27 +00:00
|
|
|
var device;
|
2017-04-26 04:24:19 +00:00
|
|
|
if (Tone.isNumber(labelOrId)){
|
2016-12-17 21:26:27 +00:00
|
|
|
device = devices[labelOrId];
|
|
|
|
} else {
|
|
|
|
device = devices.find(function(device){
|
|
|
|
return device.label === labelOrId || device.deviceId === labelOrId;
|
|
|
|
});
|
2017-05-03 00:58:14 +00:00
|
|
|
//didn't find a matching device
|
2017-10-26 04:50:22 +00:00
|
|
|
if (!device && devices.length > 0){
|
2017-10-26 05:07:53 +00:00
|
|
|
device = devices[0];
|
2018-03-05 16:32:08 +00:00
|
|
|
} else if (!device && Tone.isDefined(labelOrId)){
|
2017-05-03 00:58:14 +00:00
|
|
|
throw new Error("Tone.UserMedia: no matching device: "+labelOrId);
|
2016-12-17 21:26:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
this._device = device;
|
|
|
|
//do getUserMedia
|
|
|
|
var constraints = {
|
|
|
|
audio : {
|
2017-10-25 21:57:52 +00:00
|
|
|
"echoCancellation" : false,
|
2018-06-13 19:57:05 +00:00
|
|
|
"sampleRate" : this.context.sampleRate,
|
|
|
|
"noiseSuppression" : false,
|
|
|
|
"mozNoiseSuppression" : false,
|
2016-12-17 21:26:27 +00:00
|
|
|
}
|
|
|
|
};
|
2017-11-29 20:38:25 +00:00
|
|
|
if (device){
|
|
|
|
constraints.audio.deviceId = device.deviceId;
|
|
|
|
}
|
2016-12-17 21:26:27 +00:00
|
|
|
return navigator.mediaDevices.getUserMedia(constraints).then(function(stream){
|
|
|
|
//start a new source only 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.output);
|
2017-08-27 21:50:31 +00:00
|
|
|
}
|
2016-12-17 21:26:27 +00:00
|
|
|
return this;
|
|
|
|
}.bind(this));
|
|
|
|
}.bind(this));
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Close the media stream
|
|
|
|
* @return {Tone.UserMedia} this
|
|
|
|
*/
|
|
|
|
Tone.UserMedia.prototype.close = function(){
|
2018-01-02 15:37:27 +00:00
|
|
|
if (this._stream){
|
2016-12-17 21:35:50 +00:00
|
|
|
this._stream.getAudioTracks().forEach(function(track){
|
2016-12-17 21:26:27 +00:00
|
|
|
track.stop();
|
|
|
|
});
|
|
|
|
this._stream = null;
|
|
|
|
//remove the old media stream
|
|
|
|
this._mediaStream.disconnect();
|
|
|
|
this._mediaStream = null;
|
|
|
|
}
|
|
|
|
this._device = null;
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a promise which resolves with the list of audio input devices available.
|
|
|
|
* @return {Promise} The promise that is resolved with the devices
|
2017-05-03 00:58:14 +00:00
|
|
|
* @static
|
2016-12-17 21:26:27 +00:00
|
|
|
* @example
|
2017-05-03 00:58:14 +00:00
|
|
|
* Tone.UserMedia.enumerateDevices().then(function(devices){
|
2016-12-17 21:26:27 +00:00
|
|
|
* console.log(devices)
|
|
|
|
* })
|
|
|
|
*/
|
2017-05-03 00:58:14 +00:00
|
|
|
Tone.UserMedia.enumerateDevices = function(){
|
2016-12-17 21:26:27 +00:00
|
|
|
return navigator.mediaDevices.enumerateDevices().then(function(devices){
|
|
|
|
return devices.filter(function(device){
|
|
|
|
return device.kind === "audioinput";
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the playback state of the source, "started" when the microphone is open
|
|
|
|
* and "stopped" when the mic is closed.
|
|
|
|
* @type {Tone.State}
|
|
|
|
* @readOnly
|
|
|
|
* @memberOf Tone.UserMedia#
|
|
|
|
* @name state
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.UserMedia.prototype, "state", {
|
|
|
|
get : function(){
|
|
|
|
return this._stream && this._stream.active ? Tone.State.Started : Tone.State.Stopped;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Returns an identifier for the represented device that is
|
|
|
|
* persisted across sessions. It is un-guessable by other applications and
|
|
|
|
* unique to the origin of the calling application. It is reset when the
|
|
|
|
* user clears cookies (for Private Browsing, a different identifier is
|
|
|
|
* used that is not persisted across sessions). Returns undefined when the
|
2016-12-17 21:26:27 +00:00
|
|
|
* device is not open.
|
|
|
|
* @type {String}
|
|
|
|
* @readOnly
|
|
|
|
* @memberOf Tone.UserMedia#
|
|
|
|
* @name deviceId
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.UserMedia.prototype, "deviceId", {
|
|
|
|
get : function(){
|
|
|
|
if (this._device){
|
|
|
|
return this._device.deviceId;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Returns a group identifier. Two devices have the
|
2016-12-17 21:26:27 +00:00
|
|
|
* same group identifier if they belong to the same physical device.
|
|
|
|
* Returns undefined when the device is not open.
|
|
|
|
* @type {String}
|
|
|
|
* @readOnly
|
|
|
|
* @memberOf Tone.UserMedia#
|
|
|
|
* @name groupId
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.UserMedia.prototype, "groupId", {
|
|
|
|
get : function(){
|
|
|
|
if (this._device){
|
|
|
|
return this._device.groupId;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Returns a label describing this device (for example "Built-in Microphone").
|
2016-12-17 21:26:27 +00:00
|
|
|
* Returns undefined when the device is not open or label is not available
|
|
|
|
* because of permissions.
|
|
|
|
* @type {String}
|
|
|
|
* @readOnly
|
|
|
|
* @memberOf Tone.UserMedia#
|
|
|
|
* @name groupId
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.UserMedia.prototype, "label", {
|
|
|
|
get : function(){
|
|
|
|
if (this._device){
|
|
|
|
return this._device.label;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Mute the output.
|
2016-12-17 21:26:27 +00:00
|
|
|
* @memberOf Tone.UserMedia#
|
|
|
|
* @type {boolean}
|
|
|
|
* @name mute
|
|
|
|
* @example
|
|
|
|
* //mute the output
|
2017-01-08 22:20:55 +00:00
|
|
|
* userMedia.mute = true;
|
2016-12-17 21:26:27 +00:00
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.UserMedia.prototype, "mute", {
|
|
|
|
get : function(){
|
|
|
|
return this._volume.mute;
|
2017-08-27 21:50:31 +00:00
|
|
|
},
|
2016-12-17 21:26:27 +00:00
|
|
|
set : function(mute){
|
|
|
|
this._volume.mute = mute;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up.
|
|
|
|
* @return {Tone.UserMedia} this
|
|
|
|
*/
|
|
|
|
Tone.UserMedia.prototype.dispose = function(){
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.AudioNode.prototype.dispose.call(this);
|
2016-12-17 21:26:27 +00:00
|
|
|
this.close();
|
|
|
|
this._writable("volume");
|
|
|
|
this._volume.dispose();
|
|
|
|
this._volume = null;
|
|
|
|
this.volume = null;
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If getUserMedia is supported by the browser.
|
|
|
|
* @type {Boolean}
|
|
|
|
* @memberOf Tone.UserMedia#
|
|
|
|
* @name supported
|
|
|
|
* @static
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.UserMedia, "supported", {
|
|
|
|
get : function(){
|
2018-03-05 16:32:08 +00:00
|
|
|
return Tone.isDefined(navigator.mediaDevices) && Tone.isFunction(navigator.mediaDevices.getUserMedia);
|
2016-12-17 21:26:27 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return Tone.UserMedia;
|
2017-08-27 21:50:31 +00:00
|
|
|
});
|