Tone.js/Tone/core/Buffer.js

452 lines
12 KiB
JavaScript
Raw Normal View History

define(["Tone/core/Tone", "Tone/core/Emitter", "Tone/type/Type"], function(Tone){
2014-10-21 18:44:02 +00:00
"use strict";
/**
2015-02-25 21:20:12 +00:00
* @class Buffer loading and storage. Tone.Buffer is used internally by all
2015-06-13 23:29:25 +00:00
* classes that make requests for audio files such as Tone.Player,
* Tone.Sampler and Tone.Convolver.
* <br><br>
* Aside from load callbacks from individual buffers, Tone.Buffer
2015-02-25 21:20:12 +00:00
* provides static methods which keep track of the loading progress
* of all of the buffers. These methods are Tone.Buffer.on("load" / "progress" / "error")
2015-02-25 21:20:12 +00:00
*
2014-10-21 18:44:02 +00:00
* @constructor
2015-05-05 20:40:52 +00:00
* @extends {Tone}
2015-06-14 00:52:51 +00:00
* @param {AudioBuffer|string} url The url to load, or the audio buffer to set.
2016-08-16 19:26:51 +00:00
* @param {Function=} onload A callback which is invoked after the buffer is loaded.
2015-06-14 00:52:51 +00:00
* It's recommended to use Tone.Buffer.onload instead
* since it will give you a callback when ALL buffers are loaded.
2016-08-16 19:26:51 +00:00
* @param {Function=} onerror The callback to invoke if there is an error
2015-06-14 00:52:51 +00:00
* @example
* var buffer = new Tone.Buffer("path/to/sound.mp3", function(){
* //the buffer is now available.
* var buff = buffer.get();
* });
2014-10-21 18:44:02 +00:00
*/
Tone.Buffer = function(){
2016-08-16 19:26:51 +00:00
var options = this.optionsObject(arguments, ["url", "onload", "onerror"], Tone.Buffer.defaults);
2014-10-21 18:44:02 +00:00
/**
* stores the loaded AudioBuffer
* @type {AudioBuffer}
* @private
*/
this._buffer = null;
2015-03-26 14:51:44 +00:00
/**
* indicates if the buffer should be reversed or not
2016-08-16 19:26:51 +00:00
* @type {Boolean}
2015-03-26 14:51:44 +00:00
* @private
*/
this._reversed = options.reverse;
/**
* The download progress indicator
* @type {Number}
* @private
*/
this._progress = 0;
2016-08-16 19:26:51 +00:00
/**
* The XHR
* @type {XMLHttpRequest}
* @private
2016-08-16 19:26:51 +00:00
*/
this._xhr = null;
2016-08-16 19:26:51 +00:00
if (options.url instanceof AudioBuffer || options.url instanceof Tone.Buffer){
this.set(options.url);
// invoke the onload callback
if (options.onload){
options.onload(this);
}
2015-10-21 17:13:15 +00:00
} else if (this.isString(options.url)){
this.load(options.url, options.onload, options.onerror);
}
2014-10-21 18:44:02 +00:00
};
2014-10-21 18:44:02 +00:00
Tone.extend(Tone.Buffer);
2014-10-21 18:44:02 +00:00
/**
* the default parameters
* @type {Object}
*/
Tone.Buffer.defaults = {
"url" : undefined,
2015-03-26 14:51:44 +00:00
"reverse" : false
};
/**
2015-06-14 00:52:51 +00:00
* Pass in an AudioBuffer or Tone.Buffer to set the value
* of this buffer.
* @param {AudioBuffer|Tone.Buffer} buffer the buffer
2015-06-14 00:52:51 +00:00
* @returns {Tone.Buffer} this
*/
Tone.Buffer.prototype.set = function(buffer){
if (buffer instanceof Tone.Buffer){
this._buffer = buffer.get();
} else {
this._buffer = buffer;
}
return this;
};
/**
2015-06-14 00:52:51 +00:00
* @return {AudioBuffer} The audio buffer stored in the object.
*/
Tone.Buffer.prototype.get = function(){
return this._buffer;
};
/**
* Makes an xhr reqest for the selected url then decodes
* the file as an audio buffer. Invokes
* the callback once the audio buffer loads.
* @param {String} url The url of the buffer to load.
* filetype support depends on the
* browser.
* @returns {Promise} returns a Promise which resolves with the Tone.Buffer
*/
Tone.Buffer.prototype.load = function(url, onload, onerror){
var promise = new Promise(function(load, error){
this._xhr = Tone.Buffer.load(url,
//success
function(buff){
this._xhr = null;
this.set(buff);
load(this);
if (onload){
onload(this);
}
}.bind(this),
//error
function(err){
this._xhr = null;
error(err);
if (onerror){
onerror(err);
}
}.bind(this));
}.bind(this));
return promise;
};
/**
* dispose and disconnect
2015-06-14 00:52:51 +00:00
* @returns {Tone.Buffer} this
*/
2015-02-02 03:05:24 +00:00
Tone.Buffer.prototype.dispose = function(){
Tone.Emitter.prototype.dispose.call(this);
this._buffer = null;
if (this._xhr){
Tone.Buffer._currentDownloads--;
this._xhr.abort();
this._xhr = null;
}
2015-02-02 03:05:24 +00:00
return this;
};
/**
* If the buffer is loaded or not
* @memberOf Tone.Buffer#
* @type {Boolean}
* @name loaded
* @readOnly
*/
Object.defineProperty(Tone.Buffer.prototype, "loaded", {
get : function(){
return this.length > 0;
},
});
2015-02-02 01:02:54 +00:00
/**
2015-06-14 00:52:51 +00:00
* The duration of the buffer.
2015-02-02 01:02:54 +00:00
* @memberOf Tone.Buffer#
* @type {Number}
2015-02-02 01:02:54 +00:00
* @name duration
* @readOnly
*/
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;
}
},
});
/**
* The length of the buffer in samples
* @memberOf Tone.Buffer#
* @type {Number}
* @name length
* @readOnly
*/
Object.defineProperty(Tone.Buffer.prototype, "length", {
get : function(){
if (this._buffer){
return this._buffer.length;
} else {
return 0;
}
},
});
/**
* The number of discrete audio channels. Returns 0 if no buffer
* is loaded.
* @memberOf Tone.Buffer#
* @type {Number}
* @name numberOfChannels
* @readOnly
*/
Object.defineProperty(Tone.Buffer.prototype, "numberOfChannels", {
get : function(){
if (this._buffer){
return this._buffer.numberOfChannels;
} else {
return 0;
}
},
});
/**
* Set the audio buffer from the array
* @param {Float32Array} array The array to fill the audio buffer
* @param {Number} [channels=1] The number of channels contained in the array.
* If the channel is more than 1, the input array
* is expected to be a multidimensional array
* with dimensions equal to the number of channels.
* @return {Tone.Buffer} this
*/
Tone.Buffer.prototype.fromArray = function(array){
var isMultidimensional = array[0].length > 0;
var channels = isMultidimensional ? array.length : 1;
var len = isMultidimensional ? array[0].length : array.length;
var buffer = this.context.createBuffer(channels, len, this.context.sampleRate);
if (!isMultidimensional && channels === 1){
array = [array];
}
for (var c = 0; c < channels; c++){
if (this.isFunction(buffer.copyToChannel)){
buffer.copyToChannel(array[c], c);
} else {
var channel = buffer.getChannelData(c);
var channelArray = array[c];
for (var i = 0; i < channelArray.length; i++){
channel[i] = channelArray[i];
}
}
}
this._buffer = buffer;
return this;
};
/**
* Get the buffer as an array. Single channel buffers will return a 1-dimensional
* Float32Array, and multichannel buffers will return multidimensional arrays.
* @param {Number=} channel Optionally only copy a single channel from the array.
* @return {Array}
*/
Tone.Buffer.prototype.toArray = function(channel){
if (this.isNumber(channel)){
return this._buffer.getChannelData(channel);
} else {
var ret = [];
for (var c = 0; c < this.numberOfChannels; c++){
2016-09-09 01:20:49 +00:00
ret[c] = new Float32Array(this.length);
if (this.isFunction(this._buffer.copyFromChannel)){
this._buffer.copyFromChannel(ret[c], c);
} else {
2016-09-08 14:29:41 +00:00
var channelData = this._buffer.getChannelData(c);
var retArray = ret[c];
2016-09-08 14:29:41 +00:00
for (var i = 0; i < channelData.length; i++){
retArray[i] = channelData[i];
}
}
}
if (ret.length === 1){
return ret[0];
} else {
return ret;
}
}
};
/**
* Cut a subsection of the array and return a buffer of the
* subsection. Does not modify the original buffer
* @param {Time} start The time to start the slice
* @param {Time=} end The end time to slice. If none is given
* will default to the end of the buffer
* @return {Tone.Buffer} this
*/
Tone.Buffer.prototype.slice = function(start, end){
end = this.defaultArg(end, this.duration);
var startSamples = Math.floor(this.context.sampleRate * this.toSeconds(start));
var endSamples = Math.floor(this.context.sampleRate * this.toSeconds(end));
2016-09-08 14:29:41 +00:00
var replacement = [];
for (var i = 0; i < this.numberOfChannels; i++){
replacement[i] = this.toArray(i).slice(startSamples, endSamples);
}
var retBuffer = new Tone.Buffer().fromArray(replacement);
return retBuffer;
};
/**
* Reverse the buffer.
* @private
* @return {Tone.Buffer} this
*/
Tone.Buffer.prototype._reverse = function(){
if (this.loaded){
for (var i = 0; i < this._buffer.numberOfChannels; i++){
Array.prototype.reverse.call(this._buffer.getChannelData(i));
}
}
return this;
};
2015-03-26 14:51:44 +00:00
/**
2015-06-14 00:52:51 +00:00
* Reverse the buffer.
2015-03-26 14:51:44 +00:00
* @memberOf Tone.Buffer#
2016-08-16 19:26:51 +00:00
* @type {Boolean}
2015-03-26 14:51:44 +00:00
* @name reverse
*/
Object.defineProperty(Tone.Buffer.prototype, "reverse", {
get : function(){
return this._reversed;
},
set : function(rev){
if (this._reversed !== rev){
this._reversed = rev;
this._reverse();
}
},
});
///////////////////////////////////////////////////////////////////////////
// STATIC METHODS
///////////////////////////////////////////////////////////////////////////
//statically inherits Emitter methods
Tone.Emitter.mixin(Tone.Buffer);
/**
* the static queue for all of the xhr requests
* @type {Array}
* @private
*/
Tone.Buffer._downloadQueue = [];
/**
* the total number of downloads
2016-08-16 19:26:51 +00:00
* @type {Number}
* @private
*/
Tone.Buffer._currentDownloads = 0;
2014-10-21 18:44:02 +00:00
/**
* A path which is prefixed before every url.
* @type {String}
* @static
*/
Tone.Buffer.baseUrl = "";
2014-10-21 18:44:02 +00:00
/**
* Loads a url using XMLHttpRequest.
* @param {String} url
* @param {Function} onload
* @param {Function} onerror
* @param {Function} onprogress
* @return {XMLHttpRequest}
2014-10-21 18:44:02 +00:00
*/
Tone.Buffer.load = function(url, onload, onerror){
//defaults
onload = onload || Tone.noOp;
var errCallback = function(e){
Tone.Buffer._currentDownloads--;
if (onerror){
onerror(e);
} else {
throw new Error(e);
}
};
2014-10-21 18:44:02 +00:00
var request = new XMLHttpRequest();
request.open("GET", Tone.Buffer.baseUrl + url, true);
2014-10-21 18:44:02 +00:00
request.responseType = "arraybuffer";
Tone.Buffer._currentDownloads++;
Tone.Buffer._downloadQueue.push(request);
request.addEventListener("load", function(){
2016-08-16 19:26:51 +00:00
if (request.status === 200){
Tone.context.decodeAudioData(request.response, function(buff) {
request.progress = 1;
onload(buff);
Tone.Buffer._currentDownloads--;
if (Tone.Buffer._currentDownloads === 0){
// clear the downloads
Tone.Buffer._downloadQueue = [];
//emit the event at the end
Tone.Buffer.emit("load");
2016-08-16 19:26:51 +00:00
}
}, function(){
errCallback("Tone.Buffer: could not decode audio data: "+url);
});
2016-08-16 19:26:51 +00:00
} else {
errCallback("Tone.Buffer: could not locate file: "+url);
2016-08-16 19:26:51 +00:00
}
});
request.addEventListener("error", errCallback);
request.addEventListener("progress", function(event){
if (event.lengthComputable){
request.progress = event.loaded / event.total;
//calculate the progress
var totalProgress = 0;
for (var i = 0; i < Tone.Buffer._downloadQueue.length; i++){
totalProgress += Tone.Buffer._downloadQueue[i].progress;
}
Tone.Buffer.emit("progress", totalProgress / Tone.Buffer._downloadQueue.length);
}
});
2014-10-21 18:44:02 +00:00
request.send();
return request;
};
2016-07-07 03:09:56 +00:00
/**
* Checks a url's extension to see if the current browser can play that file type.
* @param {String} url The url/extension to test
* @return {Boolean} If the file extension can be played
* @static
* @example
* Tone.Buffer.supportsType("wav"); //returns true
* Tone.Buffer.supportsType("path/to/file.wav"); //returns true
*/
Tone.Buffer.supportsType = function(url){
var extension = url.split(".");
extension = extension[extension.length - 1];
var response = document.createElement("audio").canPlayType("audio/"+extension);
return response !== "";
};
2014-10-21 18:44:02 +00:00
return Tone.Buffer;
});