diff --git a/dist/dom-to-image-more.min.js b/dist/dom-to-image-more.min.js index 90384d15..2544ce5c 100644 --- a/dist/dom-to-image-more.min.js +++ b/dist/dom-to-image-more.min.js @@ -1,3 +1,3 @@ -/*! dom-to-image-more 15-01-2023 */ -!function(n){"use strict";const a=function(){let e=0;return{escape:function(e){return e.replace(/([.*+?^${}()|[]\/\\])/g,"\\$1")},isDataUrl:function(e){return-1!==e.search(/^(data:)/)},canvasToBlob:function(t){if(t.toBlob)return new Promise(function(e){t.toBlob(e)});return function(o){return new Promise(function(e){var t=c(o.toDataURL().split(",")[1]),n=t.length,r=new Uint8Array(n);for(let e=0;et.style.removeProperty(e)),["left","right","top","bottom"].forEach(e=>{t.style.getPropertyValue(e)&&t.style.setProperty(e,"0px")})))}e(c,s)}function t(){const l=a.uid();function t(o){const i=f(c,o),u=i.getPropertyValue("content");if(""!==u&&"none"!==u){const t=s.getAttribute("class")||"",n=(s.setAttribute("class",t+" "+l),document.createElement("style"));function e(){const e=`.${l}:`+o,t=(i.cssText?n:r)();return document.createTextNode(e+`{${t}}`);function n(){return`${i.cssText} content: ${u};`}function r(){const e=a.asArray(i).map(t).join("; ");return e+";";function t(e){const t=i.getPropertyValue(e),n=i.getPropertyPriority(e)?" !important":"";return e+": "+t+n}}}n.appendChild(e()),s.appendChild(n)}}[":before",":after"].forEach(function(e){t(e)})}function n(){a.isHTMLTextAreaElement(c)&&(s.innerHTML=c.value),a.isHTMLInputElement(c)&&s.setAttribute("value",c.value)}function r(){a.isSVGElement(s)&&(s.setAttribute("xmlns","http://www.w3.org/2000/svg"),a.isSVGRectElement(s))&&["width","height"].forEach(function(e){const t=s.getAttribute(e);t&&s.style.setProperty(e,t)})}}}(e,o.filter,null,t)}).then(d).then(p).then(function(t){o.bgcolor&&(t.style.backgroundColor=o.bgcolor);o.width&&(t.style.width=o.width+"px");o.height&&(t.style.height=o.height+"px");o.style&&Object.keys(o.style).forEach(function(e){t.style[e]=o.style[e]});let e=null;"function"==typeof o.onclone&&(e=o.onclone(t));return Promise.resolve(e).then(function(){return t})}).then(function(e){return e=e,t=o.width||a.width(r),n=o.height||a.height(r),Promise.resolve(e).then(function(e){return e.setAttribute("xmlns","http://www.w3.org/1999/xhtml"),(new XMLSerializer).serializeToString(e)}).then(a.escapeXhtml).then(function(e){return`${e}`}).then(function(e){return`${e}`}).then(function(e){return"data:image/svg+xml;charset=utf-8,"+e});var t,n}).then(function(e){return l.impl.urlCache=[],e})}function h(o,i){return s(o,i=i||{}).then(a.makeImage).then(function(e){var t="number"!=typeof i.scale?1:i.scale,n=function(e,t){var n=document.createElement("canvas");n.width=(i.width||a.width(e))*t,n.height=(i.height||a.height(e))*t,i.bgcolor&&((e=n.getContext("2d")).fillStyle=i.bgcolor,e.fillRect(0,0,n.width,n.height));return n}(o,t),r=n.getContext("2d");return r.mozImageSmoothingEnabled=!1,r.msImageSmoothingEnabled=!1,r.imageSmoothingEnabled=!1,e&&(r.scale(t,t),r.drawImage(e,0,0)),m&&(document.body.removeChild(m),m=null),y&&clearTimeout(y),y=setTimeout(()=>{y=null,v={}},2e4),n})}let m=null;function d(n){return e.resolveAll().then(function(e){var t;return""!==e&&(t=document.createElement("style"),n.appendChild(t),t.appendChild(document.createTextNode(e))),n})}function p(e){return r.inlineAll(e).then(function(){return e})}function g(i,u,e){const l=function(e){if(v[e])return v[e];{var t;m||((m=document.createElement("iframe")).style.visibility="hidden",m.style.position="fixed",document.body.appendChild(m),(t=document.createElement("meta")).setAttribute("charset",document.characterSet||"UTF-8"),m.contentDocument.head.appendChild(t),m.contentDocument.title="sandbox")}const n=m.contentWindow,r=n.document,o=r.createElement(e),i=(n.document.body.appendChild(o),o.textContent="​",n.getComputedStyle(o)),u={};return a.asArray(i).forEach(function(e){u[e]="width"===e||"height"===e?"auto":i.getPropertyValue(e)}),r.body.removeChild(o),v[e]=u}(e.tagName),c=e.style;a.asArray(i).forEach(function(e){var t,n=i.getPropertyValue(e),r=l[e],o=u?u.getPropertyValue(e):void 0;(n!==r||u&&n!==o)&&(r=i.getPropertyPriority(e),o=c,n=n,r=r,t=0<=["background-clip"].indexOf(e=e),r?(o.setProperty(e,n,r),t&&o.setProperty("-webkit-"+e,n,r)):(o.setProperty(e,n),t&&o.setProperty("-webkit-"+e,n)))})}let y=null,v={}}(this); +/*! dom-to-image-more 18-01-2023 */ +!function(n){"use strict";const s=function(){let e=0;return{escape:function(e){return e.replace(/([.*+?^${}()|[]\/\\])/g,"\\$1")},isDataUrl:function(e){return-1!==e.search(/^(data:)/)},canvasToBlob:function(t){if(t.toBlob)return new Promise(function(e){t.toBlob(e)});return function(o){return new Promise(function(e){var t=c(o.toDataURL().split(",")[1]),n=t.length,r=new Uint8Array(n);for(let e=0;et.style.removeProperty(e)),["left","right","top","bottom"].forEach(e=>{t.style.getPropertyValue(e)&&t.style.setProperty(e,"0px")})))}e(c,a)}function t(){const l=s.uid();function t(o){const i=f(c,o),u=i.getPropertyValue("content");if(""!==u&&"none"!==u){const t=a.getAttribute("class")||"",n=(a.setAttribute("class",t+" "+l),document.createElement("style"));function e(){const e=`.${l}:`+o,t=(i.cssText?n:r)();return document.createTextNode(e+`{${t}}`);function n(){return`${i.cssText} content: ${u};`}function r(){const e=s.asArray(i).map(t).join("; ");return e+";";function t(e){const t=i.getPropertyValue(e),n=i.getPropertyPriority(e)?" !important":"";return e+": "+t+n}}}n.appendChild(e()),a.appendChild(n)}}[":before",":after"].forEach(function(e){t(e)})}function n(){s.isHTMLTextAreaElement(c)&&(a.innerHTML=c.value),s.isHTMLInputElement(c)&&a.setAttribute("value",c.value)}function r(){s.isSVGElement(a)&&(a.setAttribute("xmlns","http://www.w3.org/2000/svg"),s.isSVGRectElement(a))&&["width","height"].forEach(function(e){const t=a.getAttribute(e);t&&a.style.setProperty(e,t)})}}}(e,o.filter,null,t)}).then(d).then(p).then(function(t){o.bgcolor&&(t.style.backgroundColor=o.bgcolor);o.width&&(t.style.width=o.width+"px");o.height&&(t.style.height=o.height+"px");o.style&&Object.keys(o.style).forEach(function(e){t.style[e]=o.style[e]});let e=null;"function"==typeof o.onclone&&(e=o.onclone(t));return Promise.resolve(e).then(function(){return t})}).then(function(e){return e=e,t=o.width||s.width(r),n=o.height||s.height(r),Promise.resolve(e).then(function(e){return e.setAttribute("xmlns","http://www.w3.org/1999/xhtml"),(new XMLSerializer).serializeToString(e)}).then(s.escapeXhtml).then(function(e){return`${e}`}).then(function(e){return`${e}`}).then(function(e){return"data:image/svg+xml;charset=utf-8,"+e});var t,n}).then(function(e){return l.impl.urlCache=[],e})}function h(o,i){return a(o,i=i||{}).then(s.makeImage).then(function(e){var t="number"!=typeof i.scale?1:i.scale,n=function(e,t){var n=document.createElement("canvas");n.width=(i.width||s.width(e))*t,n.height=(i.height||s.height(e))*t,i.bgcolor&&((e=n.getContext("2d")).fillStyle=i.bgcolor,e.fillRect(0,0,n.width,n.height));return n}(o,t),r=n.getContext("2d");return r.mozImageSmoothingEnabled=!1,r.msImageSmoothingEnabled=!1,r.imageSmoothingEnabled=!1,e&&(r.scale(t,t),r.drawImage(e,0,0)),m&&(document.body.removeChild(m),m=null),y&&clearTimeout(y),y=setTimeout(()=>{y=null,v={}},2e4),n})}let m=null;function d(n){return e.resolveAll().then(function(e){var t;return""!==e&&(t=document.createElement("style"),n.appendChild(t),t.appendChild(document.createTextNode(e))),n})}function p(e){return r.inlineAll(e).then(function(){return e})}function g(e,i,u,t){const l=function(e){var e=function(e){var t=[];do{if(1===e.nodeType){var n=e.tagName;if(t.push(n),E.includes(n))break}}while(e=e.parentNode,e);return t}(e),t=e.join(">");if(v[t])return v[t];var n=function(){{var e;m||((m=document.createElement("iframe")).style.visibility="hidden",m.style.position="fixed",document.body.appendChild(m),(e=document.createElement("meta")).setAttribute("charset",document.characterSet||"UTF-8"),m.contentDocument.head.appendChild(e),m.contentDocument.title="sandbox")}return m.contentWindow}(),e=function(e,t){let n=e.body;do{var r=t.pop(),r=e.createElement(r);n.appendChild(r),n=r}while(0 renderToSvg(node, { styleCaching: 'strict' }))]) + .then( + (promises) => promises[0].value + ); + let two = Promise.allSettled([ + loadTestPage('padding/dom-node.html', 'padding/style.css', 'padding/control-image') + .then((node) => renderToSvg(node, { styleCaching: 'relaxed' }))]) + .then( + (promises) => promises[0].value + ); + + Promise.allSettled([one, two]) + .then(function (promises) { + const strict = promises[0].value; + const relaxed = promises[1].value; + if (strict !== relaxed) { + console.log(`\n\nstrict: ${strict}\n\nrelaxed: ${relaxed}\n\n`); + } + assert.equal(strict, relaxed, 'SVG rendered be same'); + }) + .then(done) + .catch(done); + }); + }); + function loadTestPage(html, css, controlImage) { return loadPage() - .then(function () { - return getResource(html).then(function (html) { - $('#dom-node').html(html); - }); + .then(function (document) { + if (!html) + return document; + + return getResource(html) + .then(function (html) { + $('#dom-node').html(html); + return document; + }); }) - .then(function () { - if (css) { - return getResource(css).then(function (css) { + .then(function (document) { + if (!css) + return document; + + return getResource(css) + .then(function (css) { $('#style').append(document.createTextNode(css)); + return document; }); - } }) - .then(function () { - if (controlImage) { - return getResource(controlImage).then(function (image) { + .then(function (document) { + if (!controlImage) + return document; + + return getResource(controlImage) + .then(function (image) { $('#control-image').attr('src', image); + return document; }); - } }); } function loadPage() { - return getResource('page.html').then(function (html) { - const root = document.createElement('div'); - root.id = 'test-root'; - root.innerHTML = html; - document.body.appendChild(root); - }); + return getResource('page.html') + .then(function (html) { + const root = document.createElement('div'); + root.id = 'test-root'; + root.innerHTML = html; + document.body.appendChild(root); + return document; + }); } function purgePage() { @@ -925,5 +991,19 @@ request.send(); }); } + + function renderToPng() { + return domtoimage.toPng(domNode(), { onclone: cloneCatcher }); + } + + function renderToSvg(options) { + const debugOptions = { onclone: cloneCatcher, debugCache: true }; + return domtoimage.toSvg(domNode(), Object.assign(debugOptions, options)); + } + + function cloneCatcher(clone) { + clonedNode().replaceChildren(clone); + return clone; + } }); })(this); diff --git a/spec/resources/math/control-image b/spec/resources/math/control-image new file mode 100644 index 00000000..b76adb26 --- /dev/null +++ b/spec/resources/math/control-image @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/spec/resources/math/dom-node.html b/spec/resources/math/dom-node.html new file mode 100644 index 00000000..1067834d --- /dev/null +++ b/spec/resources/math/dom-node.html @@ -0,0 +1,34 @@ +
+ The infinite sum + + + + + n + = + 1 + + + + + + + + + 1 + + n + 2 + + + + +
Is equal to the real number + + + π + 2 + + 6 + + +
\ No newline at end of file diff --git a/spec/resources/padding/control-image b/spec/resources/padding/control-image index f5f53a4c..3082dd49 100644 --- a/spec/resources/padding/control-image +++ b/spec/resources/padding/control-image @@ -1 +1 @@ - \ No newline at end of file + \ No newline at end of file diff --git a/spec/resources/scroll/control-image b/spec/resources/scroll/control-image index 77e7950b..5918844f 100644 --- a/spec/resources/scroll/control-image +++ b/spec/resources/scroll/control-image @@ -1 +1 @@ - + \ No newline at end of file diff --git a/spec/resources/test.html b/spec/resources/test.html deleted file mode 100644 index b6b3734d..00000000 --- a/spec/resources/test.html +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - - - -
Image: - -
Control: - -
diff --git a/src/dom-to-image-more.js b/src/dom-to-image-more.js index 9709904c..cdc6e7c6 100644 --- a/src/dom-to-image-more.js +++ b/src/dom-to-image-more.js @@ -16,6 +16,8 @@ useCredentials: false, // Default resolve timeout httpTimeout: 30000, + // Style computation cache tag rules (options are strict, relaxed) + styleCaching: 'strict', }; const domtoimage = { @@ -62,6 +64,7 @@ * @param {Number} options.scale - a Number multiplier to scale up the canvas before rendering to reduce fuzzy images, defaults to 1.0. * @param {String} options.imagePlaceholder - dataURL to use as a placeholder for failed images, default behaviour is to fail fast on images we can't fetch * @param {Boolean} options.cacheBust - set to true to cache bust by appending the time to the request url + * @param {String} options.styleCaching - set to 'strict', 'relaxed' to select style caching rules * @return {Promise} - A promise that is fulfilled with a SVG image data URL * */ function toSvg(node, options) { @@ -70,7 +73,7 @@ copyOptions(options); return Promise.resolve(node) .then(function (clonee) { - return cloneNode(clonee, options.filter, null, ownerWindow); + return cloneNode(clonee, options, null, ownerWindow); }) .then(embedFonts) .then(inlineImages) @@ -99,7 +102,6 @@ if (options.height) { clone.style.height = `${options.height}px`; } - if (options.style) { Object.keys(options.style).forEach(function (property) { clone.style[property] = options.style[property]; @@ -199,6 +201,12 @@ } else { domtoimage.impl.options.httpTimeout = options.httpTimeout; } + + if (typeof options.styleCaching === 'undefined') { + domtoimage.impl.options.styleCaching = defaultOptions.styleCaching; + } else { + domtoimage.impl.options.styleCaching = options.styleCaching; + } } function draw(domNode, options) { @@ -237,7 +245,8 @@ let sandbox = null; - function cloneNode(node, filter, parentComputedStyles, ownerWindow) { + function cloneNode(node, options, parentComputedStyles, ownerWindow) { + const filter = options.filter; if ( node === sandbox || (parentComputedStyles !== null && filter && !filter(node)) @@ -271,7 +280,7 @@ done = done.then(function () { return cloneNode( originalChild, - filter, + options, originalComputedStyles, ownerWindow ).then(function (clonedChild) { @@ -329,6 +338,8 @@ copyFont(sourceComputedStyles, targetElement.style); // here we re-assign the font props. } else { copyUserComputedStyleFast( + options, + sourceElement, sourceComputedStyles, parentComputedStyles, targetElement @@ -720,6 +731,7 @@ function width(node) { var width = px(node, 'width'); + if (isNaN(width)) { const leftBorder = px(node, 'border-left-width'); const rightBorder = px(node, 'border-right-width'); @@ -730,6 +742,7 @@ function height(node) { var height = px(node, 'height'); + if (isNaN(height)) { const topBorder = px(node, 'border-top-width'); const bottomBorder = px(node, 'border-bottom-width'); @@ -870,8 +883,8 @@ cssRules.push.bind(cssRules) ); } catch (e) { - console.log( - `Error while reading CSS rules from ${sheet.href}`, + console.error( + `domtoimage: Error while reading CSS rules from ${sheet.href}`, e.toString() ); } @@ -981,11 +994,13 @@ } function copyUserComputedStyleFast( + options, + sourceElement, sourceComputedStyles, parentComputedStyles, targetElement ) { - const defaultStyle = getDefaultStyle(targetElement.tagName); + const defaultStyle = getDefaultStyle(options, sourceElement); const targetStyle = targetElement.style; util.asArray(sourceComputedStyles).forEach(function (name) { @@ -1010,45 +1025,158 @@ let removeDefaultStylesTimeoutId = null; let tagNameDefaultStyles = {}; - function getDefaultStyle(tagName) { - if (tagNameDefaultStyles[tagName]) { - return tagNameDefaultStyles[tagName]; + const ascentStoppers = [ + // these come from https://developer.mozilla.org/en-US/docs/Web/HTML/Block-level_elements + 'ADDRESS', + 'ARTICLE', + 'ASIDE', + 'BLOCKQUOTE', + 'DETAILS', + 'DIALOG', + 'DD', + 'DIV', + 'DL', + 'DT', + 'FIELDSET', + 'FIGCAPTION', + 'FIGURE', + 'FOOTER', + 'FORM', + 'H1', + 'H2', + 'H3', + 'H4', + 'H5', + 'H6', + 'HEADER', + 'HGROUP', + 'HR', + 'LI', + 'MAIN', + 'NAV', + 'OL', + 'P', + 'PRE', + 'SECTION', + 'TABLE', + 'UL', + // this is some non-standard ones + 'math', // intentionally lowercase, thanks Safari + // these are ultimate stoppers in case something drastic changes in how the DOM works + 'SVG', + 'BODY', + 'HEAD', + 'HTML', + ]; + + function getDefaultStyle(options, sourceElement) { + const tagHierarchy = computeTagHierarchy(sourceElement); + const tagKey = computeTagKey(tagHierarchy); + if (tagNameDefaultStyles[tagKey]) { + return tagNameDefaultStyles[tagKey]; } - if (!sandbox) { - // Create a hidden sandbox