From 3518cddca7cf520cf4cac6b941695b7c37543758 Mon Sep 17 00:00:00 2001 From: Dennis Snell Date: Sun, 10 Mar 2019 16:49:11 -0700 Subject: [PATCH 1/2] Fix: Pasting captions without an image fails Fixes #13527 See original work in #12315 When pasting content which includes the `[caption]` shortcode we assume that the content is well-formed (that there is not only an `` in there but also in the first position). In this patch we fix the problem by removing the old code, which removed the first `Element` node, and replaced it with code that removes the first `IMG` node _if one is found_. We're leaving other image nodes in place in case the caption contains image nodes and we're not requiring that the `IMG` be the first child of the caption shortcode in case people are wrapping the image in other valid HTML like this... ```html [caption][/caption] ``` See the new unit tests for a more complete specification of the intended behaviors. PR reviewed, debugged, and created by: -> LANNISTER MOB <- - @codebykat - @dmsnell - @gwwar - @kwight - @mmtr - @obenland - @rodrigoi - @vindl --- packages/block-library/src/image/index.js | 22 +++++++++------ .../block-library/src/image/test/index.js | 28 +++++++++++++++++++ 2 files changed, 42 insertions(+), 8 deletions(-) create mode 100644 packages/block-library/src/image/test/index.js diff --git a/packages/block-library/src/image/index.js b/packages/block-library/src/image/index.js index 92b9af7ec9a42c..ba0f9d1b237ac1 100644 --- a/packages/block-library/src/image/index.js +++ b/packages/block-library/src/image/index.js @@ -123,6 +123,19 @@ function getFirstAnchorAttributeFormHTML( html, attributeName ) { } } +export function stripFirstImage( attributes, { shortcode } ) { + const { body } = document.implementation.createHTMLDocument( '' ); + + body.innerHTML = shortcode.content; + + const firstImage = body.querySelector( 'img' ); + if ( firstImage ) { + firstImage.parentNode.removeChild( firstImage ); + } + + return body.innerHTML.trim(); +} + export const settings = { title: __( 'Image' ), @@ -196,14 +209,7 @@ export const settings = { selector: 'img', }, caption: { - shortcode: ( attributes, { shortcode } ) => { - const { body } = document.implementation.createHTMLDocument( '' ); - - body.innerHTML = shortcode.content; - body.removeChild( body.firstElementChild ); - - return body.innerHTML.trim(); - }, + shortcode: stripFirstImage, }, href: { shortcode: ( attributes, { shortcode } ) => { diff --git a/packages/block-library/src/image/test/index.js b/packages/block-library/src/image/test/index.js new file mode 100644 index 00000000000000..4cd5a8c5b9cbbb --- /dev/null +++ b/packages/block-library/src/image/test/index.js @@ -0,0 +1,28 @@ +/** + * Internal dependencies + */ +import { stripFirstImage } from '../'; + +describe( 'stripFirstImage', () => { + test( 'should do nothing if no image is present', () => { + expect( stripFirstImage( {}, { shortcode: { content: '' } } ) ).toEqual( '' ); + expect( stripFirstImage( {}, { shortcode: { content: 'Tucson' } } ) ).toEqual( 'Tucson' ); + expect( stripFirstImage( {}, { shortcode: { content: 'Tucson' } } ) ).toEqual( 'Tucson' ); + } ); + + test( 'should strip out image when leading as expected', () => { + expect( stripFirstImage( {}, { shortcode: { content: '' } } ) ).toEqual( '' ); + expect( stripFirstImage( {}, { shortcode: { content: 'Image!' } } ) ).toEqual( 'Image!' ); + expect( stripFirstImage( {}, { shortcode: { content: 'Image!' } } ) ).toEqual( 'Image!' ); + } ); + + test( 'should strip out image when not in leading position as expected', () => { + expect( stripFirstImage( {}, { shortcode: { content: 'Before' } } ) ).toEqual( 'Before' ); + expect( stripFirstImage( {}, { shortcode: { content: 'BeforeImage!' } } ) ).toEqual( 'BeforeImage!' ); + expect( stripFirstImage( {}, { shortcode: { content: 'BeforeImage!' } } ) ).toEqual( 'BeforeImage!' ); + } ); + + test( 'should strip out only the first of many images', () => { + expect( stripFirstImage( {}, { shortcode: { content: '' } } ) ).toEqual( '' ); + } ); +} ); From eb79d924a55d4d125dbab95206e8ef5bae0051c7 Mon Sep 17 00:00:00 2001 From: Kerry Liu Date: Fri, 15 Mar 2019 15:10:45 -0700 Subject: [PATCH 2/2] Update stripFirstImage behavior to also remove matching topmost parent node --- packages/block-library/src/image/index.js | 12 +++++++++--- packages/block-library/src/image/test/index.js | 4 ++++ 2 files changed, 13 insertions(+), 3 deletions(-) diff --git a/packages/block-library/src/image/index.js b/packages/block-library/src/image/index.js index ba0f9d1b237ac1..ace6b55f31636b 100644 --- a/packages/block-library/src/image/index.js +++ b/packages/block-library/src/image/index.js @@ -128,9 +128,15 @@ export function stripFirstImage( attributes, { shortcode } ) { body.innerHTML = shortcode.content; - const firstImage = body.querySelector( 'img' ); - if ( firstImage ) { - firstImage.parentNode.removeChild( firstImage ); + let nodeToRemove = body.querySelector( 'img' ); + + // if an image has parents, find the topmost node to remove + while ( nodeToRemove && nodeToRemove.parentNode && nodeToRemove.parentNode !== body ) { + nodeToRemove = nodeToRemove.parentNode; + } + + if ( nodeToRemove ) { + nodeToRemove.parentNode.removeChild( nodeToRemove ); } return body.innerHTML.trim(); diff --git a/packages/block-library/src/image/test/index.js b/packages/block-library/src/image/test/index.js index 4cd5a8c5b9cbbb..5b21dcefdf6ebd 100644 --- a/packages/block-library/src/image/test/index.js +++ b/packages/block-library/src/image/test/index.js @@ -25,4 +25,8 @@ describe( 'stripFirstImage', () => { test( 'should strip out only the first of many images', () => { expect( stripFirstImage( {}, { shortcode: { content: '' } } ) ).toEqual( '' ); } ); + + test( 'should strip out the first image and its wrapping parents', () => { + expect( stripFirstImage( {}, { shortcode: { content: '

' } } ) ).toEqual( '

' ); + } ); } );