Event: Don't crash if an element is removed on blur

In Chrome, if an element having a `focusout` handler is blurred by
clicking outside of it, it invokes the handler synchronously. If
that handler calls `.remove()` on the element, the data is cleared,
leaving private data undefined. We're reading a property from that
data so we need to guard against this.

Fixes gh-4417
Closes gh-4799

(cherry picked from commit 5c2d08704e)
This commit is contained in:
Michał Gołębiowski-Owczarek 2020-10-19 21:17:51 +02:00 committed by Michał Gołębiowski-Owczarek
parent 4c572a7fee
commit aaf9c55ad2
2 changed files with 34 additions and 1 deletions

View File

@ -581,7 +581,13 @@ function leverageNative( el, type, expectSync ) {
// Cancel the outer synthetic event
event.stopImmediatePropagation();
event.preventDefault();
return result.value;
// Support: Chrome 86+
// In Chrome, if an element having a focusout handler is blurred by
// clicking outside of it, it invokes the handler synchronously. If
// that handler calls `.remove()` on the element, the data is cleared,
// leaving `result` undefined. We need to guard against this.
return result && result.value;
}
// If this is an inner synthetic event for an event with a bubbling surrogate

View File

@ -2625,6 +2625,33 @@ QUnit.test( "focusin on document & window", function( assert ) {
jQuery( document ).off( "focusout", increment );
} );
QUnit.test( "element removed during focusout (gh-4417)", function( assert ) {
assert.expect( 1 );
var button = jQuery( "<button>Click me</button>" );
button.appendTo( "#qunit-fixture" );
button.on( "click", function() {
button.trigger( "blur" );
assert.ok( true, "Removing the element didn't crash" );
} );
// Support: Chrome 86+
// In Chrome, if an element having a focusout handler is blurred by
// clicking outside of it, it invokes the handler synchronously. However,
// if the click happens programmatically, the invocation is asynchronous.
// As we have no way to simulate real user input in unit tests, simulate
// this behavior by calling `jQuery.cleanData` & removing the element using
// native APIs.
button[ 0 ].blur = function() {
jQuery.cleanData( [ this ] );
this.parentNode.removeChild( this );
};
button[ 0 ].click();
} );
testIframe(
"jQuery.ready promise",
"event/promiseReady.html",