From b025b5e6688d8cbf4947accb745b2be64cfeb185 Mon Sep 17 00:00:00 2001 From: Josh Loehr Date: Tue, 19 Sep 2023 18:32:14 -0700 Subject: [PATCH] fix: prevent duplicate child cloning of iframe content --- src/clone-node.ts | 14 ++++++-------- test/resources/iframe/embed.html | 5 +++++ test/resources/iframe/image | 1 + test/resources/iframe/node.html | 6 ++++++ test/spec/iframe.spec.ts | 21 +++++++++++++++++++++ 5 files changed, 39 insertions(+), 8 deletions(-) create mode 100644 test/resources/iframe/embed.html create mode 100644 test/resources/iframe/image create mode 100644 test/resources/iframe/node.html create mode 100644 test/spec/iframe.spec.ts diff --git a/src/clone-node.ts b/src/clone-node.ts index 500ce332..5b57e3e7 100644 --- a/src/clone-node.ts +++ b/src/clone-node.ts @@ -76,11 +76,6 @@ async function cloneChildren( if (isSlotElement(nativeNode) && nativeNode.assignedNodes) { children = toArray(nativeNode.assignedNodes()) - } else if ( - isInstanceOfElement(nativeNode, HTMLIFrameElement) && - nativeNode.contentDocument?.body - ) { - children = toArray(nativeNode.contentDocument.body.childNodes) } else { children = toArray((nativeNode.shadowRoot ?? nativeNode).childNodes) } @@ -133,11 +128,11 @@ function cloneCSSStyle(nativeNode: T, clonedNode: T) { ) { value = 'block' } - + if (name === 'd' && clonedNode.getAttribute('d')) { value = `path(${clonedNode.getAttribute('d')})` } - + targetStyle.setProperty( name, value, @@ -171,7 +166,10 @@ function cloneSelectValue(nativeNode: T, clonedNode: T) { } function decorate(nativeNode: T, clonedNode: T): T { - if (isInstanceOfElement(clonedNode, Element)) { + if ( + isInstanceOfElement(clonedNode, Element) && + !isInstanceOfElement(nativeNode, HTMLIFrameElement) + ) { cloneCSSStyle(nativeNode, clonedNode) clonePseudoElements(nativeNode, clonedNode) cloneInputValue(nativeNode, clonedNode) diff --git a/test/resources/iframe/embed.html b/test/resources/iframe/embed.html new file mode 100644 index 00000000..62844377 --- /dev/null +++ b/test/resources/iframe/embed.html @@ -0,0 +1,5 @@ + + +
Embeddable content
+ + diff --git a/test/resources/iframe/image b/test/resources/iframe/image new file mode 100644 index 00000000..7d48dd69 --- /dev/null +++ b/test/resources/iframe/image @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/test/resources/iframe/node.html b/test/resources/iframe/node.html new file mode 100644 index 00000000..e8974be2 --- /dev/null +++ b/test/resources/iframe/node.html @@ -0,0 +1,6 @@ + diff --git a/test/spec/iframe.spec.ts b/test/spec/iframe.spec.ts new file mode 100644 index 00000000..1299b73a --- /dev/null +++ b/test/spec/iframe.spec.ts @@ -0,0 +1,21 @@ +/* eslint-disable promise/no-callback-in-promise */ + +import './setup' +import { bootstrap, renderAndCheck } from './helper' +import { delay } from '../../src/util' + +describe('work with iframe element', () => { + it('should render iframe element', (done) => { + bootstrap('iframe/node.html', undefined, 'iframe/image') + .then(delay(100)) + .then((node) => { + const iframe = node.querySelector('#iframe') as HTMLIFrameElement + return renderAndCheck(node, { + width: parseInt(iframe.width, 10), + height: parseInt(iframe.height, 10), + }) + }) + .then(done) + .catch(done) + }) +})