h5p-php-library/js/h5p-event-dispatcher.js

225 lines
5.7 KiB
JavaScript
Raw Permalink Normal View History

/** @namespace H5P */
var H5P = H5P || {};
2015-02-16 14:22:54 +01:00
/**
* The Event class for the EventDispatcher
* @class
*/
2015-03-22 11:17:58 +01:00
H5P.Event = function(type, data, extras) {
2015-02-11 15:56:35 +01:00
this.type = type;
this.data = data;
2015-03-22 20:37:10 +01:00
var bubbles = false;
2015-04-07 19:33:21 +02:00
// Is this an external event?
var external = false;
// Is this event scheduled to be sent externally?
var scheduledForExternal = false;
2015-03-22 11:17:58 +01:00
if (extras === undefined) {
extras = {};
}
2015-03-22 20:37:10 +01:00
if (extras.bubbles === true) {
bubbles = true;
2015-03-22 11:17:58 +01:00
}
2015-04-07 19:33:21 +02:00
if (extras.external === true) {
external = true;
}
/**
* Prevent this event from bubbling up to parent
*
* @returns {undefined}
*/
2015-03-21 16:45:38 +01:00
this.preventBubbling = function() {
bubbles = false;
};
2015-04-07 19:33:21 +02:00
/**
* Get bubbling status
*
* @returns {Boolean} - true if bubbling false otherwise
*/
2015-03-21 16:45:38 +01:00
this.getBubbles = function() {
return bubbles;
};
2015-04-07 19:33:21 +02:00
/**
* Try to schedule an event for externalDispatcher
*
* @returns {Boolean}
* - true if external and not already scheduled
* - false otherwise
*/
this.scheduleForExternal = function() {
if (external && !scheduledForExternal) {
scheduledForExternal = true;
return true;
}
return false;
};
};
H5P.EventDispatcher = (function () {
2015-02-20 11:41:09 +01:00
/**
* The base of the event system.
* Inherit this class if you want your H5P to dispatch events.
* @class
*/
function EventDispatcher() {
var self = this;
/**
* Keep track of listeners for each event.
* @private
* @type {Object}
*/
var triggers = {};
/**
* Add new event listener.
*
* @public
2015-02-16 14:22:54 +01:00
* @throws {TypeError} listener - Must be a function
* @param {String} type - Event type
* @param {Function} listener - Event listener
2015-02-16 14:26:09 +01:00
* @param {Function} thisArg - Optionally specify the this value when calling listener.
*/
this.on = function (type, listener, thisArg) {
2015-02-16 14:22:54 +01:00
if (thisArg === undefined) {
thisArg = self;
2015-02-11 17:31:01 +01:00
}
if (typeof listener !== 'function') {
throw TypeError('listener must be a function');
}
// Trigger event before adding to avoid recursion
2015-02-11 17:31:01 +01:00
self.trigger('newListener', {'type': type, 'listener': listener});
if (!triggers[type]) {
// First
2015-02-16 14:22:54 +01:00
triggers[type] = [{'listener': listener, 'thisArg': thisArg}];
}
else {
// Append
2015-02-16 14:22:54 +01:00
triggers[type].push({'listener': listener, 'thisArg': thisArg});
}
};
/**
* Add new event listener that will be fired only once.
*
* @public
2015-02-16 14:26:09 +01:00
* @throws {TypeError} listener - must be a function
* @param {String} type - Event type
* @param {Function} listener - Event listener
* @param {Function} thisArg - Optionally specify the this value when calling listener.
*/
this.once = function (type, listener, thisArg) {
2015-02-16 14:22:54 +01:00
if (thisArg === undefined) {
thisArg = self;
2015-02-11 17:31:01 +01:00
}
if (!(listener instanceof Function)) {
throw TypeError('listener must be a function');
}
2015-02-11 17:31:01 +01:00
var once = function (event) {
self.off(event, once);
2015-02-16 14:22:54 +01:00
listener.apply(thisArg, event);
};
2015-02-16 14:22:54 +01:00
self.on(type, once, thisArg);
};
/**
* Remove event listener.
* If no listener is specified, all listeners will be removed.
*
* @public
2015-02-16 14:26:09 +01:00
* @throws {TypeError} listener - must be a function
* @param {String} type - Event type
* @param {Function} listener - Event listener
*/
this.off = function (type, listener) {
if (listener !== undefined && !(listener instanceof Function)) {
throw TypeError('listener must be a function');
}
if (triggers[type] === undefined) {
return;
}
if (listener === undefined) {
// Remove all listeners
delete triggers[type];
self.trigger('removeListener', type);
return;
}
// Find specific listener
for (var i = 0; i < triggers[type].length; i++) {
2015-02-11 17:31:01 +01:00
if (triggers[type][i].listener === listener) {
triggers[type].unshift(i, 1);
2015-02-11 17:31:01 +01:00
self.trigger('removeListener', type, {'listener': listener});
break;
}
}
// Clean up empty arrays
if (!triggers[type].length) {
delete triggers[type];
}
};
/**
* Dispatch event.
*
* @public
2015-02-16 14:26:09 +01:00
* @param {String|Function} event - Event object or event type as string
* @param {mixed} eventData
* Custom event data(used when event type as string is used as first
* argument
*/
2015-04-07 19:33:21 +02:00
this.trigger = function (event, eventData, extras) {
2015-02-11 15:56:35 +01:00
if (event === undefined) {
return;
}
2015-02-11 15:56:35 +01:00
if (typeof event === 'string') {
2015-04-07 19:33:21 +02:00
event = new H5P.Event(event, eventData, extras);
}
2015-02-11 15:56:35 +01:00
else if (eventData !== undefined) {
event.data = eventData;
}
2015-04-07 19:33:21 +02:00
// Check to see if this event should go externally after all triggering and bubbling is done
var scheduledForExternal = event.scheduleForExternal();
if (triggers[event.type] !== undefined) {
// Call all listeners
for (var i = 0; i < triggers[event.type].length; i++) {
triggers[event.type][i].listener.call(triggers[event.type][i].thisArg, event);
}
}
2015-04-07 19:33:21 +02:00
if (triggers['*'] !== undefined) {
// Call all * listeners
for (var i = 0; i < triggers['*'].length; i++) {
triggers['*'][i].listener.call(triggers['*'][i].thisArg, event);
}
}
2015-04-07 19:33:21 +02:00
2015-03-22 20:37:10 +01:00
// Bubble
2015-03-22 11:17:58 +01:00
if (event.getBubbles() && self.parent instanceof H5P.EventDispatcher && typeof self.parent.trigger === 'function') {
2015-03-21 16:45:38 +01:00
self.parent.trigger(event);
2015-04-07 19:33:21 +02:00
}
if (scheduledForExternal) {
H5P.externalDispatcher.trigger(event);
}
};
}
return EventDispatcher;
2015-02-20 11:41:09 +01:00
})();