Make sure that jQuery.data( elem ) always returns a data object, fixes #5971. Improve the performance of .bind() and .unbind(), fixes #5972.

This commit is contained in:
jeresig 2010-01-28 14:12:44 -05:00
parent 942f8f7f75
commit eed69eccc5
3 changed files with 92 additions and 58 deletions

View file

@ -1,5 +1,4 @@
var expando = "jQuery" + now(), uuid = 0, windowData = {}; var expando = "jQuery" + now(), uuid = 0, windowData = {};
var emptyObject = {};
jQuery.extend({ jQuery.extend({
cache: {}, cache: {},
@ -25,8 +24,7 @@ jQuery.extend({
var id = elem[ expando ], cache = jQuery.cache, thisCache; var id = elem[ expando ], cache = jQuery.cache, thisCache;
// Handle the case where there's no name immediately if ( !id && typeof name === "string" && data === undefined ) {
if ( !name && !id ) {
return null; return null;
} }
@ -40,17 +38,16 @@ jQuery.extend({
if ( typeof name === "object" ) { if ( typeof name === "object" ) {
elem[ expando ] = id; elem[ expando ] = id;
thisCache = cache[ id ] = jQuery.extend(true, {}, name); thisCache = cache[ id ] = jQuery.extend(true, {}, name);
} else if ( cache[ id ] ) {
thisCache = cache[ id ]; } else if ( !cache[ id ] ) {
} else if ( typeof data === "undefined" ) { elem[ expando ] = id;
thisCache = emptyObject; cache[ id ] = {};
} else {
thisCache = cache[ id ] = {};
} }
thisCache = cache[ id ];
// Prevent overriding the named cache with undefined values // Prevent overriding the named cache with undefined values
if ( data !== undefined ) { if ( data !== undefined ) {
elem[ expando ] = id;
thisCache[ name ] = data; thisCache[ name ] = data;
} }

View file

@ -42,8 +42,16 @@ jQuery.event = {
} }
// Init the element's event structure // Init the element's event structure
var events = jQuery.data( elem, "events" ) || jQuery.data( elem, "events", {} ), var elemData = jQuery.data( elem );
handle = jQuery.data( elem, "handle" ), eventHandle;
// If no elemData is found then we must be trying to bind to one of the
// banned noData elements
if ( !elemData ) {
return;
}
var events = elemData.events || (elemData.events = {}),
handle = elemData.handle, eventHandle;
if ( !handle ) { if ( !handle ) {
eventHandle = function() { eventHandle = function() {
@ -54,13 +62,7 @@ jQuery.event = {
undefined; undefined;
}; };
handle = jQuery.data( elem, "handle", eventHandle ); handle = elemData.handle = eventHandle;
}
// If no handle is found then we must be trying to bind to one of the
// banned noData elements
if ( !handle ) {
return;
} }
// Add elem as a property of the handle function // Add elem as a property of the handle function
@ -70,15 +72,11 @@ jQuery.event = {
// Handle multiple events separated by a space // Handle multiple events separated by a space
// jQuery(...).bind("mouseover mouseout", fn); // jQuery(...).bind("mouseover mouseout", fn);
types = types.split( /\s+/ ); types = types.split(" ");
var type, i = 0; var type, i = 0, namespaces;
while ( (type = types[ i++ ]) ) { while ( (type = types[ i++ ]) ) {
// Namespaced event handlers
var namespaces = type.split(".");
type = namespaces.shift();
if ( i > 1 ) { if ( i > 1 ) {
handler = jQuery.proxy( handler ); handler = jQuery.proxy( handler );
@ -87,8 +85,17 @@ jQuery.event = {
} }
} }
// Namespaced event handlers
if ( type.indexOf(".") > -1 ) {
namespaces = type.split(".");
type = namespaces.shift();
handler.type = namespaces.slice(0).sort().join("."); handler.type = namespaces.slice(0).sort().join(".");
} else {
namespaces = [];
handler.type = "";
}
// Get the current list of functions bound to this event // Get the current list of functions bound to this event
var handlers = events[ type ], var handlers = events[ type ],
special = this.special[ type ] || {}; special = this.special[ type ] || {};
@ -104,6 +111,7 @@ jQuery.event = {
// Bind the global event handler to the element // Bind the global event handler to the element
if ( elem.addEventListener ) { if ( elem.addEventListener ) {
elem.addEventListener( type, handle, false ); elem.addEventListener( type, handle, false );
} else if ( elem.attachEvent ) { } else if ( elem.attachEvent ) {
elem.attachEvent( "on" + type, handle ); elem.attachEvent( "on" + type, handle );
} }
@ -140,7 +148,13 @@ jQuery.event = {
return; return;
} }
var events = jQuery.data( elem, "events" ), ret, type, fn; var elemData = jQuery.data( elem );
if ( !elemData ) {
return;
}
var events = elemData.events, ret, type, fn;
if ( events ) { if ( events ) {
// Unbind all events for the element // Unbind all events for the element
@ -148,6 +162,7 @@ jQuery.event = {
for ( type in events ) { for ( type in events ) {
this.remove( elem, type + (types || "") ); this.remove( elem, type + (types || "") );
} }
} else { } else {
// types is actually an event object here // types is actually an event object here
if ( types.type ) { if ( types.type ) {
@ -157,16 +172,24 @@ jQuery.event = {
// Handle multiple events separated by a space // Handle multiple events separated by a space
// jQuery(...).unbind("mouseover mouseout", fn); // jQuery(...).unbind("mouseover mouseout", fn);
types = types.split(/\s+/); types = types.split(" ");
var i = 0;
var i = 0, all, namespaces, namespace;
while ( (type = types[ i++ ]) ) { while ( (type = types[ i++ ]) ) {
all = type.indexOf(".") < 0;
namespaces = null;
if ( !all ) {
// Namespaced event handlers // Namespaced event handlers
var namespaces = type.split("."); namespaces = type.split(".");
type = namespaces.shift(); type = namespaces.shift();
var all = !namespaces.length,
cleaned = jQuery.map( namespaces.slice(0).sort(), fcleanup ), namespace = new RegExp("(^|\\.)" +
namespace = new RegExp("(^|\\.)" + cleaned.join("\\.(?:.*\\.)?") + "(\\.|$)"), jQuery.map( namespaces.slice(0).sort(), fcleanup ).join("\\.(?:.*\\.)?") + "(\\.|$)")
special = this.special[ type ] || {}; }
var special = this.special[ type ] || {};
if ( events[ type ] ) { if ( events[ type ] ) {
// remove the given handler for the given type // remove the given handler for the given type
@ -185,21 +208,23 @@ jQuery.event = {
} }
if ( special.remove ) { if ( special.remove ) {
special.remove.call( elem, namespaces, fn); special.remove.call( elem, namespaces || [], fn);
} }
// remove generic event handler if no more handlers exist // remove generic event handler if no more handlers exist
for ( ret in events[ type ] ) { for ( ret in events[ type ] ) {
break; break;
} }
if ( !ret ) { if ( !ret ) {
if ( !special.teardown || special.teardown.call( elem, namespaces ) === false ) { if ( !special.teardown || special.teardown.call( elem, namespaces ) === false ) {
if ( elem.removeEventListener ) { if ( elem.removeEventListener ) {
elem.removeEventListener( type, jQuery.data( elem, "handle" ), false ); elem.removeEventListener( type, elemData.handle, false );
} else if ( elem.detachEvent ) { } else if ( elem.detachEvent ) {
elem.detachEvent( "on" + type, jQuery.data( elem, "handle" ) ); elem.detachEvent( "on" + type, elemData.handle );
} }
} }
ret = null; ret = null;
delete events[ type ]; delete events[ type ];
} }
@ -211,13 +236,19 @@ jQuery.event = {
for ( ret in events ) { for ( ret in events ) {
break; break;
} }
if ( !ret ) { if ( !ret ) {
var handle = jQuery.data( elem, "handle" ); var handle = elemData.handle;
if ( handle ) { if ( handle ) {
handle.elem = null; handle.elem = null;
} }
jQuery.removeData( elem, "events" );
jQuery.removeData( elem, "handle" ); delete elemData.events;
delete elemData.handle;
if ( jQuery.isEmptyObject( elemData ) ) {
jQuery.removeData( elem );
}
} }
} }
}, },
@ -796,11 +827,16 @@ jQuery.each(["bind", "one"], function( i, name ) {
return fn.apply( this, arguments ); return fn.apply( this, arguments );
}) : fn; }) : fn;
return type === "unload" && name !== "one" ? if ( type === "unload" && name !== "one" ) {
this.one( type, data, fn ) : this.one( type, data, fn );
this.each(function() {
jQuery.event.add( this, type, handler, data ); } else {
}); for ( var i = 0, l = this.length; i < l; i++ ) {
jQuery.event.add( this[i], type, handler, data );
}
}
return this;
}; };
}); });
@ -811,12 +847,14 @@ jQuery.fn.extend({
for ( var key in type ) { for ( var key in type ) {
this.unbind(key, type[key]); this.unbind(key, type[key]);
} }
return this;
} else {
for ( var i = 0, l = this.length; i < l; i++ ) {
jQuery.event.remove( this[i], type, fn );
}
} }
return this.each(function() { return this;
jQuery.event.remove( this, type, fn );
});
}, },
trigger: function( type, data ) { trigger: function( type, data ) {
return this.each(function() { return this.each(function() {

View file

@ -1,20 +1,19 @@
module("data"); module("data");
test("expando", function(){ test("expando", function(){
expect(7); expect(6);
equals("expando" in jQuery, true, "jQuery is exposing the expando"); equals("expando" in jQuery, true, "jQuery is exposing the expando");
var obj = {}; var obj = {};
jQuery.data(obj); jQuery.data(obj);
equals( jQuery.expando in obj, false, "jQuery.data did not add an expando to the object" ); equals( jQuery.expando in obj, true, "jQuery.data adds an expando to the object" );
jQuery.data(obj, true);
equals( jQuery.expando in obj, false, "jQuery.data did not add an expando to the object" );
obj = {};
jQuery.data(obj, 'test'); jQuery.data(obj, 'test');
equals( jQuery.expando in obj, false, "jQuery.data did not add an expando to the object" ); equals( jQuery.expando in obj, false, "jQuery.data did not add an expando to the object" );
obj = {};
jQuery.data(obj, "foo", "bar"); jQuery.data(obj, "foo", "bar");
equals( jQuery.expando in obj, true, "jQuery.data added an expando to the object" ); equals( jQuery.expando in obj, true, "jQuery.data added an expando to the object" );