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; });