mirror of
https://github.com/niklasvh/html2canvas.git
synced 2023-08-10 21:13:10 +03:00
Fix iOS 10.3 base64 image tainting canvas (Fix #1151)
This commit is contained in:
parent
fd1447a6e7
commit
8999c76181
@ -25,6 +25,38 @@ const testRangeBounds = document => {
|
|||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
// iOS 10.3 taints canvas with base64 images unless crossOrigin = 'anonymous'
|
||||||
|
const testBase64 = (document: Document): Promise<boolean> => {
|
||||||
|
const img = new Image();
|
||||||
|
const canvas = document.createElement('canvas');
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
|
||||||
|
return new Promise(resolve => {
|
||||||
|
// Single pixel base64 image renders fine on iOS 10.3???
|
||||||
|
// TODO add a smaller base64 image that still fails on iOS 10.3
|
||||||
|
img.src = "";
|
||||||
|
|
||||||
|
const onload = () => {
|
||||||
|
try {
|
||||||
|
ctx.drawImage(img, 0, 0);
|
||||||
|
canvas.toDataURL();
|
||||||
|
} catch (e) {
|
||||||
|
return resolve(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
return resolve(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
img.onload = onload;
|
||||||
|
|
||||||
|
if (img.complete === true) {
|
||||||
|
setTimeout(() => {
|
||||||
|
onload();
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
const testSVG = document => {
|
const testSVG = document => {
|
||||||
const img = new Image();
|
const img = new Image();
|
||||||
const canvas = document.createElement('canvas');
|
const canvas = document.createElement('canvas');
|
||||||
@ -54,6 +86,13 @@ const FEATURES = {
|
|||||||
const value = testSVG(document);
|
const value = testSVG(document);
|
||||||
Object.defineProperty(FEATURES, 'SUPPORT_SVG_DRAWING', {value});
|
Object.defineProperty(FEATURES, 'SUPPORT_SVG_DRAWING', {value});
|
||||||
return value;
|
return value;
|
||||||
|
},
|
||||||
|
// $FlowFixMe - get/set properties not yet supported
|
||||||
|
get SUPPORT_BASE64_DRAWING() {
|
||||||
|
'use strict';
|
||||||
|
const value = testBase64(document);
|
||||||
|
Object.defineProperty(FEATURES, 'SUPPORT_BASE64_DRAWING', {value});
|
||||||
|
return value;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -59,19 +59,30 @@ export default class ImageLoader {
|
|||||||
if (__DEV__) {
|
if (__DEV__) {
|
||||||
this.logger.log(`Added image ${key.substring(0, 256)}`);
|
this.logger.log(`Added image ${key.substring(0, 256)}`);
|
||||||
}
|
}
|
||||||
this.cache[key] = new Promise((resolve, reject) => {
|
|
||||||
const img = new Image();
|
|
||||||
img.onload = () => resolve(img);
|
|
||||||
|
|
||||||
img.onerror = reject;
|
const imageLoadHandler = (supportsDataImages: boolean): Promise<Image> => {
|
||||||
img.src = src;
|
return new Promise((resolve, reject) => {
|
||||||
if (img.complete === true) {
|
const img = new Image();
|
||||||
// Inline XML images may fail to parse, throwing an Error later on
|
img.onload = () => resolve(img);
|
||||||
setTimeout(() => {
|
//ios safari 10.3 taints canvas with data urls unless crossOrigin is set to anonymous
|
||||||
resolve(img);
|
if (!supportsDataImages) {
|
||||||
}, 500);
|
img.crossOrigin = 'anonymous';
|
||||||
}
|
}
|
||||||
});
|
|
||||||
|
img.onerror = reject;
|
||||||
|
img.src = src;
|
||||||
|
if (img.complete === true) {
|
||||||
|
// Inline XML images may fail to parse, throwing an Error later on
|
||||||
|
setTimeout(() => {
|
||||||
|
resolve(img);
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
this.cache[key] = isInlineImage(src)
|
||||||
|
? FEATURES.SUPPORT_BASE64_DRAWING.then(imageLoadHandler)
|
||||||
|
: imageLoadHandler(true);
|
||||||
return key;
|
return key;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -122,9 +133,9 @@ export class ImageStore {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const INLINE_SVG = /^data:image\/svg\+xml/i;
|
const INLINE_SVG = /^data:image\/svg\+xml/i;
|
||||||
const INLINE_IMG = /^data:image\/.*;base64,/i;
|
const INLINE_BASE64 = /^data:image\/.*;base64,/i;
|
||||||
|
|
||||||
const isInlineImage = (src: string): boolean => INLINE_IMG.test(src);
|
const isInlineImage = (src: string): boolean => INLINE_BASE64.test(src);
|
||||||
|
|
||||||
const isSVG = (src: string): boolean =>
|
const isSVG = (src: string): boolean =>
|
||||||
src.substr(-3).toLowerCase() === 'svg' || INLINE_SVG.test(src);
|
src.substr(-3).toLowerCase() === 'svg' || INLINE_SVG.test(src);
|
||||||
|
@ -20,6 +20,9 @@
|
|||||||
style="fill:#40aa54;fill-opacity:1;stroke:#20552a;stroke-width:7.99999952;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"/>
|
style="fill:#40aa54;fill-opacity:1;stroke:#20552a;stroke-width:7.99999952;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"/>
|
||||||
</g>
|
</g>
|
||||||
</svg>
|
</svg>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="88" height="60" viewBox="0 0 88 60"><text xmlns="http://www.w3.org/2000/svg" font-family="Verdana" font-weight="700" text-anchor="middle" x="46" y="18">ABCDE</text><text xmlns="http://www.w3.org/2000/svg" font-family="Verdana" font-weight="700" text-anchor="middle" x="44" y="16" fill="white">ABCDE</text></svg>
|
||||||
|
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" width="88" height="60" viewBox="0 0 88 60"><text xmlns="http://www.w3.org/2000/svg" font-family="Verdana" font-size="12px" font-weight="700" text-anchor="middle"><tspan fill="black" x="46" y="18">ABC</tspan><tspan fill="white" x="44" y="16">ABC</tspan><tspan fill="black" x="46" y="35">123</tspan><tspan fill="white" x="44" y="33">123</tspan></text></svg>
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -95,14 +95,18 @@ const assertPath = (result, expected, desc) => {
|
|||||||
testContainer.src = url + '?selenium&run=false&reftest&' + Math.random();
|
testContainer.src = url + '?selenium&run=false&reftest&' + Math.random();
|
||||||
if (hasHistoryApi) {
|
if (hasHistoryApi) {
|
||||||
// Chrome does not resolve relative background urls correctly inside of a nested iframe
|
// Chrome does not resolve relative background urls correctly inside of a nested iframe
|
||||||
history.replaceState(null, '', url);
|
try {
|
||||||
|
history.replaceState(null, '', url);
|
||||||
|
} catch (e) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
document.body.appendChild(testContainer);
|
document.body.appendChild(testContainer);
|
||||||
});
|
});
|
||||||
after(() => {
|
after(() => {
|
||||||
if (hasHistoryApi) {
|
if (hasHistoryApi) {
|
||||||
history.replaceState(null, '', testRunnerUrl);
|
try {
|
||||||
|
history.replaceState(null, '', testRunnerUrl);
|
||||||
|
} catch (e) {}
|
||||||
}
|
}
|
||||||
document.body.removeChild(testContainer);
|
document.body.removeChild(testContainer);
|
||||||
});
|
});
|
||||||
|
Loading…
Reference in New Issue
Block a user