2024-05-03 14:10:40 +00:00
|
|
|
import { Tone } from "../Tone.js";
|
|
|
|
import { optionsFromArguments } from "../util/Defaults.js";
|
|
|
|
import { noOp } from "../util/Interface.js";
|
|
|
|
import { isString } from "../util/TypeCheck.js";
|
|
|
|
import { ToneAudioBuffer } from "./ToneAudioBuffer.js";
|
|
|
|
import { assert } from "../util/Debug.js";
|
2019-07-22 16:29:50 +00:00
|
|
|
|
2019-09-10 03:39:00 +00:00
|
|
|
export interface ToneAudioBuffersUrlMap {
|
2019-07-22 20:15:55 +00:00
|
|
|
[name: string]: string | AudioBuffer | ToneAudioBuffer;
|
|
|
|
[name: number]: string | AudioBuffer | ToneAudioBuffer;
|
2019-07-22 16:29:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
interface ToneAudioBuffersOptions {
|
|
|
|
urls: ToneAudioBuffersUrlMap;
|
|
|
|
onload: () => void;
|
|
|
|
onerror?: (error: Error) => void;
|
|
|
|
baseUrl: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A data structure for holding multiple buffers in a Map-like datastructure.
|
|
|
|
*
|
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const pianoSamples = new Tone.ToneAudioBuffers({
|
2020-07-20 18:11:00 +00:00
|
|
|
* A1: "https://tonejs.github.io/audio/casio/A1.mp3",
|
|
|
|
* A2: "https://tonejs.github.io/audio/casio/A2.mp3",
|
2019-10-24 22:01:27 +00:00
|
|
|
* }, () => {
|
2020-04-17 02:24:18 +00:00
|
|
|
* const player = new Tone.Player().toDestination();
|
2019-10-24 22:01:27 +00:00
|
|
|
* // play one of the samples when they all load
|
2020-09-02 04:07:45 +00:00
|
|
|
* player.buffer = pianoSamples.get("A2");
|
2019-09-16 15:09:44 +00:00
|
|
|
* player.start();
|
2019-07-22 16:29:50 +00:00
|
|
|
* });
|
|
|
|
* @example
|
2019-10-24 22:01:27 +00:00
|
|
|
* // To pass in additional parameters in the second parameter
|
2020-04-17 02:24:18 +00:00
|
|
|
* const buffers = new Tone.ToneAudioBuffers({
|
2019-10-24 22:01:27 +00:00
|
|
|
* urls: {
|
2020-07-20 18:11:00 +00:00
|
|
|
* A1: "A1.mp3",
|
|
|
|
* A2: "A2.mp3",
|
2019-10-24 22:01:27 +00:00
|
|
|
* },
|
|
|
|
* onload: () => console.log("loaded"),
|
2020-07-20 18:11:00 +00:00
|
|
|
* baseUrl: "https://tonejs.github.io/audio/casio/"
|
2019-10-24 22:01:27 +00:00
|
|
|
* });
|
2019-08-21 20:01:12 +00:00
|
|
|
* @category Core
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
|
|
|
export class ToneAudioBuffers extends Tone {
|
2019-09-04 23:18:44 +00:00
|
|
|
readonly name: string = "ToneAudioBuffers";
|
2019-07-22 16:29:50 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* All of the buffers
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
|
|
|
private _buffers: Map<string, ToneAudioBuffer> = new Map();
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* A path which is prefixed before every url.
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
|
|
|
baseUrl: string;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Keep track of the number of loaded buffers
|
|
|
|
*/
|
2019-11-17 18:09:19 +00:00
|
|
|
private _loadingCount = 0;
|
2019-07-22 16:29:50 +00:00
|
|
|
|
2019-08-27 17:02:31 +00:00
|
|
|
/**
|
|
|
|
* @param urls An object literal or array of urls to load.
|
|
|
|
* @param onload The callback to invoke when the buffers are loaded.
|
|
|
|
* @param baseUrl A prefix url to add before all the urls
|
|
|
|
*/
|
2019-07-22 16:29:50 +00:00
|
|
|
constructor(
|
|
|
|
urls?: ToneAudioBuffersUrlMap,
|
|
|
|
onload?: () => void,
|
2024-05-03 15:09:28 +00:00
|
|
|
baseUrl?: string
|
2019-07-30 19:35:27 +00:00
|
|
|
);
|
2019-07-22 16:29:50 +00:00
|
|
|
constructor(options?: Partial<ToneAudioBuffersOptions>);
|
|
|
|
constructor() {
|
|
|
|
super();
|
2019-07-22 20:15:55 +00:00
|
|
|
const options = optionsFromArguments(
|
2024-05-03 15:09:28 +00:00
|
|
|
ToneAudioBuffers.getDefaults(),
|
|
|
|
arguments,
|
|
|
|
["urls", "onload", "baseUrl"],
|
|
|
|
"urls"
|
2019-07-22 20:15:55 +00:00
|
|
|
);
|
2019-07-22 16:29:50 +00:00
|
|
|
|
|
|
|
this.baseUrl = options.baseUrl;
|
|
|
|
// add each one
|
2024-05-03 15:09:28 +00:00
|
|
|
Object.keys(options.urls).forEach((name) => {
|
2019-07-22 16:29:50 +00:00
|
|
|
this._loadingCount++;
|
2019-07-22 20:15:55 +00:00
|
|
|
const url = options.urls[name];
|
2024-05-03 15:09:28 +00:00
|
|
|
this.add(
|
|
|
|
name,
|
|
|
|
url,
|
|
|
|
this._bufferLoaded.bind(this, options.onload),
|
|
|
|
options.onerror
|
|
|
|
);
|
2019-07-22 16:29:50 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): ToneAudioBuffersOptions {
|
|
|
|
return {
|
|
|
|
baseUrl: "",
|
|
|
|
onerror: noOp,
|
|
|
|
onload: noOp,
|
|
|
|
urls: {},
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* True if the buffers object has a buffer by that name.
|
2019-08-30 16:06:38 +00:00
|
|
|
* @param name The key or index of the buffer.
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
2019-07-22 20:15:55 +00:00
|
|
|
has(name: string | number): boolean {
|
|
|
|
return this._buffers.has(name.toString());
|
2019-07-22 16:29:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Get a buffer by name. If an array was loaded,
|
|
|
|
* then use the array index.
|
2019-08-30 16:06:38 +00:00
|
|
|
* @param name The key or index of the buffer.
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
2019-07-22 20:15:55 +00:00
|
|
|
get(name: string | number): ToneAudioBuffer {
|
2019-12-16 20:58:31 +00:00
|
|
|
assert(this.has(name), `ToneAudioBuffers has no buffer named: ${name}`);
|
2019-07-22 20:15:55 +00:00
|
|
|
return this._buffers.get(name.toString()) as ToneAudioBuffer;
|
2019-07-22 16:29:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* A buffer was loaded. decrement the counter.
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
|
|
|
private _bufferLoaded(callback: () => void): void {
|
|
|
|
this._loadingCount--;
|
|
|
|
if (this._loadingCount === 0 && callback) {
|
|
|
|
callback();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If the buffers are loaded or not
|
|
|
|
*/
|
|
|
|
get loaded(): boolean {
|
2019-09-10 04:06:52 +00:00
|
|
|
return Array.from(this._buffers).every(([_, buffer]) => buffer.loaded);
|
2019-07-22 16:29:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Add a buffer by name and url to the Buffers
|
2019-08-30 16:06:38 +00:00
|
|
|
* @param name A unique name to give the buffer
|
|
|
|
* @param url Either the url of the bufer, or a buffer which will be added with the given name.
|
|
|
|
* @param callback The callback to invoke when the url is loaded.
|
2020-01-30 19:25:06 +00:00
|
|
|
* @param onerror Invoked if the buffer can't be loaded
|
2019-07-22 16:29:50 +00:00
|
|
|
*/
|
|
|
|
add(
|
2019-07-22 20:15:55 +00:00
|
|
|
name: string | number,
|
2019-07-22 16:29:50 +00:00
|
|
|
url: string | AudioBuffer | ToneAudioBuffer,
|
|
|
|
callback: () => void = noOp,
|
2024-05-03 15:09:28 +00:00
|
|
|
onerror: (e: Error) => void = noOp
|
2019-07-22 16:29:50 +00:00
|
|
|
): this {
|
2019-09-10 03:51:31 +00:00
|
|
|
if (isString(url)) {
|
2021-05-26 10:59:44 +00:00
|
|
|
// don't include the baseUrl if the url is a base64 encoded sound
|
2024-05-03 15:09:28 +00:00
|
|
|
if (
|
|
|
|
this.baseUrl &&
|
|
|
|
url.trim().substring(0, 11).toLowerCase() === "data:audio/"
|
|
|
|
) {
|
2021-05-26 10:59:44 +00:00
|
|
|
this.baseUrl = "";
|
|
|
|
}
|
2024-05-03 15:09:28 +00:00
|
|
|
this._buffers.set(
|
|
|
|
name.toString(),
|
|
|
|
new ToneAudioBuffer(this.baseUrl + url, callback, onerror)
|
|
|
|
);
|
2019-09-10 03:51:31 +00:00
|
|
|
} else {
|
2024-05-03 15:09:28 +00:00
|
|
|
this._buffers.set(
|
|
|
|
name.toString(),
|
|
|
|
new ToneAudioBuffer(url, callback, onerror)
|
|
|
|
);
|
2019-07-22 16:29:50 +00:00
|
|
|
}
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
2019-07-23 16:11:57 +00:00
|
|
|
super.dispose();
|
2024-05-03 15:09:28 +00:00
|
|
|
this._buffers.forEach((buffer) => buffer.dispose());
|
2019-07-22 20:15:55 +00:00
|
|
|
this._buffers.clear();
|
2019-07-22 16:29:50 +00:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|