2014-10-16 18:48:23 +00:00
|
|
|
define(["Tone/core/Tone"], function(Tone){
|
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
"use strict";
|
2015-01-06 02:47:07 +00:00
|
|
|
|
2014-10-16 18:48:23 +00:00
|
|
|
/**
|
2014-10-23 02:27:01 +00:00
|
|
|
* @class Buffer loading and storage. Tone.Buffer will load and store the buffers
|
|
|
|
* in the same data structure they were given in the argument. If given
|
2015-01-06 02:47:07 +00:00
|
|
|
* a string, this._buffer will equal an AudioBuffer. If constructed
|
2014-10-23 02:27:01 +00:00
|
|
|
* with an array, the samples will be placed in an array in the same
|
|
|
|
* order.
|
2014-10-21 18:44:02 +00:00
|
|
|
*
|
|
|
|
* @constructor
|
2015-01-05 01:59:08 +00:00
|
|
|
* @param {AudioBuffer|string} url the url to load, or the audio buffer to set
|
2014-10-21 18:44:02 +00:00
|
|
|
*/
|
|
|
|
Tone.Buffer = function(){
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2015-01-05 01:59:08 +00:00
|
|
|
var options = this.optionsObject(arguments, ["url", "onload"], Tone.Buffer.defaults);
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
/**
|
2015-01-05 01:59:08 +00:00
|
|
|
* stores the loaded AudioBuffer
|
|
|
|
* @type {AudioBuffer}
|
2015-01-06 02:47:07 +00:00
|
|
|
* @private
|
2015-01-05 01:59:08 +00:00
|
|
|
*/
|
2015-01-06 02:47:07 +00:00
|
|
|
this._buffer = null;
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2015-01-05 01:59:08 +00:00
|
|
|
/**
|
2015-02-02 01:02:54 +00:00
|
|
|
* the url of the buffer. `undefined` if it was
|
|
|
|
* constructed with a buffer
|
2015-01-05 01:59:08 +00:00
|
|
|
* @type {string}
|
2015-02-02 01:02:54 +00:00
|
|
|
* @readOnly
|
2015-01-05 01:59:08 +00:00
|
|
|
*/
|
|
|
|
this.url = undefined;
|
|
|
|
|
2015-02-02 02:32:49 +00:00
|
|
|
/**
|
|
|
|
* indicates if the buffer is loaded or not
|
|
|
|
* @type {boolean}
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
this.loaded = false;
|
|
|
|
|
2015-01-05 01:59:08 +00:00
|
|
|
/**
|
|
|
|
* the callback to invoke when everything is loaded
|
|
|
|
* @type {function}
|
|
|
|
*/
|
2015-02-04 15:16:49 +00:00
|
|
|
this.onload = options.onload.bind(this, this);
|
2015-01-05 01:59:08 +00:00
|
|
|
|
|
|
|
if (options.url instanceof AudioBuffer){
|
2015-01-06 02:47:07 +00:00
|
|
|
this._buffer.set(options.url);
|
|
|
|
this.onload(this);
|
2015-01-05 01:59:08 +00:00
|
|
|
} else if (typeof options.url === "string"){
|
|
|
|
this.url = options.url;
|
|
|
|
Tone.Buffer._addToQueue(options.url, this);
|
|
|
|
}
|
2014-10-21 18:44:02 +00:00
|
|
|
};
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
Tone.extend(Tone.Buffer);
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
/**
|
|
|
|
* the default parameters
|
|
|
|
*
|
|
|
|
* @static
|
|
|
|
* @const
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Tone.Buffer.defaults = {
|
2015-01-05 01:59:08 +00:00
|
|
|
"url" : undefined,
|
|
|
|
"onload" : function(){},
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set the buffer
|
2015-01-06 02:47:07 +00:00
|
|
|
* @param {AudioBuffer|Tone.Buffer} buffer the buffer
|
2015-01-06 04:33:05 +00:00
|
|
|
* @returns {Tone.Buffer} `this`
|
2015-01-05 01:59:08 +00:00
|
|
|
*/
|
|
|
|
Tone.Buffer.prototype.set = function(buffer){
|
2015-01-06 02:47:07 +00:00
|
|
|
if (buffer instanceof Tone.Buffer){
|
|
|
|
this._buffer = buffer.get();
|
|
|
|
} else {
|
|
|
|
this._buffer = buffer;
|
|
|
|
}
|
2015-02-04 15:16:49 +00:00
|
|
|
this.loaded = true;
|
2015-01-06 02:47:07 +00:00
|
|
|
return this;
|
2015-01-05 01:59:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return {AudioBuffer} the audio buffer
|
|
|
|
*/
|
|
|
|
Tone.Buffer.prototype.get = function(){
|
2015-01-06 02:47:07 +00:00
|
|
|
return this._buffer;
|
2015-01-05 01:59:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param {string} url the url to load
|
|
|
|
* @param {function=} callback the callback to invoke on load.
|
|
|
|
* don't need to set if `onload` is
|
|
|
|
* already set.
|
2015-01-06 04:33:05 +00:00
|
|
|
* @returns {Tone.Buffer} `this`
|
2015-01-05 01:59:08 +00:00
|
|
|
*/
|
|
|
|
Tone.Buffer.prototype.load = function(url, callback){
|
|
|
|
this.url = url;
|
|
|
|
this.onload = this.defaultArg(callback, this.onload);
|
|
|
|
Tone.Buffer._addToQueue(url, this);
|
2015-01-06 04:33:05 +00:00
|
|
|
return this;
|
2015-01-05 01:59:08 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* dispose and disconnect
|
2015-02-02 03:05:24 +00:00
|
|
|
* @returns {Tone.Buffer} `this`
|
2015-01-05 01:59:08 +00:00
|
|
|
*/
|
2015-02-02 03:05:24 +00:00
|
|
|
Tone.Buffer.prototype.dispose = function(){
|
|
|
|
Tone.prototype.dispose.call(this);
|
2015-01-05 01:59:08 +00:00
|
|
|
Tone.Buffer._removeFromQueue(this);
|
2015-01-06 02:47:07 +00:00
|
|
|
this._buffer = null;
|
2015-01-05 01:59:08 +00:00
|
|
|
this.onload = null;
|
2015-02-02 03:05:24 +00:00
|
|
|
return this;
|
2015-01-05 01:59:08 +00:00
|
|
|
};
|
|
|
|
|
2015-02-02 01:02:54 +00:00
|
|
|
/**
|
|
|
|
* the duration of the buffer
|
|
|
|
* @memberOf Tone.Buffer#
|
|
|
|
* @type {number}
|
|
|
|
* @name duration
|
|
|
|
* @readOnly
|
|
|
|
*/
|
2015-01-06 02:47:07 +00:00
|
|
|
Object.defineProperty(Tone.Buffer.prototype, "duration", {
|
|
|
|
get : function(){
|
2015-02-02 01:02:54 +00:00
|
|
|
if (this._buffer){
|
|
|
|
return this._buffer.duration;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2015-01-06 02:47:07 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
|
2015-01-05 01:59:08 +00:00
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
// STATIC METHODS
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the static queue for all of the xhr requests
|
|
|
|
* @type {Array}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._queue = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the array of current downloads
|
|
|
|
* @type {Array}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._currentDownloads = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the total number of downloads
|
|
|
|
* @type {number}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._totalDownloads = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the maximum number of simultaneous downloads
|
|
|
|
* @static
|
|
|
|
* @type {number}
|
|
|
|
*/
|
|
|
|
Tone.Buffer.MAX_SIMULTANEOUS_DOWNLOADS = 6;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds a file to be loaded to the loading queue
|
|
|
|
* @param {string} url the url to load
|
|
|
|
* @param {function} callback the callback to invoke once it's loaded
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._addToQueue = function(url, buffer){
|
|
|
|
Tone.Buffer._queue.push({
|
|
|
|
url : url,
|
|
|
|
Buffer : buffer,
|
|
|
|
progress : 0,
|
|
|
|
xhr : null
|
|
|
|
});
|
|
|
|
this._totalDownloads++;
|
|
|
|
Tone.Buffer._next();
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove an object from the queue's (if it's still there)
|
|
|
|
* Abort the XHR if it's in progress
|
|
|
|
* @param {Tone.Buffer} buffer the buffer to remove
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._removeFromQueue = function(buffer){
|
|
|
|
var i;
|
|
|
|
for (i = 0; i < Tone.Buffer._queue.length; i++){
|
|
|
|
var q = Tone.Buffer._queue[i];
|
|
|
|
if (q.Buffer === buffer){
|
|
|
|
Tone.Buffer._queue.splice(i, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for (i = 0; i < Tone.Buffer._currentDownloads.length; i++){
|
|
|
|
var dl = Tone.Buffer._currentDownloads[i];
|
|
|
|
if (dl.Buffer === buffer){
|
|
|
|
Tone.Buffer._currentDownloads.splice(i, 1);
|
|
|
|
dl.xhr.abort();
|
|
|
|
dl.xhr.onprogress = null;
|
|
|
|
dl.xhr.onload = null;
|
|
|
|
dl.xhr.onerror = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* load the next buffer in the queue
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._next = function(){
|
|
|
|
if (Tone.Buffer._queue.length > 0){
|
|
|
|
if (Tone.Buffer._currentDownloads.length < Tone.Buffer.MAX_SIMULTANEOUS_DOWNLOADS){
|
|
|
|
var next = Tone.Buffer._queue.shift();
|
|
|
|
Tone.Buffer._currentDownloads.push(next);
|
|
|
|
next.xhr = Tone.Buffer.load(next.url, function(buffer){
|
|
|
|
//remove this one from the queue
|
|
|
|
var index = Tone.Buffer._currentDownloads.indexOf(next);
|
|
|
|
Tone.Buffer._currentDownloads.splice(index, 1);
|
|
|
|
next.Buffer.set(buffer);
|
|
|
|
next.Buffer.onload(next.Buffer);
|
|
|
|
Tone.Buffer._onprogress();
|
|
|
|
Tone.Buffer._next();
|
|
|
|
});
|
|
|
|
next.xhr.onprogress = function(event){
|
|
|
|
next.progress = event.loaded / event.total;
|
|
|
|
Tone.Buffer._onprogress();
|
|
|
|
};
|
|
|
|
next.xhr.onerror = Tone.Buffer.onerror;
|
|
|
|
}
|
|
|
|
} else if (Tone.Buffer._currentDownloads.length === 0){
|
|
|
|
Tone.Buffer.onload();
|
2015-01-06 02:47:07 +00:00
|
|
|
//reset the downloads
|
|
|
|
Tone.Buffer._totalDownloads = 0;
|
2015-01-05 01:59:08 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* internal progress event handler
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Buffer._onprogress = function(){
|
|
|
|
var curretDownloadsProgress = 0;
|
|
|
|
var currentDLLen = Tone.Buffer._currentDownloads.length;
|
|
|
|
var inprogress = 0;
|
|
|
|
if (currentDLLen > 0){
|
|
|
|
for (var i = 0; i < currentDLLen; i++){
|
|
|
|
var dl = Tone.Buffer._currentDownloads[i];
|
|
|
|
curretDownloadsProgress += dl.progress;
|
|
|
|
}
|
|
|
|
inprogress = curretDownloadsProgress;
|
|
|
|
}
|
|
|
|
var currentDownloadProgress = currentDLLen - inprogress;
|
|
|
|
var completed = Tone.Buffer._totalDownloads - Tone.Buffer._queue.length - currentDownloadProgress;
|
|
|
|
Tone.Buffer.onprogress(completed / Tone.Buffer._totalDownloads);
|
2014-10-16 18:48:23 +00:00
|
|
|
};
|
2014-10-21 18:44:02 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* makes an xhr reqest for the selected url
|
|
|
|
* Load the audio file as an audio buffer.
|
|
|
|
* Decodes the audio asynchronously and invokes
|
|
|
|
* the callback once the audio buffer loads.
|
|
|
|
* @param {string} url the url of the buffer to load.
|
|
|
|
* filetype support depends on the
|
|
|
|
* browser.
|
|
|
|
* @param {function} callback function
|
2015-01-05 01:59:08 +00:00
|
|
|
* @returns {XMLHttpRequest} returns the XHR
|
2014-10-21 18:44:02 +00:00
|
|
|
*/
|
2015-01-05 01:59:08 +00:00
|
|
|
Tone.Buffer.load = function(url, callback){
|
2014-10-21 18:44:02 +00:00
|
|
|
var request = new XMLHttpRequest();
|
|
|
|
request.open("GET", url, true);
|
|
|
|
request.responseType = "arraybuffer";
|
|
|
|
// decode asynchronously
|
|
|
|
request.onload = function() {
|
2015-01-05 01:59:08 +00:00
|
|
|
Tone.context.decodeAudioData(request.response, function(buff) {
|
2014-10-21 18:44:02 +00:00
|
|
|
if(!buff){
|
2015-01-05 01:59:08 +00:00
|
|
|
throw new Error("could not decode audio data:" + url);
|
2014-10-21 18:44:02 +00:00
|
|
|
}
|
|
|
|
callback(buff);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
//send the request
|
|
|
|
request.send();
|
2015-01-05 01:59:08 +00:00
|
|
|
return request;
|
2014-10-16 18:48:23 +00:00
|
|
|
};
|
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
/**
|
2015-01-05 01:59:08 +00:00
|
|
|
* callback when all of the buffers in the queue have loaded
|
|
|
|
* @static
|
|
|
|
* @type {function}
|
2014-10-21 18:44:02 +00:00
|
|
|
*/
|
2015-01-05 01:59:08 +00:00
|
|
|
Tone.Buffer.onload = function(){};
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
/**
|
2015-01-05 01:59:08 +00:00
|
|
|
* callback with the progress of all of the loads in the queue
|
|
|
|
* @static
|
|
|
|
* @type {function}
|
2014-10-21 18:44:02 +00:00
|
|
|
*/
|
2015-01-05 01:59:08 +00:00
|
|
|
Tone.Buffer.onprogress = function(){};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* callback if one of the buffers in the queue encounters an error
|
|
|
|
* @static
|
|
|
|
* @type {function}
|
|
|
|
*/
|
|
|
|
Tone.Buffer.onerror = function(){};
|
2014-10-16 18:48:23 +00:00
|
|
|
|
2014-10-21 18:44:02 +00:00
|
|
|
return Tone.Buffer;
|
2014-10-16 18:48:23 +00:00
|
|
|
});
|