diff options
-rw-r--r-- | src/manipulation/buildFragment.js | 14 | ||||
-rw-r--r-- | src/manipulation/wrapMap.js | 11 | ||||
-rw-r--r-- | test/unit/manipulation.js | 11 |
3 files changed, 23 insertions, 13 deletions
diff --git a/src/manipulation/buildFragment.js b/src/manipulation/buildFragment.js index daf383aea..9ac71acc9 100644 --- a/src/manipulation/buildFragment.js +++ b/src/manipulation/buildFragment.js @@ -1,6 +1,7 @@ import jQuery from "../core.js"; import toType from "../core/toType.js"; import isAttached from "../core/isAttached.js"; +import arr from "../var/arr.js"; import rtagName from "./var/rtagName.js"; import rscriptType from "./var/rscriptType.js"; import wrapMap from "./wrapMap.js"; @@ -35,15 +36,16 @@ function buildFragment( elems, context, scripts, selection, ignored ) { // Deserialize a standard representation tag = ( rtagName.exec( elem ) || [ "", "" ] )[ 1 ].toLowerCase(); - wrap = wrapMap[ tag ] || wrapMap._default; - tmp.innerHTML = wrap[ 1 ] + jQuery.htmlPrefilter( elem ) + wrap[ 2 ]; + wrap = wrapMap[ tag ] || arr; - // Descend through wrappers to the right content - j = wrap[ 0 ]; - while ( j-- ) { - tmp = tmp.lastChild; + // Create wrappers & descend into them. + j = wrap.length; + while ( --j > -1 ) { + tmp = tmp.appendChild( context.createElement( wrap[ j ] ) ); } + tmp.innerHTML = jQuery.htmlPrefilter( elem ); + jQuery.merge( nodes, tmp.childNodes ); // Remember the top-level container diff --git a/src/manipulation/wrapMap.js b/src/manipulation/wrapMap.js index 01937ecc3..457902595 100644 --- a/src/manipulation/wrapMap.js +++ b/src/manipulation/wrapMap.js @@ -1,4 +1,3 @@ -// We have to close these tags to support XHTML (#13200) var wrapMap = { // Table parts need to be wrapped with `<table>` or they're @@ -6,12 +5,10 @@ var wrapMap = { // XHTML parsers do not magically insert elements in the // same way that tag soup parsers do, so we cannot shorten // this by omitting <tbody> or other required elements. - thead: [ 1, "<table>", "</table>" ], - col: [ 2, "<table><colgroup>", "</colgroup></table>" ], - tr: [ 2, "<table><tbody>", "</tbody></table>" ], - td: [ 3, "<table><tbody><tr>", "</tr></tbody></table>" ], - - _default: [ 0, "", "" ] + thead: [ "table" ], + col: [ "colgroup", "table" ], + tr: [ "tbody", "table" ], + td: [ "tr", "tbody", "table" ] }; wrapMap.tbody = wrapMap.tfoot = wrapMap.colgroup = wrapMap.caption = wrapMap.thead; diff --git a/test/unit/manipulation.js b/test/unit/manipulation.js index 81a64c762..a4a46f924 100644 --- a/test/unit/manipulation.js +++ b/test/unit/manipulation.js @@ -2969,3 +2969,14 @@ QUnit.test( "Sanitized HTML doesn't get unsanitized", function( assert ) { test( "<noembed><noembed/><img src=url404 onerror=xss(12)>" ); } } ); + +QUnit.test( "Works with invalid attempts to close the table wrapper", function( assert ) { + assert.expect( 3 ); + + // This test case attempts to close the tags which wrap input + // based on matching done in wrapMap which should be ignored. + var elem = jQuery( "<td></td></tr></tbody></table><td></td>" ); + assert.strictEqual( elem.length, 2, "Two elements created" ); + assert.strictEqual( elem[ 0 ].nodeName.toLowerCase(), "td", "First element is td" ); + assert.strictEqual( elem[ 1 ].nodeName.toLowerCase(), "td", "Second element is td" ); +} ); |