From b93825eb632a8f532050166e01a50bfce62bb903 Mon Sep 17 00:00:00 2001 From: Shachar <34343793+ShaMan123@users.noreply.github.com> Date: Mon, 17 May 2021 15:09:55 +0300 Subject: [PATCH] [Enhancement] Support Cloning Custom Elements (#129) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat(custom-elements): ✨ obtain children of shadow node for custom elements Inspired by https://github.com/tsayen/dom-to-image/pull/370 If element has `shadowRoot` property it is custom (and the shadow node is open) so fetch it's children. Closes #128 * fix: 🐛 add safety checks to support special/custom elements Introduced by 021f124 Access to shadow node exposes the code to custom elements that break the current code. Was breaking because of elements without a style node. This commit fixes these bugs with simple, NON-BREAKING safety checks. * test(htmltoimage): 🚨 Added tests covering custom elements renders mathfield custom element https://github.com/arnog/mathlive --- src/cloneNode.ts | 6 ++- src/embedImages.ts | 2 +- test/spec/index.spec.ts | 40 ++++++++++++++++++++ test/spec/resources/custom-element/image | 1 + test/spec/resources/custom-element/node.html | 9 +++++ test/spec/resources/custom-element/style.css | 0 6 files changed, 55 insertions(+), 3 deletions(-) create mode 100644 test/spec/resources/custom-element/image create mode 100644 test/spec/resources/custom-element/node.html create mode 100644 test/spec/resources/custom-element/style.css diff --git a/src/cloneNode.ts b/src/cloneNode.ts index 510e6f51..90d8b39e 100644 --- a/src/cloneNode.ts +++ b/src/cloneNode.ts @@ -27,7 +27,9 @@ async function cloneChildren( clonedNode: HTMLElement, filter?: Function, ): Promise { - const children = toArray(nativeNode.childNodes) + const children = toArray( + (nativeNode.shadowRoot ?? nativeNode).childNodes, + ) if (children.length === 0) { return Promise.resolve(clonedNode) } @@ -65,7 +67,7 @@ async function decorate( function cloneCssStyle(nativeNode: HTMLElement, clonedNode: HTMLElement) { const source = window.getComputedStyle(nativeNode) const target = clonedNode.style - + if (!target) return if (source.cssText) { target.cssText = source.cssText } else { diff --git a/src/embedImages.ts b/src/embedImages.ts index e7ff8f7d..79d55e3a 100644 --- a/src/embedImages.ts +++ b/src/embedImages.ts @@ -21,7 +21,7 @@ async function embedBackground( clonedNode: HTMLElement, options: Options, ): Promise { - const background = clonedNode.style.getPropertyValue('background') + const background = clonedNode.style?.getPropertyValue('background') if (!background) { return Promise.resolve(clonedNode) } diff --git a/test/spec/index.spec.ts b/test/spec/index.spec.ts index 0d845e10..7ab4a90a 100644 --- a/test/spec/index.spec.ts +++ b/test/spec/index.spec.ts @@ -229,6 +229,46 @@ describe('html to image', () => { .then(done) .catch(done) }) + + describe('custom element', () => { + let link: HTMLLinkElement + beforeAll(() => { + const script = document.createElement('script') + script.src = 'https://unpkg.com/mathlive/dist/mathlive.min.js' + link = document.createElement('link') + link.rel = 'stylesheet' + link.href = 'https://unpkg.com/mathlive/dist/mathlive-fonts.css' + const tasks = [ + new Promise((resolve, reject) => { + script.onload = resolve + script.onerror = reject + }), + new Promise((resolve, reject) => { + link.onload = resolve + link.onerror = reject + }), + ] + document.head.append(script, link) + return Promise.all(tasks) + }) + + afterAll(() => { + link.remove() + }) + + it('should render content from shadow node of custom element', (done) => { + Helper.bootstrap( + 'custom-element/node.html', + 'custom-element/style.css', + 'custom-element/image', + ) + .then(util.delay(1000)) + .then(Helper.renderAndCheck) + .then(util.delay(1000)) + .then(done) + .catch(done) + }) + }) }) describe('work with svg', () => { diff --git a/test/spec/resources/custom-element/image b/test/spec/resources/custom-element/image new file mode 100644 index 00000000..1122c49b --- /dev/null +++ b/test/spec/resources/custom-element/image @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/test/spec/resources/custom-element/node.html b/test/spec/resources/custom-element/node.html new file mode 100644 index 00000000..e0a4aab6 --- /dev/null +++ b/test/spec/resources/custom-element/node.html @@ -0,0 +1,9 @@ +S_{\triangle }=\frac{ab\sin \left(\gamma \right)}{2} + diff --git a/test/spec/resources/custom-element/style.css b/test/spec/resources/custom-element/style.css new file mode 100644 index 00000000..e69de29b