define(["Tone/core/Tone", "Tone/core/Buffer"], function (Tone) { /** * @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; }; return Tone.Buffers; });