mirror of
https://github.com/niklasvh/html2canvas.git
synced 2023-08-10 21:13:10 +03:00
added support for CORS images and option to create canvas as tainted
This commit is contained in:
parent
c86d12b915
commit
3ad49efa00
@ -38,7 +38,7 @@ For more information and examples, please visit the <a href="http://html2canvas.
|
|||||||
### Changelog ###
|
### Changelog ###
|
||||||
|
|
||||||
v0.33 -
|
v0.33 -
|
||||||
|
* Added support for CORS images and option to create canvas as tainted (<a href="#">niklasvh</a>)
|
||||||
* Improved minification saved ~1K! (<a href="https://github.com/cobexer/html2canvas/commit/b82be022b2b9240bd503e078ac980bde2b953e43">cobexer</a>)
|
* Improved minification saved ~1K! (<a href="https://github.com/cobexer/html2canvas/commit/b82be022b2b9240bd503e078ac980bde2b953e43">cobexer</a>)
|
||||||
* Added integrated support for Flashcanvas (<a href="https://github.com/niklasvh/html2canvas/commit/e9257191519f67d74fd5e364d8dee3c0963ba5fc">niklasvh</a>)
|
* Added integrated support for Flashcanvas (<a href="https://github.com/niklasvh/html2canvas/commit/e9257191519f67d74fd5e364d8dee3c0963ba5fc">niklasvh</a>)
|
||||||
* Fixed a variety of legacy IE bugs (<a href="https://github.com/niklasvh/html2canvas/commit/b65357c55d0701017bafcd357bc654b54d458f8f">niklasvh</a>)
|
* Fixed a variety of legacy IE bugs (<a href="https://github.com/niklasvh/html2canvas/commit/b65357c55d0701017bafcd357bc654b54d458f8f">niklasvh</a>)
|
||||||
|
@ -10,7 +10,9 @@ html2canvas.Preload = function(element, opts){
|
|||||||
|
|
||||||
var options = {
|
var options = {
|
||||||
proxy: "http://html2canvas.appspot.com/",
|
proxy: "http://html2canvas.appspot.com/",
|
||||||
timeout: 0 // no timeout
|
timeout: 0, // no timeout
|
||||||
|
useCORS: false, // try to load images as CORS (where available), before falling back to proxy
|
||||||
|
allowTaint: false // whether to allow images to taint the canvas, won't need proxy if set to true
|
||||||
},
|
},
|
||||||
images = {
|
images = {
|
||||||
numLoaded: 0, // also failed are counted here
|
numLoaded: 0, // also failed are counted here
|
||||||
@ -26,6 +28,9 @@ html2canvas.Preload = function(element, opts){
|
|||||||
domImages = doc.images, // TODO probably should limit it to images present in the element only
|
domImages = doc.images, // TODO probably should limit it to images present in the element only
|
||||||
imgLen = domImages.length,
|
imgLen = domImages.length,
|
||||||
link = doc.createElement("a"),
|
link = doc.createElement("a"),
|
||||||
|
supportCORS = (function( img ){
|
||||||
|
return (img.crossOrigin !== undefined);
|
||||||
|
})(new Image()),
|
||||||
timeoutTimer;
|
timeoutTimer;
|
||||||
|
|
||||||
link.href = window.location.href;
|
link.href = window.location.href;
|
||||||
@ -41,7 +46,7 @@ html2canvas.Preload = function(element, opts){
|
|||||||
function isSameOrigin(url){
|
function isSameOrigin(url){
|
||||||
link.href = url;
|
link.href = url;
|
||||||
var origin = link.protocol + link.host;
|
var origin = link.protocol + link.host;
|
||||||
return ":" === origin || (origin === pageOrigin);
|
return (origin === pageOrigin);
|
||||||
}
|
}
|
||||||
|
|
||||||
function start(){
|
function start(){
|
||||||
@ -215,18 +220,50 @@ html2canvas.Preload = function(element, opts){
|
|||||||
|
|
||||||
function setImageLoadHandlers(img, imageObj) {
|
function setImageLoadHandlers(img, imageObj) {
|
||||||
img.onload = function() {
|
img.onload = function() {
|
||||||
|
if ( imageObj.timer !== undefined ) {
|
||||||
|
// CORS succeeded
|
||||||
|
window.clearTimeout( imageObj.timer );
|
||||||
|
}
|
||||||
images.numLoaded++;
|
images.numLoaded++;
|
||||||
imageObj.succeeded = true;
|
imageObj.succeeded = true;
|
||||||
start();
|
start();
|
||||||
};
|
};
|
||||||
img.onerror = function() {
|
img.onerror = function() {
|
||||||
|
|
||||||
|
if (img.crossOrigin === "anonymous") {
|
||||||
|
// CORS failed
|
||||||
|
window.clearTimeout( imageObj.timer );
|
||||||
|
|
||||||
|
// let's try with proxy instead
|
||||||
|
if ( options.proxy ) {
|
||||||
|
var src = img.src;
|
||||||
|
img = new Image();
|
||||||
|
imageObj.img = img;
|
||||||
|
img.src = src;
|
||||||
|
|
||||||
|
proxyGetImage( img.src, img, imageObj );
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
images.numLoaded++;
|
images.numLoaded++;
|
||||||
images.numFailed++;
|
images.numFailed++;
|
||||||
imageObj.succeeded = false;
|
imageObj.succeeded = false;
|
||||||
start();
|
start();
|
||||||
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// work around for https://bugs.webkit.org/show_bug.cgi?id=80028
|
||||||
|
function isComplete() {
|
||||||
|
if (!this.img.complete) {
|
||||||
|
this.timer = window.setTimeout(this.img.customComplete, 100)
|
||||||
|
} else {
|
||||||
|
this.img.onerror();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
methods = {
|
methods = {
|
||||||
loadImage: function( src ) {
|
loadImage: function( src ) {
|
||||||
var img, imageObj;
|
var img, imageObj;
|
||||||
@ -237,14 +274,24 @@ html2canvas.Preload = function(element, opts){
|
|||||||
imageObj = images[src] = { img: img };
|
imageObj = images[src] = { img: img };
|
||||||
images.numTotal++;
|
images.numTotal++;
|
||||||
setImageLoadHandlers(img, imageObj);
|
setImageLoadHandlers(img, imageObj);
|
||||||
}
|
} else if ( isSameOrigin( src ) || options.allowTaint === true ) {
|
||||||
else if ( isSameOrigin( src ) ) {
|
|
||||||
imageObj = images[src] = { img: img };
|
imageObj = images[src] = { img: img };
|
||||||
images.numTotal++;
|
images.numTotal++;
|
||||||
setImageLoadHandlers(img, imageObj);
|
setImageLoadHandlers(img, imageObj);
|
||||||
img.src = src;
|
img.src = src;
|
||||||
}
|
} else if ( supportCORS && !options.allowTaint && options.useCORS ) {
|
||||||
else if ( options.proxy ) {
|
// attempt to load with CORS
|
||||||
|
|
||||||
|
img.crossOrigin = "anonymous";
|
||||||
|
imageObj = images[src] = { img: img };
|
||||||
|
images.numTotal++;
|
||||||
|
setImageLoadHandlers(img, imageObj);
|
||||||
|
img.src = src;
|
||||||
|
|
||||||
|
img.customComplete = isComplete.bind(imageObj);
|
||||||
|
img.customComplete();
|
||||||
|
|
||||||
|
} else if ( options.proxy ) {
|
||||||
imageObj = images[src] = { img: img };
|
imageObj = images[src] = { img: img };
|
||||||
images.numTotal++;
|
images.numTotal++;
|
||||||
proxyGetImage( src, img, imageObj );
|
proxyGetImage( src, img, imageObj );
|
||||||
|
@ -127,35 +127,16 @@ html2canvas.Renderer = function(parseQueue, opts){
|
|||||||
if (renderItem.name === "fillRect") {
|
if (renderItem.name === "fillRect") {
|
||||||
|
|
||||||
if (!usingFlashcanvas || renderItem['arguments'][0] + renderItem['arguments'][2] < flashMaxSize && renderItem['arguments'][1] + renderItem['arguments'][3] < flashMaxSize) {
|
if (!usingFlashcanvas || renderItem['arguments'][0] + renderItem['arguments'][2] < flashMaxSize && renderItem['arguments'][1] + renderItem['arguments'][3] < flashMaxSize) {
|
||||||
ctx.fillRect(
|
ctx.fillRect.apply( ctx, renderItem['arguments'] );
|
||||||
renderItem['arguments'][0],
|
|
||||||
renderItem['arguments'][1],
|
|
||||||
renderItem['arguments'][2],
|
|
||||||
renderItem['arguments'][3]
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}else if(renderItem.name === "fillText") {
|
}else if(renderItem.name === "fillText") {
|
||||||
if (!usingFlashcanvas || renderItem['arguments'][1] < flashMaxSize && renderItem['arguments'][2] < flashMaxSize) {
|
if (!usingFlashcanvas || renderItem['arguments'][1] < flashMaxSize && renderItem['arguments'][2] < flashMaxSize) {
|
||||||
ctx.fillText(
|
ctx.fillText.apply( ctx, renderItem['arguments'] );
|
||||||
renderItem['arguments'][0],
|
|
||||||
renderItem['arguments'][1],
|
|
||||||
renderItem['arguments'][2]
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}else if(renderItem.name === "drawImage") {
|
}else if(renderItem.name === "drawImage") {
|
||||||
|
|
||||||
if (renderItem['arguments'][8] > 0 && renderItem['arguments'][7]){
|
if (renderItem['arguments'][8] > 0 && renderItem['arguments'][7]){
|
||||||
ctx.drawImage(
|
ctx.drawImage.apply( ctx, renderItem['arguments'] );
|
||||||
renderItem['arguments'][0],
|
|
||||||
renderItem['arguments'][1],
|
|
||||||
renderItem['arguments'][2],
|
|
||||||
renderItem['arguments'][3],
|
|
||||||
renderItem['arguments'][4],
|
|
||||||
renderItem['arguments'][5],
|
|
||||||
renderItem['arguments'][6],
|
|
||||||
renderItem['arguments'][7],
|
|
||||||
renderItem['arguments'][8]
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,7 +5,7 @@
|
|||||||
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
|
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
|
||||||
<script type="text/javascript" src="test.js"></script>
|
<script type="text/javascript" src="test.js"></script>
|
||||||
|
|
||||||
<base href="http://www.google.com/" />
|
<base href="http://www.google.com/" />
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<h1>External image</h1>
|
<h1>External image</h1>
|
||||||
@ -14,5 +14,8 @@
|
|||||||
<h1>External image (using <base> href)</h1>
|
<h1>External image (using <base> href)</h1>
|
||||||
<img src="/logos/2011/gregormendel11-res.jpg" />
|
<img src="/logos/2011/gregormendel11-res.jpg" />
|
||||||
|
|
||||||
|
<h1>External image (CORS)</h1>
|
||||||
|
<img src="http://publishmydata.com/assets/home/blue_bg.png" />
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -19,7 +19,8 @@
|
|||||||
setTimeout(function() {
|
setTimeout(function() {
|
||||||
$(document.body).html2canvas({
|
$(document.body).html2canvas({
|
||||||
logging: true,
|
logging: true,
|
||||||
profile: true
|
profile: true,
|
||||||
|
useCORS: true
|
||||||
});
|
});
|
||||||
}, 100);
|
}, 100);
|
||||||
};
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user