2019-09-20 22:51:17 +00:00
|
|
|
import { connect, OutputNode, ToneAudioNode, ToneAudioNodeOptions } from "../core/context/ToneAudioNode";
|
|
|
|
import { Decibels } from "../core/type/Units";
|
|
|
|
import { Volume } from "../component";
|
|
|
|
import { optionsFromArguments } from "../core/util/Defaults";
|
|
|
|
import { Param } from "../core/context/Param";
|
|
|
|
import { readOnly } from "../core/util/Interface";
|
2019-10-31 19:43:16 +00:00
|
|
|
import { isDefined, isNumber } from "../core/util/TypeCheck";
|
2019-09-20 22:51:17 +00:00
|
|
|
|
|
|
|
export interface UserMediaOptions extends ToneAudioNodeOptions {
|
|
|
|
volume: Decibels;
|
|
|
|
mute: boolean;
|
|
|
|
}
|
|
|
|
/**
|
|
|
|
* UserMedia uses MediaDevices.getUserMedia to open up and external microphone or audio input.
|
|
|
|
* Check [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.
|
|
|
|
* @example
|
2019-10-25 20:54:33 +00:00
|
|
|
* import { UserMedia } from "tone";
|
|
|
|
* const mic = new UserMedia();
|
|
|
|
* mic.open().then(() => {
|
|
|
|
* // promise resolves when input is available
|
2019-09-20 22:51:17 +00:00
|
|
|
* });
|
2019-09-20 22:51:45 +00:00
|
|
|
* @category Source
|
2019-09-20 22:51:17 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
export class UserMedia extends ToneAudioNode<UserMediaOptions> {
|
|
|
|
|
|
|
|
readonly name: string = "UserMedia";
|
|
|
|
|
|
|
|
readonly input: undefined;
|
|
|
|
readonly output: OutputNode;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The MediaStreamNode
|
|
|
|
*/
|
|
|
|
private _mediaStream?: MediaStreamAudioSourceNode;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The media stream created by getUserMedia.
|
|
|
|
*/
|
|
|
|
private _stream?: MediaStream;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The open device
|
|
|
|
*/
|
|
|
|
private _device?: MediaDeviceInfo;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The output volume node
|
|
|
|
*/
|
|
|
|
private _volume: Volume;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The volume of the output in decibels.
|
|
|
|
*/
|
2019-10-28 15:37:53 +00:00
|
|
|
readonly volume: Param<"decibels">;
|
2019-09-20 22:51:17 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param volume The level of the input in decibels
|
|
|
|
*/
|
|
|
|
constructor(volume?: Decibels);
|
|
|
|
constructor(options?: Partial<UserMediaOptions>);
|
|
|
|
constructor() {
|
|
|
|
|
|
|
|
super(optionsFromArguments(UserMedia.getDefaults(), arguments, ["volume"]));
|
|
|
|
const options = optionsFromArguments(UserMedia.getDefaults(), arguments, ["volume"]);
|
|
|
|
|
|
|
|
this._volume = this.output = new Volume({
|
|
|
|
context: this.context,
|
|
|
|
volume: options.volume,
|
|
|
|
});
|
|
|
|
this.volume = this._volume.volume;
|
|
|
|
readOnly(this, "volume");
|
|
|
|
this.mute = options.mute;
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): UserMediaOptions {
|
|
|
|
return Object.assign(ToneAudioNode.getDefaults(), {
|
|
|
|
mute: false,
|
|
|
|
volume: 0
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* 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.
|
|
|
|
* @param labelOrId The label or id of the audio input media device.
|
|
|
|
* With no argument, the default stream is opened.
|
|
|
|
* @return The promise is resolved when the stream is open.
|
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
async open(labelOrId?: string | number): Promise<void> {
|
|
|
|
this.assert(UserMedia.supported, "UserMedia is not supported");
|
|
|
|
// close the previous stream
|
|
|
|
if (this.state === "started") {
|
|
|
|
this.close();
|
|
|
|
}
|
|
|
|
const devices = await UserMedia.enumerateDevices();
|
|
|
|
if (isNumber(labelOrId)) {
|
|
|
|
this._device = devices[labelOrId];
|
|
|
|
} else {
|
|
|
|
this._device = devices.find((device) => {
|
|
|
|
return device.label === labelOrId || device.deviceId === labelOrId;
|
|
|
|
});
|
|
|
|
// didn't find a matching device
|
|
|
|
if (!this._device && devices.length > 0) {
|
|
|
|
this._device = devices[0];
|
|
|
|
}
|
2019-09-29 14:35:46 +00:00
|
|
|
this.assert(isDefined(this._device), `No matching device ${labelOrId}`);
|
2019-09-20 22:51:17 +00:00
|
|
|
}
|
|
|
|
// do getUserMedia
|
|
|
|
let constraints = {
|
|
|
|
audio: {
|
|
|
|
echoCancellation: false,
|
|
|
|
sampleRate: this.context.sampleRate,
|
|
|
|
noiseSuppression: false,
|
|
|
|
mozNoiseSuppression: false,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
if (this._device) {
|
|
|
|
// @ts-ignore
|
|
|
|
constraints.audio.deviceId = this._device.deviceId;
|
|
|
|
}
|
|
|
|
const stream = await navigator.mediaDevices.getUserMedia(constraints);
|
|
|
|
// 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.
|
|
|
|
const mediaStreamNode = this.context.createMediaStreamSource(stream);
|
|
|
|
// Connect the MediaStreamSourceNode to a gate gain node
|
|
|
|
connect(mediaStreamNode, this.output);
|
|
|
|
this._mediaStream = mediaStreamNode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* Close the media stream
|
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
close(): this {
|
|
|
|
if (this._stream && this._mediaStream) {
|
|
|
|
this._stream.getAudioTracks().forEach((track) => {
|
|
|
|
track.stop();
|
|
|
|
});
|
|
|
|
this._stream = undefined;
|
|
|
|
// remove the old media stream
|
|
|
|
this._mediaStream.disconnect();
|
|
|
|
this._mediaStream = undefined;
|
|
|
|
}
|
|
|
|
this._device = undefined;
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* Returns a promise which resolves with the list of audio input devices available.
|
|
|
|
* @return The promise that is resolved with the devices
|
|
|
|
* @example
|
2019-10-25 20:54:33 +00:00
|
|
|
* import { UserMedia } from "tone";
|
2019-10-23 03:04:52 +00:00
|
|
|
* UserMedia.enumerateDevices().then((devices) => {
|
2019-10-25 20:54:33 +00:00
|
|
|
* console.log(devices);
|
|
|
|
* });
|
2019-10-23 03:04:52 +00:00
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
static async enumerateDevices(): Promise<MediaDeviceInfo[]> {
|
|
|
|
const allDevices = await navigator.mediaDevices.enumerateDevices();
|
|
|
|
return allDevices.filter(device => {
|
|
|
|
return device.kind === "audioinput";
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* Returns the playback state of the source, "started" when the microphone is open
|
|
|
|
* and "stopped" when the mic is closed.
|
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
get state() {
|
|
|
|
return this._stream && this._stream.active ? "started" : "stopped";
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +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
|
|
|
|
* device is not open.
|
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
get deviceId(): string | undefined {
|
|
|
|
if (this._device) {
|
|
|
|
return this._device.deviceId;
|
|
|
|
} else {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* Returns a group identifier. Two devices have the
|
|
|
|
* same group identifier if they belong to the same physical device.
|
|
|
|
* Returns null when the device is not open.
|
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
get groupId(): string | undefined {
|
|
|
|
if (this._device) {
|
|
|
|
return this._device.groupId;
|
|
|
|
} else {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* Returns a label describing this device (for example "Built-in Microphone").
|
|
|
|
* Returns undefined when the device is not open or label is not available
|
|
|
|
* because of permissions.
|
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
get label(): string | undefined {
|
|
|
|
if (this._device) {
|
|
|
|
return this._device.label;
|
|
|
|
} else {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* Mute the output.
|
|
|
|
* @example
|
2019-10-25 20:54:33 +00:00
|
|
|
* import { UserMedia } from "tone";
|
|
|
|
* const mic = new UserMedia();
|
|
|
|
* mic.open().then(() => {
|
|
|
|
* // promise resolves when input is available
|
|
|
|
* });
|
|
|
|
* // mute the output
|
|
|
|
* mic.mute = true;
|
2019-10-23 03:04:52 +00:00
|
|
|
*/
|
2019-09-20 22:51:17 +00:00
|
|
|
get mute(): boolean {
|
|
|
|
return this._volume.mute;
|
|
|
|
}
|
|
|
|
set mute(mute) {
|
|
|
|
this._volume.mute = mute;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this.close();
|
|
|
|
this._volume.dispose();
|
|
|
|
this.volume.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-10-23 03:04:52 +00:00
|
|
|
* If getUserMedia is supported by the browser.
|
2019-09-20 22:51:17 +00:00
|
|
|
*/
|
|
|
|
static get supported(): boolean {
|
|
|
|
return isDefined(navigator.mediaDevices) &&
|
|
|
|
isDefined(navigator.mediaDevices.getUserMedia);
|
|
|
|
}
|
|
|
|
}
|