Files
simple-bookreader/client/islets/core/y-event-emitter/y-event-emitter.js
Oleg Mokhov f3546ef3a5 Release
2015-06-20 14:48:34 +05:00

212 lines
6.8 KiB
JavaScript

modules.define(
'y-event-emitter',
['inherit'],
function (provide, inherit) {
var slice = Array.prototype.slice;
/**
* @name YEventEmitter
*/
var YEventEmitter = inherit({
/**
* Добавляет обработчик события.
*
* @param {String} event
* @param {Function} callback
* @param {Object} [context]
* @returns {YEventEmitter}
*/
on: function (event, callback, context) {
if (typeof callback !== 'function') {
throw new TypeError('callback must be a function');
}
if (!this._events) {
this._events = {};
}
var listener = {
callback: callback,
context: context
};
var listeners = this._events[event];
if (listeners) {
listeners.push(listener);
} else {
this._events[event] = [listener];
this._onAddEvent(event);
}
return this;
},
/**
* Добавляет обработчик события, который исполнится только 1 раз, затем удалится.
*
* @param {String} event
* @param {Function} callback
* @param {Object} [context]
* @returns {YEventEmitter}
*/
once: function (event, callback, context) {
if (typeof callback !== 'function') {
throw new TypeError('callback must be a function');
}
var _this = this;
function once() {
_this.off(event, once, context);
callback.apply(context, arguments);
}
// Сохраняем ссылку на оригинальный колбэк. Благодаря этому можно удалить колбэк `once`,
// используя оригинальный колбэк в методе `off()`.
once._callback = callback;
this.on(event, once, context);
return this;
},
/**
* Удаляет обработчик события.
*
* @param {String} event
* @param {Function} callback
* @param {Object} [context]
* @returns {YEventEmitter}
*/
off: function (event, callback, context) {
if (typeof callback !== 'function') {
throw new TypeError('callback must be a function');
}
if (!this._events) {
return this;
}
var listeners = this._events[event];
if (!listeners) {
return this;
}
var len = listeners.length;
for (var i = 0; i < len; i++) {
var listener = listeners[i];
var cb = listener.callback;
if ((cb === callback || cb._callback === callback) && listener.context === context) {
if (len === 1) {
delete this._events[event];
this._onRemoveEvent(event);
} else {
listeners.splice(i, 1);
}
break;
}
}
return this;
},
/**
* Удаляет все обработчики всех событий или все обработчики переданного события `event`.
*
* @param {String} [event]
* @returns {YEventEmitter}
*/
offAll: function (event) {
if (this._events) {
if (event) {
if (this._events[event]) {
delete this._events[event];
this._onRemoveEvent(event);
}
} else {
for (event in this._events) {
if (this._events.hasOwnProperty(event)) {
this._onRemoveEvent(event);
}
}
delete this._events;
}
}
return this;
},
/**
* Исполняет все обработчики события `event`.
*
* @param {String} event
* @param {...*} [args] Аргументы, которые будут переданы в обработчики события.
* @returns {YEventEmitter}
*/
emit: function (event) {
if (!this._events) {
return this;
}
var listeners = this._events[event];
if (!listeners) {
return this;
}
// Копируем массив обработчиков, чтобы добавление/удаление обработчиков внутри колбэков не оказывало
// влияния в цикле.
var listenersCopy = listeners.slice(0);
var len = listenersCopy.length;
var listener;
var i = -1;
switch (arguments.length) {
// Оптимизируем наиболее частые случаи.
case 1:
while (++i < len) {
listener = listenersCopy[i];
listener.callback.call(listener.context);
}
break;
case 2:
while (++i < len) {
listener = listenersCopy[i];
listener.callback.call(listener.context, arguments[1]);
}
break;
case 3:
while (++i < len) {
listener = listenersCopy[i];
listener.callback.call(listener.context, arguments[1], arguments[2]);
}
break;
default:
var args = slice.call(arguments, 1);
while (++i < len) {
listener = listenersCopy[i];
listener.callback.apply(listener.context, args);
}
}
return this;
},
/**
* Вызывается когда было добавлено новое событие.
*
* @protected
* @param {String} event
*/
_onAddEvent: function () {},
/**
* Вызывается когда все обработчики события были удалены.
*
* @protected
* @param {String} event
*/
_onRemoveEvent: function () {}
});
provide(YEventEmitter);
});