2012-03-02 23:25:08 +04:00
|
|
|
/*
|
|
|
|
html2canvas @VERSION@ <http://html2canvas.hertzen.com>
|
|
|
|
Copyright (c) 2011 Niklas von Hertzen. All rights reserved.
|
|
|
|
http://www.twitter.com/niklasvh
|
|
|
|
|
|
|
|
Released under MIT License
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
2012-03-12 10:15:03 +04:00
|
|
|
_html2canvas.Renderer.Canvas = function( options ) {
|
2012-03-02 23:25:08 +04:00
|
|
|
|
|
|
|
options = options || {};
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
var doc = document,
|
|
|
|
canvas = options.canvas || doc.createElement('canvas'),
|
|
|
|
usingFlashcanvas = false,
|
|
|
|
_createCalled = false,
|
|
|
|
canvasReadyToDraw = false,
|
|
|
|
methods,
|
|
|
|
flashMaxSize = 2880; // flash bitmap limited to 2880x2880px // http://stackoverflow.com/questions/2033792/argumenterror-error-2015-invalid-bitmapdata
|
|
|
|
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
if (canvas.getContext){
|
2012-03-12 10:15:03 +04:00
|
|
|
h2clog("html2canvas: Renderer: using canvas renderer");
|
2012-03-02 23:25:08 +04:00
|
|
|
canvasReadyToDraw = true;
|
|
|
|
} else if ( options.flashcanvas !== undefined ){
|
|
|
|
usingFlashcanvas = true;
|
2012-03-12 10:15:03 +04:00
|
|
|
h2clog("html2canvas: Renderer: canvas not available, using flashcanvas");
|
2012-03-02 23:25:08 +04:00
|
|
|
var script = doc.createElement("script");
|
|
|
|
script.src = options.flashcanvas;
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
script.onload = (function(script, func){
|
2012-05-29 05:26:26 +04:00
|
|
|
var intervalFunc;
|
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
if (script.onload === undefined) {
|
|
|
|
// IE lack of support for script onload
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
if( script.onreadystatechange !== undefined ) {
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
intervalFunc = function() {
|
|
|
|
if (script.readyState !== "loaded" && script.readyState !== "complete") {
|
|
|
|
window.setTimeout( intervalFunc, 250 );
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
} else {
|
|
|
|
// it is loaded
|
|
|
|
func();
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
};
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
window.setTimeout( intervalFunc, 250 );
|
|
|
|
|
|
|
|
} else {
|
2012-03-12 10:15:03 +04:00
|
|
|
h2clog("html2canvas: Renderer: Can't track when flashcanvas is loaded");
|
2012-03-02 23:25:08 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
} else {
|
|
|
|
return func;
|
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
})(script, function(){
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
if (typeof window.FlashCanvas !== "undefined") {
|
2012-03-12 10:15:03 +04:00
|
|
|
h2clog("html2canvas: Renderer: Flashcanvas initialized");
|
2012-03-02 23:25:08 +04:00
|
|
|
window.FlashCanvas.initElement( canvas );
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
canvasReadyToDraw = true;
|
|
|
|
if ( _createCalled !== false ) {
|
|
|
|
methods._create.apply( null, _createCalled );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
doc.body.appendChild( script );
|
|
|
|
|
2012-05-29 05:26:26 +04:00
|
|
|
}
|
2012-03-02 23:25:08 +04:00
|
|
|
|
|
|
|
methods = {
|
|
|
|
_create: function( zStack, options, doc, queue, _html2canvas ) {
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
if ( !canvasReadyToDraw ) {
|
|
|
|
_createCalled = arguments;
|
|
|
|
return canvas;
|
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
var ctx = canvas.getContext("2d"),
|
|
|
|
storageContext,
|
|
|
|
i,
|
|
|
|
queueLen,
|
|
|
|
a,
|
|
|
|
newCanvas,
|
|
|
|
bounds,
|
|
|
|
testCanvas = document.createElement("canvas"),
|
|
|
|
hasCTX = ( testCanvas.getContext !== undefined ),
|
|
|
|
storageLen,
|
|
|
|
renderItem,
|
|
|
|
testctx = ( hasCTX ) ? testCanvas.getContext("2d") : {},
|
|
|
|
safeImages = [],
|
|
|
|
fstyle;
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
canvas.width = canvas.style.width = (!usingFlashcanvas) ? options.width || zStack.ctx.width : Math.min(flashMaxSize, (options.width || zStack.ctx.width) );
|
|
|
|
canvas.height = canvas.style.height = (!usingFlashcanvas) ? options.height || zStack.ctx.height : Math.min(flashMaxSize, (options.height || zStack.ctx.height) );
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
fstyle = ctx.fillStyle;
|
|
|
|
ctx.fillStyle = zStack.backgroundColor;
|
|
|
|
ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
|
|
ctx.fillStyle = fstyle;
|
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
if ( options.svgRendering && zStack.svgRender !== undefined ) {
|
2012-05-29 05:26:26 +04:00
|
|
|
// TODO: enable async rendering to support this
|
2012-03-06 19:11:10 +04:00
|
|
|
ctx.drawImage( zStack.svgRender, 0, 0 );
|
|
|
|
} else {
|
|
|
|
for ( i = 0, queueLen = queue.length; i < queueLen; i+=1 ) {
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
storageContext = queue.splice(0, 1)[0];
|
2012-05-29 05:26:26 +04:00
|
|
|
storageContext.canvasPosition = storageContext.canvasPosition || {};
|
|
|
|
|
|
|
|
//this.canvasRenderContext(storageContext,parentctx);
|
2012-03-02 23:25:08 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
// set common settings for canvas
|
|
|
|
ctx.textBaseline = "bottom";
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
if (storageContext.clip){
|
|
|
|
ctx.save();
|
|
|
|
ctx.beginPath();
|
|
|
|
// console.log(storageContext);
|
|
|
|
ctx.rect(storageContext.clip.left, storageContext.clip.top, storageContext.clip.width, storageContext.clip.height);
|
|
|
|
ctx.clip();
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
if (storageContext.ctx.storage){
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
for (a = 0, storageLen = storageContext.ctx.storage.length; a < storageLen; a+=1){
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
renderItem = storageContext.ctx.storage[a];
|
2012-05-29 05:26:26 +04:00
|
|
|
|
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
switch(renderItem.type){
|
|
|
|
case "variable":
|
2012-05-29 05:26:26 +04:00
|
|
|
ctx[renderItem.name] = renderItem['arguments'];
|
2012-03-06 19:11:10 +04:00
|
|
|
break;
|
|
|
|
case "function":
|
|
|
|
if (renderItem.name === "fillRect") {
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
if (!usingFlashcanvas || renderItem['arguments'][0] + renderItem['arguments'][2] < flashMaxSize && renderItem['arguments'][1] + renderItem['arguments'][3] < flashMaxSize) {
|
|
|
|
ctx.fillRect.apply( ctx, renderItem['arguments'] );
|
|
|
|
}
|
2012-06-26 16:15:46 +04:00
|
|
|
} else if (renderItem.name === "drawShape") {
|
|
|
|
|
|
|
|
( function( args ) {
|
|
|
|
|
|
|
|
var i, len = args.length;
|
|
|
|
ctx.beginPath();
|
|
|
|
for ( i = 0; i < len; i++ ) {
|
|
|
|
ctx[ args[ i ].name ].apply( ctx, args[ i ]['arguments'] );
|
|
|
|
}
|
|
|
|
ctx.closePath();
|
|
|
|
ctx.fill();
|
|
|
|
})( renderItem['arguments'] );
|
|
|
|
|
|
|
|
} else if (renderItem.name === "fillText") {
|
2012-03-06 19:11:10 +04:00
|
|
|
if (!usingFlashcanvas || renderItem['arguments'][1] < flashMaxSize && renderItem['arguments'][2] < flashMaxSize) {
|
|
|
|
ctx.fillText.apply( ctx, renderItem['arguments'] );
|
|
|
|
}
|
2012-06-26 16:15:46 +04:00
|
|
|
} else if (renderItem.name === "drawImage") {
|
2012-05-29 05:26:26 +04:00
|
|
|
|
|
|
|
if (renderItem['arguments'][8] > 0 && renderItem['arguments'][7]){
|
2012-03-06 19:11:10 +04:00
|
|
|
if ( hasCTX && options.taintTest ) {
|
|
|
|
if ( safeImages.indexOf( renderItem['arguments'][ 0 ].src ) === -1 ) {
|
|
|
|
testctx.drawImage( renderItem['arguments'][ 0 ], 0, 0 );
|
|
|
|
try {
|
|
|
|
testctx.getImageData( 0, 0, 1, 1 );
|
2012-05-29 05:26:26 +04:00
|
|
|
} catch(e) {
|
2012-03-06 19:11:10 +04:00
|
|
|
testCanvas = doc.createElement("canvas");
|
|
|
|
testctx = testCanvas.getContext("2d");
|
|
|
|
continue;
|
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
safeImages.push( renderItem['arguments'][ 0 ].src );
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
}
|
2012-03-02 23:25:08 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
ctx.drawImage.apply( ctx, renderItem['arguments'] );
|
|
|
|
}
|
2012-03-06 19:11:10 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
break;
|
|
|
|
default:
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-06 19:11:10 +04:00
|
|
|
}
|
2012-03-02 23:25:08 +04:00
|
|
|
|
2012-05-29 05:26:26 +04:00
|
|
|
}
|
2012-03-06 19:11:10 +04:00
|
|
|
if (storageContext.clip){
|
|
|
|
ctx.restore();
|
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
|
|
|
}
|
2012-03-02 23:25:08 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
h2clog("html2canvas: Renderer: Canvas renderer done - returning canvas obj");
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
queueLen = options.elements.length;
|
|
|
|
|
|
|
|
if (queueLen === 1) {
|
|
|
|
if (typeof options.elements[ 0 ] === "object" && options.elements[ 0 ].nodeName !== "BODY" && usingFlashcanvas === false) {
|
|
|
|
// crop image to the bounds of selected (single) element
|
|
|
|
bounds = _html2canvas.Util.Bounds( options.elements[ 0 ] );
|
|
|
|
newCanvas = doc.createElement('canvas');
|
|
|
|
newCanvas.width = bounds.width;
|
|
|
|
newCanvas.height = bounds.height;
|
|
|
|
ctx = newCanvas.getContext("2d");
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
ctx.drawImage( canvas, bounds.left, bounds.top, bounds.width, bounds.height, 0, 0, bounds.width, bounds.height );
|
|
|
|
canvas = null;
|
|
|
|
return newCanvas;
|
|
|
|
}
|
|
|
|
} /*else {
|
|
|
|
// TODO clip and resize multiple elements
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
for ( i = 0; i < queueLen; i+=1 ) {
|
|
|
|
if (options.elements[ i ] instanceof Element) {
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
}
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
*/
|
2012-05-29 05:26:26 +04:00
|
|
|
|
|
|
|
|
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
return canvas;
|
|
|
|
}
|
2012-03-03 23:03:59 +04:00
|
|
|
};
|
2012-05-29 05:26:26 +04:00
|
|
|
|
2012-03-02 23:25:08 +04:00
|
|
|
return methods;
|
|
|
|
|
2012-03-12 10:15:03 +04:00
|
|
|
};
|