Fix iOS 10.3 base64 image tainting canvas (Fix #1151)

This commit is contained in:
Niklas von Hertzen 2017-08-13 23:27:03 +08:00
parent fd1447a6e7
commit 8999c76181
4 changed files with 73 additions and 16 deletions

View File

@ -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;
} }
}; };

View File

@ -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);

View File

@ -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>

View File

@ -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);
}); });