From 8999c7618184e1564d19d9c367a488c22c0d65a3 Mon Sep 17 00:00:00 2001 From: Niklas von Hertzen Date: Sun, 13 Aug 2017 23:27:03 +0800 Subject: [PATCH] Fix iOS 10.3 base64 image tainting canvas (Fix #1151) --- src/Feature.js | 39 +++++++++++++++++++++++++++++ src/ImageLoader.js | 39 ++++++++++++++++++----------- tests/reftests/images/svg/node.html | 3 +++ tests/testrunner.js | 8 ++++-- 4 files changed, 73 insertions(+), 16 deletions(-) diff --git a/src/Feature.js b/src/Feature.js index ff3dd48..1b467f2 100644 --- a/src/Feature.js +++ b/src/Feature.js @@ -25,6 +25,38 @@ const testRangeBounds = document => { return false; }; +// iOS 10.3 taints canvas with base64 images unless crossOrigin = 'anonymous' +const testBase64 = (document: Document): Promise => { + 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 img = new Image(); const canvas = document.createElement('canvas'); @@ -54,6 +86,13 @@ const FEATURES = { const value = testSVG(document); Object.defineProperty(FEATURES, 'SUPPORT_SVG_DRAWING', {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; } }; diff --git a/src/ImageLoader.js b/src/ImageLoader.js index d1aeee3..3a6736a 100644 --- a/src/ImageLoader.js +++ b/src/ImageLoader.js @@ -59,19 +59,30 @@ export default class ImageLoader { if (__DEV__) { 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; - img.src = src; - if (img.complete === true) { - // Inline XML images may fail to parse, throwing an Error later on - setTimeout(() => { - resolve(img); - }, 500); - } - }); + const imageLoadHandler = (supportsDataImages: boolean): Promise => { + return new Promise((resolve, reject) => { + const img = new Image(); + img.onload = () => resolve(img); + //ios safari 10.3 taints canvas with data urls unless crossOrigin is set to anonymous + if (!supportsDataImages) { + 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; } @@ -122,9 +133,9 @@ export class ImageStore { } 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 => src.substr(-3).toLowerCase() === 'svg' || INLINE_SVG.test(src); diff --git a/tests/reftests/images/svg/node.html b/tests/reftests/images/svg/node.html index f5e5469..8db5433 100644 --- a/tests/reftests/images/svg/node.html +++ b/tests/reftests/images/svg/node.html @@ -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"/> + ABCDEABCDE + + ABCABC123123 diff --git a/tests/testrunner.js b/tests/testrunner.js index c13c026..cc61727 100644 --- a/tests/testrunner.js +++ b/tests/testrunner.js @@ -95,14 +95,18 @@ const assertPath = (result, expected, desc) => { testContainer.src = url + '?selenium&run=false&reftest&' + Math.random(); if (hasHistoryApi) { // 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); }); after(() => { if (hasHistoryApi) { - history.replaceState(null, '', testRunnerUrl); + try { + history.replaceState(null, '', testRunnerUrl); + } catch (e) {} } document.body.removeChild(testContainer); });