final patch for #510 -- new datastructure suggested and implemented by james for the events entry cache. speeds up all event operations

git-svn-id: http://svn.openlayers.org/trunk/openlayers@2896 dc9f47b5-9b13-0410-9fdd-eb0c1a62fdaf
This commit is contained in:
euzuro
2007-03-26 17:33:35 +00:00
parent d82108c2d7
commit 87977f7bfe
3 changed files with 58 additions and 28 deletions

View File

@@ -13,6 +13,7 @@ Tim Schaub
Christopher Schmidt Christopher Schmidt
Cameron Shorter Cameron Shorter
Paul Spencer Paul Spencer
James Stembridge
Erik Uzureau Erik Uzureau
Bill Woodall Bill Woodall
Steve Woodbridge Steve Woodbridge

View File

@@ -55,23 +55,40 @@ OpenLayers.Event = {
return element; return element;
}, },
/** A hashtable cach of the event observers, keyed by element.id
*
* @type Object
*/
observers: false, observers: false,
_observeAndCache: function(element, name, observer, useCapture) { _observeAndCache: function(element, name, observer, useCapture) {
if (!this.observers) this.observers = []; if (!this.observers) this.observers = new Object();
//if there is not yet a hash entry for this element, add one
if (!this.observers[element.id]) {
this.observers[element.id] = new Array();
}
//add a new observer to this element's list
this.observers[element.id].push([name, observer, useCapture]);
//add the actual browser event listener
if (element.addEventListener) { if (element.addEventListener) {
this.observers.push([element, name, observer, useCapture]);
element.addEventListener(name, observer, useCapture); element.addEventListener(name, observer, useCapture);
} else if (element.attachEvent) { } else if (element.attachEvent) {
this.observers.push([element, name, observer, useCapture]);
element.attachEvent('on' + name, observer); element.attachEvent('on' + name, observer);
} }
}, },
unloadCache: function() { unloadCache: function() {
if (!OpenLayers.Event.observers) return; if (!OpenLayers.Event.observers) return;
for (var i = 0; i < OpenLayers.Event.observers.length; i++) { for (var elementId in OpenLayers.Event.observers) {
OpenLayers.Event.stopObserving.apply(this, OpenLayers.Event.observers[i]); var elementObservers = OpenLayers.Event.observers[elementId];
if (elementObservers) {
for (var i = 0; i < elementObservers.length; i++) {
var args = new Array(elementId).concat(elementObservers[i]);
OpenLayers.Event.stopObserving.apply(this, args);
}
}
} }
OpenLayers.Event.observers = false; OpenLayers.Event.observers = false;
}, },
@@ -90,6 +107,8 @@ OpenLayers.Event = {
stopObserving: function(elementParam, name, observer, useCapture) { stopObserving: function(elementParam, name, observer, useCapture) {
var element = OpenLayers.Util.getElement(elementParam); var element = OpenLayers.Util.getElement(elementParam);
if (!element) return;
useCapture = useCapture || false; useCapture = useCapture || false;
if (name == 'keypress' && if (name == 'keypress' &&
@@ -97,15 +116,19 @@ OpenLayers.Event = {
|| element.detachEvent)) || element.detachEvent))
name = 'keydown'; name = 'keydown';
// find entry in this.observers cache array and remove it // find element's entry in this.observers cache and remove it
var observers = OpenLayers.Event.observers; var elementObservers = OpenLayers.Event.observers[element.id];
var entry = [element, name, observer, useCapture]; if (elementObservers) {
for (var i = 0; i < observers.length; i++) { var entry = [name, observer, useCapture];
//compare all 4 elements of entry with observer // find the specific event type in the element's list
for (var i = 0; i < elementObservers.length; i++) {
var cacheEntry = elementObservers[i];
//compare all 3 elements of entry with observer
var sameEntry = true; var sameEntry = true;
for (var j = 0; j < entry.length; j++) { for (var j = 0; j < entry.length; j++) {
if (entry[j] != observers[i][j]) { if (entry[j] != cacheEntry[j]) {
sameEntry = false; sameEntry = false;
break; break;
} }
@@ -113,12 +136,17 @@ OpenLayers.Event = {
//if we've found it, remove it from the observers array //if we've found it, remove it from the observers array
if (sameEntry) { if (sameEntry) {
observers.splice(i--, 1); elementObservers.splice(i--, 1);
if (elementObservers.length == 0) {
OpenLayers.Event.observers[element.id] = null;
}
break; break;
} }
} }
}
if (element && element.removeEventListener) { //actually remove the event listener from browser
if (element.removeEventListener) {
element.removeEventListener(name, observer, useCapture); element.removeEventListener(name, observer, useCapture);
} else if (element && element.detachEvent) { } else if (element && element.detachEvent) {
element.detachEvent('on' + name, observer); element.detachEvent('on' + name, observer);

View File

@@ -204,18 +204,19 @@
function test_05_Event_destroy (t) { function test_05_Event_destroy (t) {
t.plan(2); t.plan(2);
var start = OpenLayers.Event.observers.length;
var div = OpenLayers.Util.getElement('test'); var div = OpenLayers.Util.getElement('test');
var obj = {}; var obj = {};
var events = new OpenLayers.Events(obj, div); var events = new OpenLayers.Events(obj, div);
// +1 because of blocking dragstart in attachToElement() // +1 because of blocking dragstart in attachToElement()
t.eq(OpenLayers.Event.observers.length, t.eq(OpenLayers.Event.observers["test"].length,
start + OpenLayers.Events.prototype.BROWSER_EVENTS.length + 1, OpenLayers.Events.prototype.BROWSER_EVENTS.length + 1,
"construction increases the number of event observers"); "construction creates new arrayin hash, registers appropriate events");
events.destroy(); events.destroy();
events = null; events = null;
t.eq(OpenLayers.Event.observers.length, start, t.eq(OpenLayers.Event.observers["test"], null,
"destruction restores the number of event observers"); "destruction removes the event observer from hash");
} }
// --> // -->