Skip to content

Commit

Permalink
[Enhancement] Support Cloning Custom Elements (#129)
Browse files Browse the repository at this point in the history
* feat(custom-elements): ✨ obtain children of shadow node for custom elements

Inspired by tsayen/dom-to-image#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
  • Loading branch information
ShaMan123 committed May 17, 2021
1 parent 50ed80e commit b93825e
Show file tree
Hide file tree
Showing 6 changed files with 55 additions and 3 deletions.
6 changes: 4 additions & 2 deletions src/cloneNode.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,9 @@ async function cloneChildren(
clonedNode: HTMLElement,
filter?: Function,
): Promise<HTMLElement> {
const children = toArray<HTMLElement>(nativeNode.childNodes)
const children = toArray<HTMLElement>(
(nativeNode.shadowRoot ?? nativeNode).childNodes,
)
if (children.length === 0) {
return Promise.resolve(clonedNode)
}
Expand Down Expand Up @@ -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 {
Expand Down
2 changes: 1 addition & 1 deletion src/embedImages.ts
Original file line number Diff line number Diff line change
Expand Up @@ -21,7 +21,7 @@ async function embedBackground(
clonedNode: HTMLElement,
options: Options,
): Promise<HTMLElement> {
const background = clonedNode.style.getPropertyValue('background')
const background = clonedNode.style?.getPropertyValue('background')
if (!background) {
return Promise.resolve(clonedNode)
}
Expand Down
40 changes: 40 additions & 0 deletions test/spec/index.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -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', () => {
Expand Down
1 change: 1 addition & 0 deletions test/spec/resources/custom-element/image
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@

9 changes: 9 additions & 0 deletions test/spec/resources/custom-element/node.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<math-field
value="S_{\triangle }=\frac{ab\sin \left(\gamma \right)}{2}"
scrollingelement="[object HTMLHtmlElement]"
class="math no-touch"
role="textbox"
tabindex="0"
read-only=""
>S_{\triangle }=\frac{ab\sin \left(\gamma \right)}{2}
</math-field>
Empty file.

0 comments on commit b93825e

Please sign in to comment.