2018-08-27 02:29:17 +00:00
|
|
|
define(["../core/Tone", "../source/Player", "../component/Volume", "../core/AudioNode"], function(Tone){
|
2017-05-22 03:40:20 +00:00
|
|
|
|
|
|
|
"use strict";
|
2017-08-27 21:50:31 +00:00
|
|
|
|
2017-05-22 03:40:20 +00:00
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* @class Tone.Players combines multiple [Tone.Player](Player) objects.
|
|
|
|
*
|
2017-05-22 03:40:20 +00:00
|
|
|
* @constructor
|
2017-08-27 21:50:31 +00:00
|
|
|
* @extends {Tone.AudioNode}
|
2017-05-22 03:40:20 +00:00
|
|
|
* @param {Object} urls An object mapping a name to a url.
|
2017-09-29 14:26:50 +00:00
|
|
|
* @param {function=} onload The function to invoke when all buffers are loaded.
|
2017-05-22 03:40:20 +00:00
|
|
|
*/
|
|
|
|
Tone.Players = function(urls){
|
|
|
|
|
|
|
|
var args = Array.prototype.slice.call(arguments);
|
|
|
|
args.shift();
|
|
|
|
var options = Tone.defaults(args, ["onload"], Tone.Players);
|
2018-07-30 17:30:21 +00:00
|
|
|
Tone.AudioNode.call(this, options);
|
2017-05-22 03:40:20 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The output volume node
|
|
|
|
* @type {Tone.Volume}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._volume = this.output = new Tone.Volume(options.volume);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The volume of the output in decibels.
|
|
|
|
* @type {Decibels}
|
|
|
|
* @signal
|
|
|
|
* @example
|
|
|
|
* source.volume.value = -6;
|
|
|
|
*/
|
|
|
|
this.volume = this._volume.volume;
|
|
|
|
this._readOnly("volume");
|
|
|
|
|
|
|
|
//make the output explicitly stereo
|
|
|
|
this._volume.output.output.channelCount = 2;
|
|
|
|
this._volume.output.output.channelCountMode = "explicit";
|
|
|
|
//mute initially
|
|
|
|
this.mute = options.mute;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The container of all of the players
|
|
|
|
* @type {Object}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._players = {};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The loading count
|
|
|
|
* @type {Number}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._loadingCount = 0;
|
|
|
|
|
2017-06-19 19:18:50 +00:00
|
|
|
/**
|
|
|
|
* private holder of the fadeIn time
|
|
|
|
* @type {Time}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._fadeIn = options.fadeIn;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* private holder of the fadeOut time
|
|
|
|
* @type {Time}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
this._fadeOut = options.fadeOut;
|
|
|
|
|
|
|
|
//add all of the players
|
2017-05-22 03:40:20 +00:00
|
|
|
for (var name in urls){
|
|
|
|
this._loadingCount++;
|
|
|
|
this.add(name, urls[name], this._bufferLoaded.bind(this, options.onload));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.extend(Tone.Players, Tone.AudioNode);
|
2017-05-22 03:40:20 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The default values
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Tone.Players.defaults = {
|
|
|
|
"volume" : 0,
|
|
|
|
"mute" : false,
|
2017-06-19 19:18:50 +00:00
|
|
|
"onload" : Tone.noOp,
|
|
|
|
"fadeIn" : 0,
|
|
|
|
"fadeOut" : 0
|
2017-05-22 03:40:20 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A buffer was loaded. decrement the counter.
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {Function} callback
|
2017-05-22 03:40:20 +00:00
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype._bufferLoaded = function(callback){
|
|
|
|
this._loadingCount--;
|
|
|
|
if (this._loadingCount === 0 && callback){
|
|
|
|
callback(this);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Mute the output.
|
2017-05-22 03:40:20 +00:00
|
|
|
* @memberOf Tone.Source#
|
|
|
|
* @type {boolean}
|
|
|
|
* @name mute
|
|
|
|
* @example
|
|
|
|
* //mute the output
|
|
|
|
* source.mute = true;
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Players.prototype, "mute", {
|
|
|
|
get : function(){
|
|
|
|
return this._volume.mute;
|
2017-08-27 21:50:31 +00:00
|
|
|
},
|
2017-05-22 03:40:20 +00:00
|
|
|
set : function(mute){
|
|
|
|
this._volume.mute = mute;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-06-19 19:18:50 +00:00
|
|
|
/**
|
|
|
|
* The fadeIn time of the amplitude envelope.
|
|
|
|
* @memberOf Tone.Source#
|
|
|
|
* @type {Time}
|
|
|
|
* @name fadeIn
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Players.prototype, "fadeIn", {
|
|
|
|
get : function(){
|
|
|
|
return this._fadeIn;
|
2017-08-27 21:50:31 +00:00
|
|
|
},
|
2017-06-19 19:18:50 +00:00
|
|
|
set : function(fadeIn){
|
|
|
|
this._fadeIn = fadeIn;
|
|
|
|
this._forEach(function(player){
|
|
|
|
player.fadeIn = fadeIn;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The fadeOut time of the amplitude envelope.
|
|
|
|
* @memberOf Tone.Source#
|
|
|
|
* @type {Time}
|
|
|
|
* @name fadeOut
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Players.prototype, "fadeOut", {
|
|
|
|
get : function(){
|
|
|
|
return this._fadeOut;
|
2017-08-27 21:50:31 +00:00
|
|
|
},
|
2017-06-19 19:18:50 +00:00
|
|
|
set : function(fadeOut){
|
|
|
|
this._fadeOut = fadeOut;
|
|
|
|
this._forEach(function(player){
|
|
|
|
player.fadeOut = fadeOut;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-05-22 03:40:20 +00:00
|
|
|
/**
|
|
|
|
* The state of the players object. Returns "started" if any of the players are playing.
|
|
|
|
* @memberOf Tone.Players#
|
|
|
|
* @type {String}
|
|
|
|
* @name state
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Players.prototype, "state", {
|
|
|
|
get : function(){
|
|
|
|
var playing = false;
|
|
|
|
this._forEach(function(player){
|
|
|
|
playing = playing || player.state === Tone.State.Started;
|
|
|
|
});
|
|
|
|
return playing ? Tone.State.Started : Tone.State.Stopped;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if the buffers object has a buffer by that name.
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {String|Number} name The key or index of the
|
2017-05-22 03:40:20 +00:00
|
|
|
* buffer.
|
|
|
|
* @return {Boolean}
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype.has = function(name){
|
|
|
|
return this._players.hasOwnProperty(name);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2017-08-27 21:50:31 +00:00
|
|
|
* Get a player by name.
|
|
|
|
* @param {String} name The players name as defined in
|
|
|
|
* the constructor object or `add` method.
|
2017-05-22 03:40:20 +00:00
|
|
|
* @return {Tone.Player}
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype.get = function(name){
|
|
|
|
if (this.has(name)){
|
|
|
|
return this._players[name];
|
|
|
|
} else {
|
|
|
|
throw new Error("Tone.Players: no player named "+name);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Iterate over all of the players
|
|
|
|
* @param {Function} callback
|
|
|
|
* @return {Tone.Players} this
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype._forEach = function(callback){
|
|
|
|
for (var playerName in this._players){
|
|
|
|
callback(this._players[playerName], playerName);
|
|
|
|
}
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If all the buffers are loaded or not
|
|
|
|
* @memberOf Tone.Players#
|
|
|
|
* @type {Boolean}
|
|
|
|
* @name loaded
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Players.prototype, "loaded", {
|
|
|
|
get : function(){
|
|
|
|
var isLoaded = true;
|
|
|
|
this._forEach(function(player){
|
|
|
|
isLoaded = isLoaded && player.loaded;
|
|
|
|
});
|
|
|
|
return isLoaded;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add a player by name and url to the Players
|
|
|
|
* @param {String} name A unique name to give the player
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {String|Tone.Buffer|Audiobuffer} url Either the url of the bufer,
|
2017-05-22 03:40:20 +00:00
|
|
|
* or a buffer which will be added
|
|
|
|
* with the given name.
|
2017-08-27 21:50:31 +00:00
|
|
|
* @param {Function=} callback The callback to invoke
|
2017-05-22 03:40:20 +00:00
|
|
|
* when the url is loaded.
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype.add = function(name, url, callback){
|
|
|
|
this._players[name] = new Tone.Player(url, callback).connect(this.output);
|
2017-06-19 19:18:50 +00:00
|
|
|
this._players[name].fadeIn = this._fadeIn;
|
|
|
|
this._players[name].fadeOut = this._fadeOut;
|
2017-05-22 03:40:20 +00:00
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Stop all of the players at the given time
|
|
|
|
* @param {Time} time The time to stop all of the players.
|
|
|
|
* @return {Tone.Players} this
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype.stopAll = function(time){
|
|
|
|
this._forEach(function(player){
|
|
|
|
player.stop(time);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dispose and disconnect.
|
|
|
|
* @return {Tone.Players} this
|
|
|
|
*/
|
|
|
|
Tone.Players.prototype.dispose = function(){
|
2017-08-27 21:50:31 +00:00
|
|
|
Tone.AudioNode.prototype.dispose.call(this);
|
2017-05-22 03:40:20 +00:00
|
|
|
this._volume.dispose();
|
|
|
|
this._volume = null;
|
|
|
|
this._writable("volume");
|
|
|
|
this.volume = null;
|
|
|
|
this.output = null;
|
|
|
|
this._forEach(function(player){
|
|
|
|
player.dispose();
|
|
|
|
});
|
|
|
|
this._players = null;
|
|
|
|
return this;
|
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.Players;
|
|
|
|
});
|