From 9c1e5045aec4833446c2881de001a5887a357d89 Mon Sep 17 00:00:00 2001 From: Alexander Ebert Date: Thu, 27 Oct 2011 14:17:35 +0200 Subject: [PATCH] DOMNodeInsertedHandler implemented Certain functions perform actions on a specific element collection available at page load, but they miss elements inserted afterwards. Register your class with DOMNodeInsertedHandler and you will be notified to update your collection once a DOM node was inserted. Caution: This handler blocks any DOMNodeInserted event which occurs within 100 ms after the event fired to prevent an infinite loop. --- wcfsetup/install/files/js/WCF.js | 95 +++++++++++++++++++++++++++++++- 1 file changed, 94 insertions(+), 1 deletion(-) diff --git a/wcfsetup/install/files/js/WCF.js b/wcfsetup/install/files/js/WCF.js index 5503d63877..6b768be91b 100644 --- a/wcfsetup/install/files/js/WCF.js +++ b/wcfsetup/install/files/js/WCF.js @@ -1381,6 +1381,17 @@ WCF.Date.Time.prototype = { // re-calculate relative datetime every minute new WCF.PeriodicalExecuter($.proxy(this._refresh, this), 60000); + + // bind dom node inserted listener + WCF.DOMNodeInsertedHandler.addCallback('WCF.Date.Time', $.proxy(this._domNodeInserted, this)); + }, + + /** + * Updates element collection once a DOM node was inserted. + */ + _domNodeInserted: function() { + this.elements = $('time.datetime'); + this._refresh(); }, /** @@ -2785,6 +2796,88 @@ WCF.CloseOverlayHandler = { } }; +/** + * Notifies objects once a DOM node was inserted. + */ +WCF.DOMNodeInsertedHandler = { + /** + * list of callbacks + * @var WCF.Dictionary + */ + _callbacks: new WCF.Dictionary(), + + /** + * prevent infinite loop if a callback manipulates DOM + * @var boolean + */ + _isExecuting: false, + + /** + * indicates that overlay handler is listening to click events on body-tag + * @var boolean + */ + _isListening: false, + + /** + * Adds a new callback. + * + * @param string identifier + * @param object callback + */ + addCallback: function(identifier, callback) { + this._bindListener(); + + if (this._callbacks.isset(identifier)) { + cosole.debug("[WCF.DOMNodeInsertedHandler] identifier '" + identifier + "' is already bound to a callback"); + return false; + } + + this._callbacks.add(identifier, callback); + }, + + /** + * Removes a callback from list. + * + * @param string identifier + */ + removeCallback: function(identifier) { + if (this._callbacks.isset(identifier)) { + this._callbacks.remove(identifier); + } + }, + + /** + * Binds click event handler. + */ + _bindListener: function() { + if (this._isListening) return; + + $(document).bind('DOMNodeInserted', $.proxy(this._executeCallbacks, this)); + + this._isListening = true; + }, + + /** + * Executes callbacks on click. + */ + _executeCallbacks: function(event) { + if (this._isExecuting) return; + + // do not track events fired within the next 100 ms + this._isExecuting = true; + new WCF.PeriodicalExecuter($.proxy(function(pe) { + this._isExecuting = false; + + pe.stop(); + }, this), 100); + + this._callbacks.each(function(pair) { + // execute callback + pair.value(event); + }); + } +}; + /** * Basic implementation for WCF dialogs. */ @@ -2999,7 +3092,7 @@ $.widget('ui.wcfPages', { // language // we use options here instead of language variables, because the paginator is not only usable with pages nextPage: null, - previousPage: null, + previousPage: null }, /** -- 2.20.1