Skip to content

Commit 3299236

Browse files
committed
Selector:Manipulation: Fix DOM manip within template contents
The `<template/>` element `contents` property is a document fragment that may have a `null` `documentElement`. In Safari 16 this happens in more cases due to recent spec changes - in particular, even if that document fragment is explicitly adopted into an outer document. We're testing both of those cases now. The crash used to happen in `jQuery.contains`. As it turns out, we don't need to query the supposed container `documentElement` if it has the `Node.DOCUMENT_NODE` (9) `nodeType`; we can call `.contains()` directly on the `document`. That avoids the crash. Fixes gh-5147 Closes gh-5158
1 parent 0208224 commit 3299236

File tree

3 files changed

+56
-4
lines changed

3 files changed

+56
-4
lines changed

src/selector/contains.js

+3-4
Original file line numberDiff line numberDiff line change
@@ -2,15 +2,14 @@ import jQuery from "../core.js";
22

33
// Note: an element does not contain itself
44
jQuery.contains = function( a, b ) {
5-
var adown = a.nodeType === 9 ? a.documentElement : a,
6-
bup = b && b.parentNode;
5+
var bup = b && b.parentNode;
76

87
return a === bup || !!( bup && bup.nodeType === 1 && (
98

109
// Support: IE 9 - 11+
1110
// IE doesn't have `contains` on SVG.
12-
adown.contains ?
13-
adown.contains( bup ) :
11+
a.contains ?
12+
a.contains( bup ) :
1413
a.compareDocumentPosition && a.compareDocumentPosition( bup ) & 16
1514
) );
1615
};

test/unit/manipulation.js

+40
Original file line numberDiff line numberDiff line change
@@ -2830,6 +2830,46 @@ QUnit.test( "Make sure tr is not appended to the wrong tbody (gh-3439)", functio
28302830
assert.strictEqual( htmlOut, htmlExpected );
28312831
} );
28322832

2833+
[ true, false ].forEach( function( adoptedCase ) {
2834+
QUnit.testUnlessIE(
2835+
"Manip within <template /> content moved back & forth doesn't throw - " + (
2836+
adoptedCase ? "explicitly adopted" : "not explicitly adopted"
2837+
) + " (gh-5147)",
2838+
function( assert ) {
2839+
assert.expect( 1 );
2840+
2841+
var fragment, diva, divb,
2842+
div = jQuery( "" +
2843+
"<div>\n" +
2844+
" <div><div class='a'></div></div>\n" +
2845+
" <div><div class='b'></div></div>\n" +
2846+
"</div>" +
2847+
"" ),
2848+
template = jQuery( "<template></template>" );
2849+
2850+
jQuery( "#qunit-fixture" )
2851+
.append( div )
2852+
.append( template );
2853+
2854+
fragment = template[ 0 ].content;
2855+
diva = div.find( ".a" );
2856+
divb = div.find( ".b" );
2857+
2858+
if ( adoptedCase ) {
2859+
document.adoptNode( fragment );
2860+
}
2861+
2862+
fragment.appendChild( div.children()[ 0 ] );
2863+
fragment.appendChild( div.children()[ 0 ] );
2864+
2865+
diva.insertBefore( divb );
2866+
2867+
assert.strictEqual( diva.siblings( ".b" ).length, 1,
2868+
"Insertion worked" );
2869+
}
2870+
);
2871+
} );
2872+
28332873
QUnit.test( "Make sure tags with single-character names are found (gh-4124)", function( assert ) {
28342874
assert.expect( 1 );
28352875

test/unit/selector.js

+13
Original file line numberDiff line numberDiff line change
@@ -1895,6 +1895,19 @@ QUnit.test( "jQuery.contains in SVG (jQuery trac-10832)", function( assert ) {
18951895
"parent (negative)" );
18961896
} );
18971897

1898+
QUnit.testUnlessIE( "jQuery.contains within <template/> doesn't throw (gh-5147)", function( assert ) {
1899+
assert.expect( 1 );
1900+
1901+
var template = jQuery( "<template><div><div class='a'></div></div></template>" ),
1902+
a = jQuery( template[ 0 ].content ).find( ".a" );
1903+
1904+
template.appendTo( "#qunit-fixture" );
1905+
1906+
jQuery.contains( a[ 0 ].ownerDocument, a[ 0 ] );
1907+
1908+
assert.ok( true, "Didn't throw" );
1909+
} );
1910+
18981911
QUnit.test( "jQuery.uniqueSort", function( assert ) {
18991912
assert.expect( 14 );
19001913

0 commit comments

Comments
 (0)