Skip to content

Commit 754108f

Browse files
committed
Event: Make trigger(focus/blur/click) work with native handlers
In `leverageNative`, instead of calling `event.stopImmediatePropagation()` which would abort both native & jQuery handlers, set the wrapper's `isImmediatePropagationStopped` property to a function returning `true`. Since for each element + type pair jQuery attaches only one native handler, there is also only one wrapper jQuery event so this achieves the goal: on the target element jQuery handlers don't fire but native ones do. Unfortunately, this workaround doesn't work for handlers on ancestors - since the native event is re-wrapped by a jQuery one on each level of the propagation, the only way to stop it for jQuery was to stop it for everyone via native `stopPropagation()`. This is not a problem for `focus`/`blur` which don't bubble, but it does also stop `click` on checkboxes and radios. We accept this limitation. Fixes gh-5015 Closes gh-5228 (cherry picked from commit 6ad3651)
1 parent 59f7b55 commit 754108f

File tree

2 files changed

+73
-6
lines changed

2 files changed

+73
-6
lines changed

src/event.js

+12-4
Original file line numberDiff line numberDiff line change
@@ -559,8 +559,8 @@ function leverageNative( el, type, isSetup ) {
559559
}
560560

561561
// If this is an inner synthetic event for an event with a bubbling surrogate
562-
// (focus or blur), assume that the surrogate already propagated from triggering the
563-
// native event and prevent that from happening again here.
562+
// (focus or blur), assume that the surrogate already propagated from triggering
563+
// the native event and prevent that from happening again here.
564564
// This technically gets the ordering wrong w.r.t. to `.trigger()` (in which the
565565
// bubbling surrogate propagates *after* the non-bubbling base), but that seems
566566
// less bad than duplication.
@@ -579,8 +579,16 @@ function leverageNative( el, type, isSetup ) {
579579
this
580580
) );
581581

582-
// Abort handling of the native event
583-
event.stopImmediatePropagation();
582+
// Abort handling of the native event by all jQuery handlers while allowing
583+
// native handlers on the same element to run. On target, this is achieved
584+
// by stopping immediate propagation just on the jQuery event. However,
585+
// the native event is re-wrapped by a jQuery one on each level of the
586+
// propagation so the only way to stop it for jQuery is to stop it for
587+
// everyone via native `stopPropagation()`. This is not a problem for
588+
// focus/blur which don't bubble, but it does also stop click on checkboxes
589+
// and radios. We accept this limitation.
590+
event.stopPropagation();
591+
event.isImmediatePropagationStopped = returnTrue;
584592
}
585593
}
586594
} );

test/unit/event.js

+61-2
Original file line numberDiff line numberDiff line change
@@ -3412,6 +3412,36 @@ QUnit.test( "trigger(focus) works after focusing when hidden (gh-4950)", functio
34123412
assert.equal( document.activeElement, input[ 0 ], "input has focus" );
34133413
} );
34143414

3415+
QUnit.test( "trigger(focus) fires native & jQuery handlers (gh-5015)", function( assert ) {
3416+
assert.expect( 3 );
3417+
3418+
var input = jQuery( "<input />" ),
3419+
3420+
// Support: IE 9 - 11+
3421+
// focus is async in IE; we now emulate it via sync focusin in jQuery
3422+
// but this test also attaches native handlers.
3423+
done = assert.async( 3 );
3424+
3425+
input.appendTo( "#qunit-fixture" );
3426+
3427+
input[ 0 ].addEventListener( "focus", function() {
3428+
assert.ok( true, "1st native handler fired" );
3429+
done();
3430+
} );
3431+
3432+
input.on( "focus", function() {
3433+
assert.ok( true, "jQuery handler fired" );
3434+
done();
3435+
} );
3436+
3437+
input[ 0 ].addEventListener( "focus", function() {
3438+
assert.ok( true, "2nd native handler fired" );
3439+
done();
3440+
} );
3441+
3442+
input.trigger( "focus" );
3443+
} );
3444+
34153445
// TODO replace with an adaptation of
34163446
// https://github.com/jquery/jquery/pull/1367/files#diff-a215316abbaabdf71857809e8673ea28R2464
34173447
( function() {
@@ -3420,10 +3450,13 @@ QUnit.test( "trigger(focus) works after focusing when hidden (gh-4950)", functio
34203450
checkbox: "<input type='checkbox'>",
34213451
radio: "<input type='radio'>"
34223452
},
3423-
makeTestFor3751
3453+
function( type, html ) {
3454+
makeTestForGh3751( type, html );
3455+
makeTestForGh5015( type, html );
3456+
}
34243457
);
34253458

3426-
function makeTestFor3751( type, html ) {
3459+
function makeTestForGh3751( type, html ) {
34273460
var testName = "native-backed namespaced clicks are handled correctly (gh-3751) - " + type;
34283461
QUnit.test( testName, function( assert ) {
34293462
assert.expect( 2 );
@@ -3450,4 +3483,30 @@ QUnit.test( "trigger(focus) works after focusing when hidden (gh-4950)", functio
34503483
target.trigger( "click.fired" );
34513484
} );
34523485
}
3486+
3487+
function makeTestForGh5015( type, html ) {
3488+
var testName = "trigger(click) fires native & jQuery handlers (gh-5015) - " + type;
3489+
QUnit.test( testName, function( assert ) {
3490+
assert.expect( 3 );
3491+
3492+
var parent = supportjQuery( "<div class='parent'>" + html + "</div>" ),
3493+
input = jQuery( parent[ 0 ].firstChild );
3494+
3495+
parent.appendTo( "#qunit-fixture" );
3496+
3497+
input[ 0 ].addEventListener( "click", function() {
3498+
assert.ok( true, "1st native handler fired" );
3499+
} );
3500+
3501+
input.on( "click", function() {
3502+
assert.ok( true, "jQuery handler fired" );
3503+
} );
3504+
3505+
input[ 0 ].addEventListener( "click", function() {
3506+
assert.ok( true, "2nd native handler fired" );
3507+
} );
3508+
3509+
input.trigger( "click" );
3510+
} );
3511+
}
34533512
} )();

0 commit comments

Comments
 (0)