A large refactor of the event handling logic. Data and namespace information is maintained in a separate object now, no longer on the event handler. Proxy functions are no longer needed, as a result. Additionally execution order of the handlers is maintained, fixing #4261, and the execution of handlers is maintained even while they're being removed. Live events will be refactored separately.
This commit is contained in:
parent
8a4b2102ff
commit
e7912805d6
2 changed files with 144 additions and 150 deletions
287
src/event.js
287
src/event.js
|
@ -29,18 +29,6 @@ jQuery.event = {
|
||||||
handler.guid = jQuery.guid++;
|
handler.guid = jQuery.guid++;
|
||||||
}
|
}
|
||||||
|
|
||||||
// if data is passed, bind to handler
|
|
||||||
if ( data !== undefined ) {
|
|
||||||
// Create temporary function pointer to original handler
|
|
||||||
var fn = handler;
|
|
||||||
|
|
||||||
// Create unique handler function, wrapped around original handler
|
|
||||||
handler = jQuery.proxy( fn );
|
|
||||||
|
|
||||||
// Store data in unique handler
|
|
||||||
handler.data = data;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Init the element's event structure
|
// Init the element's event structure
|
||||||
var elemData = jQuery.data( elem );
|
var elemData = jQuery.data( elem );
|
||||||
|
|
||||||
|
@ -50,25 +38,22 @@ jQuery.event = {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var events = elemData.events || (elemData.events = {}),
|
var events = elemData.events = elemData.events || {},
|
||||||
handle = elemData.handle, eventHandle;
|
eventHandle = elemData.handle, eventHandle;
|
||||||
|
|
||||||
if ( !handle ) {
|
if ( !eventHandle ) {
|
||||||
eventHandle = function() {
|
elemData.handle = eventHandle = function() {
|
||||||
// Handle the second event of a trigger and when
|
// Handle the second event of a trigger and when
|
||||||
// an event is called after a page has unloaded
|
// an event is called after a page has unloaded
|
||||||
return typeof jQuery !== "undefined" && !jQuery.event.triggered ?
|
return typeof jQuery !== "undefined" && !jQuery.event.triggered ?
|
||||||
jQuery.event.handle.apply( eventHandle.elem, arguments ) :
|
jQuery.event.handle.apply( eventHandle.elem, arguments ) :
|
||||||
undefined;
|
undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
handle = elemData.handle = eventHandle;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add elem as a property of the handle function
|
// Add elem as a property of the handle function
|
||||||
// This is to prevent a memory leak with non-native
|
// This is to prevent a memory leak with non-native events in IE.
|
||||||
// event in IE.
|
eventHandle.elem = elem;
|
||||||
handle.elem = elem;
|
|
||||||
|
|
||||||
// Handle multiple events separated by a space
|
// Handle multiple events separated by a space
|
||||||
// jQuery(...).bind("mouseover mouseout", fn);
|
// jQuery(...).bind("mouseover mouseout", fn);
|
||||||
|
@ -77,62 +62,56 @@ jQuery.event = {
|
||||||
var type, i = 0, namespaces;
|
var type, i = 0, namespaces;
|
||||||
|
|
||||||
while ( (type = types[ i++ ]) ) {
|
while ( (type = types[ i++ ]) ) {
|
||||||
if ( i > 1 ) {
|
var handleObj = {
|
||||||
handler = jQuery.proxy( handler );
|
handler: handler,
|
||||||
|
data: data,
|
||||||
if ( data !== undefined ) {
|
namespace: "",
|
||||||
handler.data = data;
|
guid: handler.guid
|
||||||
}
|
};
|
||||||
}
|
|
||||||
|
|
||||||
// Namespaced event handlers
|
// Namespaced event handlers
|
||||||
if ( type.indexOf(".") > -1 ) {
|
if ( type.indexOf(".") > -1 ) {
|
||||||
namespaces = type.split(".");
|
namespaces = type.split(".");
|
||||||
type = namespaces.shift();
|
type = namespaces.shift();
|
||||||
handler.type = namespaces.slice(0).sort().join(".");
|
handleObj.namespace = namespaces.slice(0).sort().join(".");
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
namespaces = [];
|
namespaces = [];
|
||||||
handler.type = "";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
handleObj.type = 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 = jQuery.event.special[ type ] || {};
|
||||||
|
|
||||||
// Init the event handler queue
|
// Init the event handler queue
|
||||||
if ( !handlers ) {
|
if ( !handlers ) {
|
||||||
handlers = events[ type ] = {};
|
handlers = events[ type ] = [];
|
||||||
|
|
||||||
// Check for a special event handler
|
// Check for a special event handler
|
||||||
// Only use addEventListener/attachEvent if the special
|
// Only use addEventListener/attachEvent if the special
|
||||||
// events handler returns false
|
// events handler returns false
|
||||||
if ( !special.setup || special.setup.call( elem, data, namespaces, handler) === false ) {
|
if ( !special.setup || special.setup.call( elem, data, namespaces, handler ) === false ) {
|
||||||
// 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, eventHandle, false );
|
||||||
|
|
||||||
} else if ( elem.attachEvent ) {
|
} else if ( elem.attachEvent ) {
|
||||||
elem.attachEvent( "on" + type, handle );
|
elem.attachEvent( "on" + type, eventHandle );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( special.add ) {
|
if ( special.add ) {
|
||||||
var modifiedHandler = special.add.call( elem, handler, data, namespaces, handlers );
|
special.add.call( elem, handleObj );
|
||||||
if ( modifiedHandler && jQuery.isFunction( modifiedHandler ) ) {
|
|
||||||
modifiedHandler.guid = modifiedHandler.guid || handler.guid;
|
|
||||||
modifiedHandler.data = modifiedHandler.data || handler.data;
|
|
||||||
modifiedHandler.type = modifiedHandler.type || handler.type;
|
|
||||||
handler = modifiedHandler;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add the function to the element's handler list
|
// Add the function to the element's handler list
|
||||||
handlers[ handler.guid ] = handler;
|
handlers.push( handleObj );
|
||||||
|
|
||||||
// Keep track of which events have been used, for global triggering
|
// Keep track of which events have been used, for global triggering
|
||||||
this.global[ type ] = true;
|
jQuery.event.global[ type ] = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Nullify elem to prevent memory leaks in IE
|
// Nullify elem to prevent memory leaks in IE
|
||||||
|
@ -142,112 +121,123 @@ jQuery.event = {
|
||||||
global: {},
|
global: {},
|
||||||
|
|
||||||
// Detach an event or set of events from an element
|
// Detach an event or set of events from an element
|
||||||
remove: function( elem, types, handler ) {
|
remove: function( elem, types, handler, pos ) {
|
||||||
// don't do events on text and comment nodes
|
// don't do events on text and comment nodes
|
||||||
if ( elem.nodeType === 3 || elem.nodeType === 8 ) {
|
if ( elem.nodeType === 3 || elem.nodeType === 8 ) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var elemData = jQuery.data( elem );
|
var ret, type, fn, i = 0, all, namespaces, namespace, special, eventType, handleObj, origType,
|
||||||
|
elemData = jQuery.data( elem ),
|
||||||
|
events = elemData && elemData.events;
|
||||||
|
|
||||||
if ( !elemData ) {
|
if ( !elemData || !events ) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
var events = elemData.events, ret, type, fn;
|
// types is actually an event object here
|
||||||
|
if ( types && types.type ) {
|
||||||
|
handler = types.handler;
|
||||||
|
types = types.type;
|
||||||
|
}
|
||||||
|
|
||||||
if ( events ) {
|
// Unbind all events for the element
|
||||||
// Unbind all events for the element
|
if ( !types || typeof types === "string" && types.charAt(0) === "." ) {
|
||||||
if ( types === undefined || (typeof types === "string" && types.charAt(0) === ".") ) {
|
types = types || "";
|
||||||
for ( type in events ) {
|
|
||||||
this.remove( elem, type + (types || "") );
|
|
||||||
}
|
|
||||||
|
|
||||||
} else {
|
for ( type in events ) {
|
||||||
// types is actually an event object here
|
jQuery.event.remove( elem, type + types );
|
||||||
if ( types.type ) {
|
}
|
||||||
handler = types.handler;
|
|
||||||
types = types.type;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Handle multiple events separated by a space
|
return;
|
||||||
// jQuery(...).unbind("mouseover mouseout", fn);
|
}
|
||||||
types = types.split(" ");
|
|
||||||
|
|
||||||
var i = 0, all, namespaces, namespace;
|
// Handle multiple events separated by a space
|
||||||
|
// jQuery(...).unbind("mouseover mouseout", fn);
|
||||||
|
types = types.split(" ");
|
||||||
|
|
||||||
while ( (type = types[ i++ ]) ) {
|
while ( (type = types[ i++ ]) ) {
|
||||||
all = type.indexOf(".") < 0;
|
origType = type;
|
||||||
namespaces = null;
|
handleObj = null;
|
||||||
|
all = type.indexOf(".") < 0;
|
||||||
|
namespaces = [];
|
||||||
|
|
||||||
if ( !all ) {
|
if ( !all ) {
|
||||||
// Namespaced event handlers
|
// Namespaced event handlers
|
||||||
namespaces = type.split(".");
|
namespaces = type.split(".");
|
||||||
type = namespaces.shift();
|
type = namespaces.shift();
|
||||||
|
|
||||||
namespace = new RegExp("(^|\\.)" +
|
namespace = new RegExp("(^|\\.)" +
|
||||||
jQuery.map( namespaces.slice(0).sort(), fcleanup ).join("\\.(?:.*\\.)?") + "(\\.|$)")
|
jQuery.map( namespaces.slice(0).sort(), fcleanup ).join("\\.(?:.*\\.)?") + "(\\.|$)")
|
||||||
|
}
|
||||||
|
|
||||||
} else {
|
eventType = events[ type ];
|
||||||
namespaces = [];
|
|
||||||
|
if ( !eventType ) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( !handler ) {
|
||||||
|
for ( var j = 0; j < eventType.length; j++ ) {
|
||||||
|
handleObj = eventType[ j ];
|
||||||
|
|
||||||
|
if ( all || namespace.test( handleObj.namespace ) ) {
|
||||||
|
jQuery.event.remove( elem, origType, handleObj.handler, j );
|
||||||
|
eventType.splice( j--, 1 );
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
var special = this.special[ type ] || {};
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if ( events[ type ] ) {
|
special = jQuery.event.special[ type ] || {};
|
||||||
// remove the given handler for the given type
|
|
||||||
if ( handler ) {
|
|
||||||
fn = events[ type ][ handler.guid ];
|
|
||||||
delete events[ type ][ handler.guid ];
|
|
||||||
|
|
||||||
// remove all handlers for the given type
|
for ( var j = pos || 0; j < eventType.length; j++ ) {
|
||||||
} else {
|
handleObj = eventType[ j ];
|
||||||
for ( var handle in events[ type ] ) {
|
|
||||||
// Handle the removal of namespaced events
|
if ( handler.guid === handleObj.guid ) {
|
||||||
if ( all || namespace.test( events[ type ][ handle ].type ) ) {
|
// remove the given handler for the given type
|
||||||
delete events[ type ][ handle ];
|
if ( all || namespace.test( handleObj.namespace ) ) {
|
||||||
}
|
fn = handleObj.handler;
|
||||||
}
|
|
||||||
|
if ( pos == null ) {
|
||||||
|
eventType.splice( j--, 1 );
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( special.remove ) {
|
if ( special.remove ) {
|
||||||
special.remove.call( elem, namespaces, fn);
|
special.remove.call( elem, namespaces, handleObj );
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// remove generic event handler if no more handlers exist
|
if ( pos != null ) {
|
||||||
for ( ret in events[ type ] ) {
|
break;
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( !ret ) {
|
|
||||||
if ( !special.teardown || special.teardown.call( elem, namespaces ) === false ) {
|
|
||||||
removeEvent( elem, type, elemData.handle );
|
|
||||||
}
|
|
||||||
|
|
||||||
ret = null;
|
|
||||||
delete events[ type ];
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove the expando if it's no longer used
|
// remove generic event handler if no more handlers exist
|
||||||
for ( ret in events ) {
|
if ( jQuery.isEmptyObject( events[ type ] ) ) {
|
||||||
break;
|
if ( !special.teardown || special.teardown.call( elem, namespaces ) === false ) {
|
||||||
|
removeEvent( elem, type, elemData.handle );
|
||||||
|
}
|
||||||
|
|
||||||
|
ret = null;
|
||||||
|
delete events[ type ];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove the expando if it's no longer used
|
||||||
|
if ( jQuery.isEmptyObject( events ) ) {
|
||||||
|
var handle = elemData.handle;
|
||||||
|
if ( handle ) {
|
||||||
|
handle.elem = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( !ret ) {
|
delete elemData.events;
|
||||||
var handle = elemData.handle;
|
delete elemData.handle;
|
||||||
if ( handle ) {
|
|
||||||
handle.elem = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
delete elemData.events;
|
if ( jQuery.isEmptyObject( elemData ) ) {
|
||||||
delete elemData.handle;
|
jQuery.removeData( elem );
|
||||||
|
|
||||||
if ( jQuery.isEmptyObject( elemData ) ) {
|
|
||||||
jQuery.removeData( elem );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
@ -278,7 +268,7 @@ jQuery.event = {
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
|
|
||||||
// Only trigger if we've ever bound an event for it
|
// Only trigger if we've ever bound an event for it
|
||||||
if ( this.global[ type ] ) {
|
if ( jQuery.event.global[ type ] ) {
|
||||||
jQuery.each( jQuery.cache, function() {
|
jQuery.each( jQuery.cache, function() {
|
||||||
if ( this.events && this.events[type] ) {
|
if ( this.events && this.events[type] ) {
|
||||||
jQuery.event.trigger( event, data, this.handle.elem );
|
jQuery.event.trigger( event, data, this.handle.elem );
|
||||||
|
@ -344,7 +334,7 @@ jQuery.event = {
|
||||||
target[ "on" + type ] = null;
|
target[ "on" + type ] = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.triggered = true;
|
jQuery.event.triggered = true;
|
||||||
target[ type ]();
|
target[ type ]();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -355,53 +345,56 @@ jQuery.event = {
|
||||||
target[ "on" + type ] = old;
|
target[ "on" + type ] = old;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.triggered = false;
|
jQuery.event.triggered = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
handle: function( event ) {
|
handle: function( event ) {
|
||||||
// returned undefined or false
|
var all, handlers, namespaces, namespace, events;
|
||||||
var all, handlers;
|
|
||||||
|
|
||||||
event = arguments[0] = jQuery.event.fix( event || window.event );
|
event = arguments[0] = jQuery.event.fix( event || window.event );
|
||||||
event.currentTarget = this;
|
event.currentTarget = this;
|
||||||
|
|
||||||
// Namespaced event handlers
|
// Namespaced event handlers
|
||||||
var namespaces = event.type.split(".");
|
all = event.type.indexOf(".") < 0;
|
||||||
event.type = namespaces.shift();
|
|
||||||
|
|
||||||
// Cache this now, all = true means, any handler
|
if ( !all ) {
|
||||||
all = !namespaces.length && !event.exclusive;
|
namespaces = event.type.split(".");
|
||||||
|
event.type = namespaces.shift();
|
||||||
|
namespace = new RegExp("(^|\\.)" + namespaces.slice(0).sort().join("\\.(?:.*\\.)?") + "(\\.|$)");
|
||||||
|
}
|
||||||
|
|
||||||
var namespace = new RegExp("(^|\\.)" + namespaces.slice(0).sort().join("\\.(?:.*\\.)?") + "(\\.|$)");
|
var events = jQuery.data(this, "events"), handlers = events[ event.type ];
|
||||||
|
|
||||||
handlers = ( jQuery.data(this, "events") || {} )[ event.type ];
|
if ( events && handlers ) {
|
||||||
|
// Clone the handlers to prevent manipulation
|
||||||
|
handlers = handlers.slice(0);
|
||||||
|
|
||||||
for ( var j in handlers ) {
|
for ( var j = 0, l = handlers.length; j < l; j++ ) {
|
||||||
var handler = handlers[ j ];
|
var handleObj = handlers[ j ];
|
||||||
|
|
||||||
// Filter the functions by class
|
// Filter the functions by class
|
||||||
if ( all || namespace.test(handler.type) ) {
|
if ( (all && !event.exclusive) || namespace.test( handleObj.namespace ) ) {
|
||||||
// Pass in a reference to the handler function itself
|
// Pass in a reference to the handler function itself
|
||||||
// So that we can later remove it
|
// So that we can later remove it
|
||||||
event.handler = handler;
|
event.handler = handleObj.handler;
|
||||||
event.data = handler.data;
|
event.data = handleObj.data;
|
||||||
|
|
||||||
var ret = handler.apply( this, arguments );
|
var ret = handleObj.handler.apply( this, arguments );
|
||||||
|
|
||||||
if ( ret !== undefined ) {
|
if ( ret !== undefined ) {
|
||||||
event.result = ret;
|
event.result = ret;
|
||||||
if ( ret === false ) {
|
if ( ret === false ) {
|
||||||
event.preventDefault();
|
event.preventDefault();
|
||||||
event.stopPropagation();
|
event.stopPropagation();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ( event.isImmediatePropagationStopped() ) {
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ( event.isImmediatePropagationStopped() ) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -88,8 +88,9 @@ test("bind(), namespace with special add", function() {
|
||||||
},
|
},
|
||||||
setup: function(){},
|
setup: function(){},
|
||||||
teardown: function(){},
|
teardown: function(){},
|
||||||
add: function( handler, data, namespaces ) {
|
add: function( handleObj ) {
|
||||||
return function(e) {
|
var handler = handleObj.handler;
|
||||||
|
handleObj.handler = function(e) {
|
||||||
e.xyz = ++i;
|
e.xyz = ++i;
|
||||||
handler.apply( this, arguments );
|
handler.apply( this, arguments );
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue