Tone.js/Tone/core/Emitter.js
2017-12-15 12:49:53 -05:00

138 lines
3.8 KiB
JavaScript

define(["Tone/core/Tone"], function (Tone) {
"use strict";
/**
* @class Tone.Emitter gives classes which extend it
* the ability to listen for and emit events.
* Inspiration and reference from Jerome Etienne's [MicroEvent](https://github.com/jeromeetienne/microevent.js).
* MIT (c) 2011 Jerome Etienne.
*
* @extends {Tone}
*/
Tone.Emitter = function(){
Tone.call(this);
/**
* Contains all of the events.
* @private
* @type {Object}
*/
this._events = {};
};
Tone.extend(Tone.Emitter);
/**
* Bind a callback to a specific event.
* @param {String} event The name of the event to listen for.
* @param {Function} callback The callback to invoke when the
* event is emitted
* @return {Tone.Emitter} this
*/
Tone.Emitter.prototype.on = function(event, callback){
//split the event
var events = event.split(/\W+/);
for (var i = 0; i < events.length; i++){
var eventName = events[i];
if (!this._events.hasOwnProperty(eventName)){
this._events[eventName] = [];
}
this._events[eventName].push(callback);
}
return this;
};
/**
* Bind a callback which is only invoked once
* @param {String} event The name of the event to listen for.
* @param {Function} callback The callback to invoke when the
* event is emitted
* @return {Tone.Emitter} this
*/
Tone.Emitter.prototype.once = function(event, callback){
var boundCallback = function(){
//invoke the callback
callback.apply(this, arguments);
this.off(event, boundCallback);
}.bind(this);
this.on(event, boundCallback);
return this;
};
/**
* Remove the event listener.
* @param {String} event The event to stop listening to.
* @param {Function=} callback The callback which was bound to
* the event with Tone.Emitter.on.
* If no callback is given, all callbacks
* events are removed.
* @return {Tone.Emitter} this
*/
Tone.Emitter.prototype.off = function(event, callback){
var events = event.split(/\W+/);
for (var ev = 0; ev < events.length; ev++){
event = events[ev];
if (this._events.hasOwnProperty(event)){
if (Tone.isUndef(callback)){
this._events[event] = [];
} else {
var eventList = this._events[event];
for (var i = 0; i < eventList.length; i++){
if (eventList[i] === callback){
eventList.splice(i, 1);
}
}
}
}
}
return this;
};
/**
* Invoke all of the callbacks bound to the event
* with any arguments passed in.
* @param {String} event The name of the event.
* @param {*} args... The arguments to pass to the functions listening.
* @return {Tone.Emitter} this
*/
Tone.Emitter.prototype.emit = function(event){
if (this._events){
var args = Array.apply(null, arguments).slice(1);
if (this._events.hasOwnProperty(event)){
var eventList = this._events[event].slice(0);
for (var i = 0, len = eventList.length; i < len; i++){
eventList[i].apply(this, args);
}
}
}
return this;
};
/**
* Add Emitter functions (on/off/emit) to the object
* @param {Object|Function} object The object or class to extend.
* @returns {Tone.Emitter}
*/
Tone.Emitter.mixin = function(object){
var functions = ["on", "once", "off", "emit"];
object._events = {};
for (var i = 0; i < functions.length; i++){
var func = functions[i];
var emitterFunc = Tone.Emitter.prototype[func];
object[func] = emitterFunc;
}
return Tone.Emitter;
};
/**
* Clean up
* @return {Tone.Emitter} this
*/
Tone.Emitter.prototype.dispose = function(){
Tone.prototype.dispose.call(this);
this._events = null;
return this;
};
return Tone.Emitter;
});