Tone.js/Tone/source/UserMedia.ts

275 lines
7.2 KiB
TypeScript
Raw Normal View History

2024-05-03 15:09:28 +00:00
import {
connect,
OutputNode,
ToneAudioNode,
ToneAudioNodeOptions,
} from "../core/context/ToneAudioNode.js";
2024-05-03 14:10:40 +00:00
import { Decibels } from "../core/type/Units.js";
import { Volume } from "../component/channel/Volume.js";
import { optionsFromArguments } from "../core/util/Defaults.js";
import { assert } from "../core/util/Debug.js";
import { Param } from "../core/context/Param.js";
import { readOnly } from "../core/util/Interface.js";
import { isDefined, isNumber } from "../core/util/TypeCheck.js";
2019-09-20 22:51:17 +00:00
export interface UserMediaOptions extends ToneAudioNodeOptions {
volume: Decibels;
mute: boolean;
}
/**
2024-05-03 15:09:28 +00:00
* UserMedia uses MediaDevices.getUserMedia to open up and external microphone or audio input.
2019-09-20 22:51:17 +00:00
* 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
2020-06-17 14:31:26 +00:00
* const meter = new Tone.Meter();
* const mic = new Tone.UserMedia().connect(meter);
2019-10-25 20:54:33 +00:00
* mic.open().then(() => {
* // promise resolves when input is available
2020-06-17 14:31:26 +00:00
* console.log("mic open");
* // print the incoming mic levels in decibels
* setInterval(() => console.log(meter.getValue()), 100);
* }).catch(e => {
* // promise is rejected when the user doesn't have or allow mic access
* console.log("mic not open");
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.
*/
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() {
2024-05-03 15:09:28 +00:00
super(
optionsFromArguments(UserMedia.getDefaults(), arguments, ["volume"])
);
const options = optionsFromArguments(
UserMedia.getDefaults(),
arguments,
["volume"]
);
2019-09-20 22:51:17 +00:00
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,
2024-05-03 15:09:28 +00:00
volume: 0,
2019-09-20 22:51:17 +00:00
});
}
/**
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-12-13 16:25:28 +00:00
async open(labelOrId?: string | number): Promise<this> {
assert(UserMedia.supported, "UserMedia is not supported");
2019-09-20 22:51:17 +00:00
// 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) => {
2024-05-03 15:09:28 +00:00
return (
device.label === labelOrId || device.deviceId === labelOrId
);
2019-09-20 22:51:17 +00:00
});
// didn't find a matching device
if (!this._device && devices.length > 0) {
this._device = devices[0];
}
assert(isDefined(this._device), `No matching device ${labelOrId}`);
2019-09-20 22:51:17 +00:00
}
// do getUserMedia
2019-11-17 18:09:19 +00:00
const constraints = {
2019-09-20 22:51:17 +00:00
audio: {
echoCancellation: false,
sampleRate: this.context.sampleRate,
noiseSuppression: false,
mozNoiseSuppression: false,
2024-05-03 15:09:28 +00:00
},
2019-09-20 22:51:17 +00:00
};
if (this._device) {
// @ts-ignore
constraints.audio.deviceId = this._device.deviceId;
2019-09-20 22:51:17 +00:00
}
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.
2024-05-03 15:09:28 +00:00
const mediaStreamNode =
this.context.createMediaStreamSource(stream);
2019-09-20 22:51:17 +00:00
// Connect the MediaStreamSourceNode to a gate gain node
connect(mediaStreamNode, this.output);
this._mediaStream = mediaStreamNode;
}
2019-12-13 16:25:28 +00:00
return this;
2019-09-20 22:51:17 +00:00
}
/**
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
* Tone.UserMedia.enumerateDevices().then((devices) => {
2019-11-04 02:44:25 +00:00
* // print the device labels
* console.log(devices.map(device => device.label));
2019-10-25 20:54:33 +00:00
* });
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();
2024-05-03 15:09:28 +00:00
return allDevices.filter((device) => {
2019-09-20 22:51:17 +00:00
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
* const mic = new Tone.UserMedia();
2019-10-25 20:54:33 +00:00
* 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 {
2024-05-03 15:09:28 +00:00
return (
isDefined(navigator.mediaDevices) &&
isDefined(navigator.mediaDevices.getUserMedia)
);
2019-09-20 22:51:17 +00:00
}
}