mirror of
https://github.com/niklasvh/html2canvas.git
synced 2023-08-10 21:13:10 +03:00
180 lines
5.8 KiB
JavaScript
180 lines
5.8 KiB
JavaScript
/* @flow */
|
|
'use strict';
|
|
|
|
import {createForeignObjectSVG, loadSerializedSVG} from './renderer/ForeignObjectRenderer';
|
|
|
|
const testRangeBounds = document => {
|
|
const TEST_HEIGHT = 123;
|
|
|
|
if (document.createRange) {
|
|
const range = document.createRange();
|
|
if (range.getBoundingClientRect) {
|
|
const testElement = document.createElement('boundtest');
|
|
testElement.style.height = `${TEST_HEIGHT}px`;
|
|
testElement.style.display = 'block';
|
|
document.body.appendChild(testElement);
|
|
|
|
range.selectNode(testElement);
|
|
const rangeBounds = range.getBoundingClientRect();
|
|
const rangeHeight = Math.round(rangeBounds.height);
|
|
document.body.removeChild(testElement);
|
|
if (rangeHeight === TEST_HEIGHT) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
};
|
|
|
|
// iOS 10.3 taints canvas with base64 images unless crossOrigin = 'anonymous'
|
|
const testBase64 = (document: Document, src: string): 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???
|
|
img.src = src;
|
|
|
|
const onload = () => {
|
|
try {
|
|
ctx.drawImage(img, 0, 0);
|
|
canvas.toDataURL();
|
|
} catch (e) {
|
|
return resolve(false);
|
|
}
|
|
|
|
return resolve(true);
|
|
};
|
|
|
|
img.onload = onload;
|
|
img.onerror = () => resolve(false);
|
|
|
|
if (img.complete === true) {
|
|
setTimeout(() => {
|
|
onload();
|
|
}, 500);
|
|
}
|
|
});
|
|
};
|
|
|
|
const testCORS = () => typeof new Image().crossOrigin !== 'undefined';
|
|
|
|
const testResponseType = () => typeof new XMLHttpRequest().responseType === 'string';
|
|
|
|
const testSVG = document => {
|
|
const img = new Image();
|
|
const canvas = document.createElement('canvas');
|
|
const ctx = canvas.getContext('2d');
|
|
img.src = `data:image/svg+xml,<svg xmlns='http://www.w3.org/2000/svg'></svg>`;
|
|
|
|
try {
|
|
ctx.drawImage(img, 0, 0);
|
|
canvas.toDataURL();
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
return true;
|
|
};
|
|
|
|
const isGreenPixel = data => data[0] === 0 && data[1] === 255 && data[2] === 0 && data[3] === 255;
|
|
|
|
const testForeignObject = document => {
|
|
const canvas = document.createElement('canvas');
|
|
const size = 100;
|
|
canvas.width = size;
|
|
canvas.height = size;
|
|
const ctx = canvas.getContext('2d');
|
|
ctx.fillStyle = 'rgb(0, 255, 0)';
|
|
ctx.fillRect(0, 0, size, size);
|
|
|
|
const img = new Image();
|
|
const greenImageSrc = canvas.toDataURL();
|
|
img.src = greenImageSrc;
|
|
const svg = createForeignObjectSVG(size, size, 0, 0, img);
|
|
ctx.fillStyle = 'red';
|
|
ctx.fillRect(0, 0, size, size);
|
|
|
|
return loadSerializedSVG(svg)
|
|
.then(img => {
|
|
ctx.drawImage(img, 0, 0);
|
|
const data = ctx.getImageData(0, 0, size, size).data;
|
|
ctx.fillStyle = 'red';
|
|
ctx.fillRect(0, 0, size, size);
|
|
|
|
const node = document.createElement('div');
|
|
node.style.backgroundImage = `url(${greenImageSrc})`;
|
|
node.style.height = `${size}px`;
|
|
// Firefox 55 does not render inline <img /> tags
|
|
return isGreenPixel(data)
|
|
? loadSerializedSVG(createForeignObjectSVG(size, size, 0, 0, node))
|
|
: Promise.reject(false);
|
|
})
|
|
.then(img => {
|
|
ctx.drawImage(img, 0, 0);
|
|
// Edge does not render background-images
|
|
return isGreenPixel(ctx.getImageData(0, 0, size, size).data);
|
|
})
|
|
.catch(e => false);
|
|
};
|
|
|
|
const FEATURES = {
|
|
// $FlowFixMe - get/set properties not yet supported
|
|
get SUPPORT_RANGE_BOUNDS() {
|
|
'use strict';
|
|
const value = testRangeBounds(document);
|
|
Object.defineProperty(FEATURES, 'SUPPORT_RANGE_BOUNDS', {value});
|
|
return value;
|
|
},
|
|
// $FlowFixMe - get/set properties not yet supported
|
|
get SUPPORT_SVG_DRAWING() {
|
|
'use strict';
|
|
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';
|
|
return (src: string) => {
|
|
const value = testBase64(document, src);
|
|
Object.defineProperty(FEATURES, 'SUPPORT_BASE64_DRAWING', {value: () => value});
|
|
return value;
|
|
};
|
|
},
|
|
// $FlowFixMe - get/set properties not yet supported
|
|
get SUPPORT_FOREIGNOBJECT_DRAWING() {
|
|
'use strict';
|
|
const value =
|
|
typeof Array.from === 'function' && typeof window.fetch === 'function'
|
|
? testForeignObject(document)
|
|
: Promise.resolve(false);
|
|
Object.defineProperty(FEATURES, 'SUPPORT_FOREIGNOBJECT_DRAWING', {value});
|
|
return value;
|
|
},
|
|
// $FlowFixMe - get/set properties not yet supported
|
|
get SUPPORT_CORS_IMAGES() {
|
|
'use strict';
|
|
const value = testCORS();
|
|
Object.defineProperty(FEATURES, 'SUPPORT_CORS_IMAGES', {value});
|
|
return value;
|
|
},
|
|
// $FlowFixMe - get/set properties not yet supported
|
|
get SUPPORT_RESPONSE_TYPE() {
|
|
'use strict';
|
|
const value = testResponseType();
|
|
Object.defineProperty(FEATURES, 'SUPPORT_RESPONSE_TYPE', {value});
|
|
return value;
|
|
},
|
|
// $FlowFixMe - get/set properties not yet supported
|
|
get SUPPORT_CORS_XHR() {
|
|
'use strict';
|
|
const value = 'withCredentials' in new XMLHttpRequest();
|
|
Object.defineProperty(FEATURES, 'SUPPORT_CORS_XHR', {value});
|
|
return value;
|
|
}
|
|
};
|
|
|
|
export default FEATURES;
|