Tone.js/Tone/core/context/Buffers.js
2019-05-23 14:00:49 -04:00

165 lines
4.1 KiB
JavaScript

import Tone from "../core/Tone";
import "./Buffer";
/**
* @class A data structure for holding multiple buffers.
*
* @param {Object|Array} urls An object literal or array
* of urls to load.
* @param {Function=} callback The callback to invoke when
* the buffers are loaded.
* @extends {Tone}
* @example
* //load a whole bank of piano samples
* var pianoSamples = new Tone.Buffers({
* "C4" : "path/to/C4.mp3"
* "C#4" : "path/to/C#4.mp3"
* "D4" : "path/to/D4.mp3"
* "D#4" : "path/to/D#4.mp3"
* ...
* }, function(){
* //play one of the samples when they all load
* player.buffer = pianoSamples.get("C4");
* player.start();
* });
* @example
* //To pass in additional parameters in the second parameter
* var buffers = new Tone.Buffers(urls, {
* "onload" : callback,
* "baseUrl" : "../path/to/audio/"
* })
*/
Tone.Buffers = function(urls) {
//remove the urls from the options
var args = Array.prototype.slice.call(arguments);
args.shift();
var options = Tone.defaults(args, ["onload", "baseUrl"], Tone.Buffers);
Tone.call(this);
/**
* All of the buffers
* @type {Object}
* @private
*/
this._buffers = {};
/**
* A path which is prefixed before every url.
* @type {String}
*/
this.baseUrl = options.baseUrl;
this._loadingCount = 0;
//add each one
for (var key in urls) {
this._loadingCount++;
this.add(key, urls[key], this._bufferLoaded.bind(this, options.onload));
}
};
Tone.extend(Tone.Buffers);
/**
* Defaults
* @type {Object}
*/
Tone.Buffers.defaults = {
"onload" : Tone.noOp,
"baseUrl" : "",
};
/**
* True if the buffers object has a buffer by that name.
* @param {String|Number} name The key or index of the
* buffer.
* @return {Boolean}
*/
Tone.Buffers.prototype.has = function(name) {
return this._buffers.hasOwnProperty(name);
};
/**
* Get a buffer by name. If an array was loaded,
* then use the array index.
* @param {String|Number} name The key or index of the
* buffer.
* @return {Tone.Buffer}
*/
Tone.Buffers.prototype.get = function(name) {
if (this.has(name)) {
return this._buffers[name];
} else {
throw new Error("Tone.Buffers: no buffer named " + name);
}
};
/**
* A buffer was loaded. decrement the counter.
* @param {Function} callback
* @private
*/
Tone.Buffers.prototype._bufferLoaded = function(callback) {
this._loadingCount--;
if (this._loadingCount === 0 && callback) {
callback(this);
}
};
/**
* If the buffers are loaded or not
* @memberOf Tone.Buffers#
* @type {Boolean}
* @name loaded
* @readOnly
*/
Object.defineProperty(Tone.Buffers.prototype, "loaded", {
get : function() {
var isLoaded = true;
for (var buffName in this._buffers) {
var buff = this.get(buffName);
isLoaded = isLoaded && buff.loaded;
}
return isLoaded;
},
});
/**
* Add a buffer by name and url to the Buffers
* @param {String} name A unique name to give
* the buffer
* @param {String|Tone.Buffer|Audiobuffer} url Either the url of the bufer,
* or a buffer which will be added
* with the given name.
* @param {Function=} callback The callback to invoke
* when the url is loaded.
*/
Tone.Buffers.prototype.add = function(name, url, callback) {
callback = Tone.defaultArg(callback, Tone.noOp);
if (url instanceof Tone.Buffer) {
this._buffers[name] = url;
callback(this);
} else if (url instanceof AudioBuffer) {
this._buffers[name] = new Tone.Buffer(url);
callback(this);
} else if (Tone.isString(url)) {
this._buffers[name] = new Tone.Buffer(this.baseUrl + url, callback);
}
return this;
};
/**
* Clean up.
* @return {Tone.Buffers} this
*/
Tone.Buffers.prototype.dispose = function() {
Tone.prototype.dispose.call(this);
for (var name in this._buffers) {
this._buffers[name].dispose();
}
this._buffers = null;
return this;
};
export default Tone.Buffers;