Skip to content

Commit

Permalink
Work on table padding issue #95
Browse files Browse the repository at this point in the history
Changed `getDefaultStyle` to create the element in the sandbox so it is pure (not tainted by CSS / ShadowDOM in the main document).
Never clone SCRIPT elements (that's just asking for trouble).
Removed `imagediff` package as it's not useful anymore.
  • Loading branch information
IDisposable committed Jan 12, 2023
1 parent a29ba3d commit b51f0d0
Show file tree
Hide file tree
Showing 8 changed files with 136 additions and 77 deletions.
1 change: 0 additions & 1 deletion karma.conf.js
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,6 @@ module.exports = function(config) {
},

'test-lib/jquery/dist/jquery.js',
'node_modules/imagediff/imagediff.js',
'test-lib/tesseract-1.0.19.js',

'src/dom-to-image-more.js',
Expand Down
55 changes: 9 additions & 46 deletions package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

1 change: 0 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,6 @@
"grunt-contrib-uglify": "^5.2.2",
"grunt-contrib-watch": "^1.1.0",
"grunt-karma": "^4.0.2",
"imagediff": "^1.0.8",
"js-yaml": "^4.1.0",
"karma": "^6.4.1",
"karma-chai": "^0.1.0",
Expand Down
46 changes: 31 additions & 15 deletions spec/dom-to-image-more.spec.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,6 @@
'use strict';

const assert = global.chai.assert;
const imagediff = global.imagediff;
const domtoimage = global.domtoimage;
const Promise = global.Promise;
const BASE_URL = '/base/spec/resources/';
Expand Down Expand Up @@ -384,20 +383,31 @@
.then(done).catch(done);
});

it('should honor zero-padding table elements', function (done) {
loadTestPage(
'padding/dom-node.html',
'padding/style.css',
'padding/control-image'
)
.then(renderAndCheck)
.then(done)
.catch(done);
});

function compareToControlImage(image, tolerance) {
const control = controlImage();
if (imagediff.equal(image, control, tolerance)) {
assert.isTrue(true, 'rendered and control images should be same');
} else {
// get the data representation so we can update the control images easily
const imageUrl = getImageBase64(image, 'image/png');
const controlUrl = getImageBase64(control, 'image/png');
assert.equal(imageUrl, controlUrl, 'rendered and control images should be same');
if (imageUrl !== controlUrl) {
console.log(` image: ${image.src}`);
console.log(` imageBase64: ${imageUrl}`);
console.log(`controlBase64: ${controlUrl}`);
}

const imageUrl = getImageBase64(image, 'image/png');
const controlUrl = getImageBase64(control, 'image/png');
assert.equal(
imageUrl,
controlUrl,
'rendered and control images should be same'
);
if (imageUrl !== controlUrl) {
console.log(` image: ${image.src}`);
console.log(` imageBase64: ${imageUrl}`);
console.log(`controlBase64: ${controlUrl}`);
}
}

Expand Down Expand Up @@ -474,8 +484,14 @@
it('should parse urls', function () {
const parse = domtoimage.impl.inliner.impl.readUrls;

assert.deepEqual(parse('url("http://acme.com/file")'), ['http://acme.com/file']);
assert.deepEqual(parse('url(foo.com), url(\'bar.org\')'), ['foo.com', 'bar.org']);
assert.deepEqual(parse('url("http://acme.com/file")'), [
'http://acme.com/file',
]);
// eslint-disable-next-line quotes
assert.deepEqual(parse("url(foo.com), url('bar.org')"), [
'foo.com',
'bar.org',
]);
});

it('should ignore data urls', function () {
Expand Down
1 change: 1 addition & 0 deletions spec/resources/padding/control-image
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@

51 changes: 51 additions & 0 deletions spec/resources/padding/dom-node.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
<div id="mywrap" class="wrapper">
<style type="text/css">
.tg {
border-collapse: collapse;
border-spacing: 0;
}

.tg .tg-lqy6 {
text-align: right;
vertical-align: top
}

.tg .tg-ogn4 {
background-color: #cb0000;
text-align: right;
vertical-align: top
}

.tg .tg-zv49 {
background-color: #6200c9;
color: #ffffff;
text-align: right;
vertical-align: top
}

.wrapper {
display: inline-block;
}
</style>
<table class="tg">
<thead>
<tr>
<th class="tg-lqy6"></th>
<th class="tg-lqy6" style="padding-left: 5px"></th>
<th class="tg-ogn4">aaaaa</th>
</tr>
</thead>
<tbody>
<tr>
<td class="tg-lqy6"></td>
<td class="tg-lqy6"></td>
<td class="tg-ogn4">aaaaaa</td>
</tr>
<tr>
<td class="tg-zv49">ssssss</td>
<td class="tg-zv49">sssss</td>
<td class="tg-zv49">aaaaaaa</td>
</tr>
</tbody>
</table>
</div>
17 changes: 17 additions & 0 deletions spec/resources/padding/style.css
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
th,
td {
padding: 0px;
border: 2px solid #585c5c;
}

* {
box-sizing: border-box;
}

table {
border-collapse: collapse;
}

.wrapper {
display: inline-block;
}
41 changes: 27 additions & 14 deletions src/dom-to-image-more.js
Original file line number Diff line number Diff line change
Expand Up @@ -192,7 +192,7 @@
} else {
domtoimage.impl.options.useCredentials = options.useCredentials;
}

if (typeof (options.httpTimeout) === 'undefined') {
domtoimage.impl.options.httpTimeout = defaultOptions.httpTimeout;
} else {
Expand Down Expand Up @@ -236,7 +236,10 @@
}

function cloneNode(node, filter, root, parentComputedStyles, ownerWindow) {
if (!root && filter && !filter(node)) {
// NEVER clone SCRIPT blocks and if not at root, and there's a filter
// ignore anything for which filter returns falsey
if (node.tagName === 'SCRIPT'
|| (!root && filter && !filter(node))) {
return Promise.resolve();
}

Expand All @@ -248,7 +251,7 @@
.then(function (clone) {
return processClone(node, clone);
});

function makeNodeCopy(original) {
return util.isHTMLCanvasElement(original)
? util.makeImage(original.toDataURL())
Expand Down Expand Up @@ -358,7 +361,10 @@

function formatPseudoElementStyle() {
const selector = `.${cloneClassName}:${element}`;
const cssText = style.cssText ? formatCssText() : formatCssProperties();
const cssText = style.cssText
? formatCssText()
: formatCssProperties();

return document.createTextNode(`${selector}{${cssText}}`);

function formatCssText() {
Expand Down Expand Up @@ -878,7 +884,7 @@
const value = node.style.getPropertyValue(propertyName);
const priority = node.style.getPropertyPriority(propertyName);

if(!value) {
if (!value) {
return Promise.resolve();
}

Expand Down Expand Up @@ -921,10 +927,15 @@

util.asArray(sourceComputedStyles).forEach(function (name) {
const sourceValue = sourceComputedStyles.getPropertyValue(name);
const defaultValue = defaultStyle[name];
const parentValue = parentComputedStyles?.getPropertyValue(name);

// If the style does not match the default, or it does not match the parent's, set it. We don't know which
// styles are inherited from the parent and which aren't, so we have to always check both.
if (sourceValue !== defaultStyle[name] ||
(parentComputedStyles && sourceValue !== parentComputedStyles.getPropertyValue(name))) {
if (
sourceValue !== defaultValue ||
(parentComputedStyles && sourceValue !== parentValue)
) {
const priority = sourceComputedStyles.getPropertyPriority(name);
setStyleProperty(targetStyle, name, sourceValue, priority);
}
Expand Down Expand Up @@ -953,32 +964,34 @@
sandbox.contentDocument.head.appendChild(charset);
sandbox.contentDocument.title = 'sandbox';
}
const defaultElement = document.createElement(tagName);
sandbox.contentWindow.document.body.appendChild(defaultElement);
const sandboxWindow = sandbox.contentWindow;
const defaultElement = sandboxWindow.document.createElement(tagName);
sandboxWindow.document.body.appendChild(defaultElement);
// Ensure that there is some content, so that properties like margin are applied.
defaultElement.textContent = '.';
const defaultComputedStyle = sandbox.contentWindow.getComputedStyle(defaultElement);
const defaultComputedStyle = sandboxWindow.getComputedStyle(defaultElement);
const defaultStyle = {};
// Copy styles to an object, making sure that 'width' and 'height' are given the default value of 'auto', since
// their initial value is always 'auto' despite that the default computed value is sometimes an absolute length.
util.asArray(defaultComputedStyle).forEach(function (name) {
defaultStyle[name] =
(name === 'width' || name === 'height') ? 'auto' : defaultComputedStyle.getPropertyValue(name);
});
sandbox.contentWindow.document.body.removeChild(defaultElement);
sandboxWindow.document.body.removeChild(defaultElement);
tagNameDefaultStyles[tagName] = defaultStyle;
return defaultStyle;
}

function removeSandbox() {
if (!sandbox) {
return;
document.body.removeChild(sandbox);
sandbox = null;
}
document.body.removeChild(sandbox);
sandbox = null;

if (removeDefaultStylesTimeoutId) {
clearTimeout(removeDefaultStylesTimeoutId);
}

removeDefaultStylesTimeoutId = setTimeout(() => {
removeDefaultStylesTimeoutId = null;
tagNameDefaultStyles = {};
Expand Down

0 comments on commit b51f0d0

Please sign in to comment.