2020-04-15 02:03:47 +00:00
|
|
|
import { ToneAudioNode, ToneAudioNodeOptions } from "../../core/context/ToneAudioNode";
|
|
|
|
import { Gain } from "../../core/context/Gain";
|
|
|
|
import { assert } from "../../core/util/Debug";
|
|
|
|
import { theWindow } from "../../core/context/AudioContext";
|
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults";
|
|
|
|
import { PlaybackState } from "../../core/util/StateTimeline";
|
|
|
|
|
|
|
|
export interface RecorderOptions extends ToneAudioNodeOptions {
|
|
|
|
mimeType?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-05-27 01:09:32 +00:00
|
|
|
* A wrapper around the MediaRecorder API. Unlike the rest of Tone.js, this module does not offer
|
|
|
|
* any sample-accurate scheduling because it is not a feature of the MediaRecorder API.
|
2020-04-15 02:03:47 +00:00
|
|
|
* This is only natively supported in Chrome and Firefox.
|
2020-04-26 22:02:18 +00:00
|
|
|
* For a cross-browser shim, install (audio-recorder-polyfill)[https://www.npmjs.com/package/audio-recorder-polyfill].
|
2020-04-15 02:03:47 +00:00
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const recorder = new Tone.Recorder();
|
|
|
|
* const synth = new Tone.Synth().connect(recorder);
|
2020-04-15 02:03:47 +00:00
|
|
|
* // start recording
|
|
|
|
* recorder.start();
|
|
|
|
* // generate a few notes
|
|
|
|
* synth.triggerAttackRelease("C3", 0.5);
|
|
|
|
* synth.triggerAttackRelease("C4", 0.5, "+1");
|
|
|
|
* synth.triggerAttackRelease("C5", 0.5, "+2");
|
|
|
|
* // wait for the notes to end and stop the recording
|
|
|
|
* setTimeout(async () => {
|
|
|
|
* // the recorded audio is returned as a blob
|
|
|
|
* const recording = await recorder.stop();
|
|
|
|
* // download the recording by creating an anchor element and blob url
|
|
|
|
* const url = URL.createObjectURL(recording);
|
|
|
|
* const anchor = document.createElement("a");
|
|
|
|
* anchor.download = "recording.webm";
|
|
|
|
* anchor.href = url;
|
|
|
|
* anchor.click();
|
|
|
|
* }, 4000);
|
2020-09-02 20:53:38 +00:00
|
|
|
* @category Component
|
2020-04-15 02:03:47 +00:00
|
|
|
*/
|
|
|
|
export class Recorder extends ToneAudioNode<RecorderOptions> {
|
|
|
|
|
|
|
|
readonly name = "Recorder";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recorder uses the Media Recorder API
|
|
|
|
*/
|
|
|
|
private _recorder: MediaRecorder;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* MediaRecorder requires
|
|
|
|
*/
|
|
|
|
private _stream: MediaStreamAudioDestinationNode;
|
|
|
|
|
|
|
|
readonly input: Gain;
|
|
|
|
readonly output: undefined;
|
|
|
|
|
|
|
|
constructor(options?: Partial<RecorderOptions>);
|
|
|
|
constructor() {
|
|
|
|
|
2020-07-19 19:04:03 +00:00
|
|
|
super(optionsFromArguments(Recorder.getDefaults(), arguments));
|
|
|
|
const options = optionsFromArguments(Recorder.getDefaults(), arguments);
|
2020-04-15 02:03:47 +00:00
|
|
|
|
|
|
|
this.input = new Gain({
|
|
|
|
context: this.context
|
|
|
|
});
|
|
|
|
|
|
|
|
assert(Recorder.supported, "Media Recorder API is not available");
|
|
|
|
|
|
|
|
this._stream = this.context.createMediaStreamDestination();
|
|
|
|
this.input.connect(this._stream);
|
|
|
|
this._recorder = new MediaRecorder(this._stream.stream, {
|
|
|
|
mimeType: options.mimeType
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): RecorderOptions {
|
|
|
|
return ToneAudioNode.getDefaults();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The mime type is the format that the audio is encoded in. For Chrome
|
|
|
|
* that is typically webm encoded as "vorbis".
|
|
|
|
*/
|
|
|
|
get mimeType(): string {
|
|
|
|
return this._recorder.mimeType;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Test if your platform supports the Media Recorder API. If it's not available,
|
2020-04-26 22:02:18 +00:00
|
|
|
* try installing this (polyfill)[https://www.npmjs.com/package/audio-recorder-polyfill].
|
2020-04-15 02:03:47 +00:00
|
|
|
*/
|
|
|
|
static get supported(): boolean {
|
|
|
|
return theWindow !== null && Reflect.has(theWindow, "MediaRecorder");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the playback state of the Recorder, either "started", "stopped" or "paused"
|
|
|
|
*/
|
|
|
|
get state(): PlaybackState {
|
|
|
|
if (this._recorder.state === "inactive") {
|
|
|
|
return "stopped";
|
|
|
|
} else if (this._recorder.state === "paused") {
|
|
|
|
return "paused";
|
|
|
|
} else {
|
|
|
|
return "started";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-27 01:09:32 +00:00
|
|
|
/**
|
|
|
|
* Start the Recorder. Returns a promise which resolves
|
|
|
|
* when the recorder has started.
|
|
|
|
*/
|
2020-04-26 19:38:14 +00:00
|
|
|
async start() {
|
2020-04-15 02:03:47 +00:00
|
|
|
assert(this.state !== "started", "Recorder is already started");
|
2021-10-13 17:10:36 +00:00
|
|
|
const startPromise = new Promise<void>(done => {
|
2020-04-26 19:38:14 +00:00
|
|
|
const handleStart = () => {
|
|
|
|
this._recorder.removeEventListener("start", handleStart, false);
|
|
|
|
|
|
|
|
done();
|
|
|
|
};
|
|
|
|
|
|
|
|
this._recorder.addEventListener("start", handleStart, false);
|
|
|
|
});
|
|
|
|
|
2020-04-15 02:03:47 +00:00
|
|
|
this._recorder.start();
|
2020-04-26 19:38:14 +00:00
|
|
|
return await startPromise;
|
2020-04-15 02:03:47 +00:00
|
|
|
}
|
|
|
|
|
2020-05-27 01:09:32 +00:00
|
|
|
/**
|
|
|
|
* Stop the recorder. Returns a promise with the recorded content until this point
|
2024-04-29 14:48:37 +00:00
|
|
|
* encoded as {@link mimeType}
|
2020-05-27 01:09:32 +00:00
|
|
|
*/
|
2020-04-15 02:03:47 +00:00
|
|
|
async stop(): Promise<Blob> {
|
|
|
|
assert(this.state !== "stopped", "Recorder is not started");
|
|
|
|
const dataPromise: Promise<Blob> = new Promise(done => {
|
2020-04-26 19:38:14 +00:00
|
|
|
const handleData = (e: BlobEvent) => {
|
|
|
|
this._recorder.removeEventListener("dataavailable", handleData, false);
|
|
|
|
|
|
|
|
done(e.data);
|
|
|
|
};
|
|
|
|
|
|
|
|
this._recorder.addEventListener("dataavailable", handleData, false);
|
2020-04-15 02:03:47 +00:00
|
|
|
});
|
|
|
|
this._recorder.stop();
|
|
|
|
return await dataPromise;
|
|
|
|
}
|
|
|
|
|
2020-05-27 01:09:32 +00:00
|
|
|
/**
|
|
|
|
* Pause the recorder
|
|
|
|
*/
|
2020-04-15 02:03:47 +00:00
|
|
|
pause(): this {
|
|
|
|
assert(this.state === "started", "Recorder must be started");
|
|
|
|
this._recorder.pause();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this.input.dispose();
|
|
|
|
this._stream.disconnect();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|