added support for CORS images and option to create canvas as tainted

This commit is contained in:
Niklas von Hertzen 2012-03-01 19:44:25 +02:00
parent c86d12b915
commit 3ad49efa00
5 changed files with 65 additions and 33 deletions

View File

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

View File

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

View File

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

View File

@ -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 &lt;base&gt; href)</h1> <h1>External image (using &lt;base&gt; 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>

View File

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