mirror of
https://github.com/jquery/jquery.git
synced 2025-01-10 18:24:24 +00:00
Fix #14180. Allow cross-frame use of focusin/out. Close gh-1369.
This commit is contained in:
parent
083edd60a6
commit
9b6f074580
21
src/event.js
21
src/event.js
@ -727,22 +727,29 @@ jQuery.each({
|
|||||||
if ( !support.focusinBubbles ) {
|
if ( !support.focusinBubbles ) {
|
||||||
jQuery.each({ focus: "focusin", blur: "focusout" }, function( orig, fix ) {
|
jQuery.each({ focus: "focusin", blur: "focusout" }, function( orig, fix ) {
|
||||||
|
|
||||||
// Attach a single capturing handler while someone wants focusin/focusout
|
// Attach a single capturing handler on the document while someone wants focusin/focusout
|
||||||
var attaches = 0,
|
var handler = function( event ) {
|
||||||
handler = function( event ) {
|
|
||||||
jQuery.event.simulate( fix, event.target, jQuery.event.fix( event ), true );
|
jQuery.event.simulate( fix, event.target, jQuery.event.fix( event ), true );
|
||||||
};
|
};
|
||||||
|
|
||||||
jQuery.event.special[ fix ] = {
|
jQuery.event.special[ fix ] = {
|
||||||
setup: function() {
|
setup: function() {
|
||||||
if ( attaches++ === 0 ) {
|
var doc = this.ownerDocument,
|
||||||
document.addEventListener( orig, handler, true );
|
attaches = data_priv.access( doc, "focusCount" );
|
||||||
|
|
||||||
|
if ( !attaches ) {
|
||||||
|
doc.addEventListener( orig, handler, true );
|
||||||
}
|
}
|
||||||
|
data_priv.access( doc, "focusCount", ( attaches || 0 ) + 1 );
|
||||||
},
|
},
|
||||||
teardown: function() {
|
teardown: function() {
|
||||||
if ( --attaches === 0 ) {
|
var doc = this.ownerDocument,
|
||||||
document.removeEventListener( orig, handler, true );
|
attaches = data_priv.access( doc, "focusCount" ) - 1;
|
||||||
|
|
||||||
|
if ( !attaches ) {
|
||||||
|
doc.removeEventListener( orig, handler, true );
|
||||||
}
|
}
|
||||||
|
data_priv.access( doc, "focusCount", attaches );
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
18
test/data/event/focusinCrossFrame.html
Normal file
18
test/data/event/focusinCrossFrame.html
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
<!doctype html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<meta http-equiv="Content-type" content="text/html; charset=utf-8">
|
||||||
|
<title>focusin event cross-frame (#14180)</title>
|
||||||
|
|
||||||
|
<script src="../../jquery.js"></script>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<input type="text" id="frame-input" />
|
||||||
|
<script>
|
||||||
|
// Call parent when this frame is fully loaded, it will mess with #frame-input
|
||||||
|
jQuery( window ).one( "load", function() {
|
||||||
|
window.parent.iframeCallback( document );
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
@ -2434,6 +2434,33 @@ test("fixHooks extensions", function() {
|
|||||||
jQuery.event.fixHooks.click = saved;
|
jQuery.event.fixHooks.click = saved;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
testIframeWithCallback( "focusin from an iframe", "event/focusinCrossFrame.html", function( frameDoc ) {
|
||||||
|
expect(1);
|
||||||
|
|
||||||
|
var input = jQuery( frameDoc ).find( "#frame-input" );
|
||||||
|
|
||||||
|
// Create a focusin handler on the parent; shouldn't affect the iframe's fate
|
||||||
|
jQuery ( "body" ).on( "focusin.iframeTest", function() {
|
||||||
|
ok( false, "fired a focusin event in the parent document" );
|
||||||
|
});
|
||||||
|
|
||||||
|
input.on( "focusin", function() {
|
||||||
|
ok( true, "fired a focusin event in the iframe" );
|
||||||
|
});
|
||||||
|
|
||||||
|
// Avoid a native event; Chrome can't force focus to another frame
|
||||||
|
input.trigger( "focusin" );
|
||||||
|
|
||||||
|
// Must manually remove handler to avoid leaks in our data store
|
||||||
|
input.remove();
|
||||||
|
|
||||||
|
// Be sure it was removed; nothing should happen
|
||||||
|
input.trigger( "focusin" );
|
||||||
|
|
||||||
|
// Remove body handler manually since it's outside the fixture
|
||||||
|
jQuery( "body" ).off( "focusin.iframeTest" );
|
||||||
|
});
|
||||||
|
|
||||||
testIframeWithCallback( "jQuery.ready promise", "event/promiseReady.html", function( isOk ) {
|
testIframeWithCallback( "jQuery.ready promise", "event/promiseReady.html", function( isOk ) {
|
||||||
expect(1);
|
expect(1);
|
||||||
ok( isOk, "$.when( $.ready ) works" );
|
ok( isOk, "$.when( $.ready ) works" );
|
||||||
|
Loading…
Reference in New Issue
Block a user