Initial OpenECOMP policy/engine commit
[policy/engine.git] / ecomp-sdk-app / src / main / webapp / app / policyApp / libs / html2canvas.js
1 /*
2   html2canvas 0.4.1 <http://html2canvas.hertzen.com>
3   Copyright (c) 2013 Niklas von Hertzen
4
5   Released under MIT License
6 */
7
8 (function(window, document, undefined){
9
10 //"use strict";
11
12 var _html2canvas = {},
13 previousElement,
14 computedCSS,
15 html2canvas;
16
17 _html2canvas.Util = {};
18
19 _html2canvas.Util.log = function(a) {
20   if (_html2canvas.logging && window.console && window.console.log) {
21     window.console.log(a);
22   }
23 };
24
25 _html2canvas.Util.trimText = (function(isNative){
26   return function(input) {
27     return isNative ? isNative.apply(input) : ((input || '') + '').replace( /^\s+|\s+$/g , '' );
28   };
29 })(String.prototype.trim);
30
31 _html2canvas.Util.asFloat = function(v) {
32   return parseFloat(v);
33 };
34
35 (function() {
36   // TODO: support all possible length values
37   var TEXT_SHADOW_PROPERTY = /((rgba|rgb)\([^\)]+\)(\s-?\d+px){0,})/g;
38   var TEXT_SHADOW_VALUES = /(-?\d+px)|(#.+)|(rgb\(.+\))|(rgba\(.+\))/g;
39   _html2canvas.Util.parseTextShadows = function (value) {
40     if (!value || value === 'none') {
41       return [];
42     }
43
44     // find multiple shadow declarations
45     var shadows = value.match(TEXT_SHADOW_PROPERTY),
46       results = [];
47     for (var i = 0; shadows && (i < shadows.length); i++) {
48       var s = shadows[i].match(TEXT_SHADOW_VALUES);
49       results.push({
50         color: s[0],
51         offsetX: s[1] ? s[1].replace('px', '') : 0,
52         offsetY: s[2] ? s[2].replace('px', '') : 0,
53         blur: s[3] ? s[3].replace('px', '') : 0
54       });
55     }
56     return results;
57   };
58 })();
59
60 _html2canvas.Util.parseBackgroundImage = function (value) {
61     var whitespace = ' \r\n\t',
62         method, definition, prefix, prefix_i, block, results = [],
63         c, mode = 0, numParen = 0, quote, args;
64
65     var appendResult = function(){
66         if(method) {
67             if(definition.substr( 0, 1 ) === '"') {
68                 definition = definition.substr( 1, definition.length - 2 );
69             }
70             if(definition) {
71                 args.push(definition);
72             }
73             if(method.substr( 0, 1 ) === '-' &&
74                     (prefix_i = method.indexOf( '-', 1 ) + 1) > 0) {
75                 prefix = method.substr( 0, prefix_i);
76                 method = method.substr( prefix_i );
77             }
78             results.push({
79                 prefix: prefix,
80                 method: method.toLowerCase(),
81                 value: block,
82                 args: args
83             });
84         }
85         args = []; //for some odd reason, setting .length = 0 didn't work in safari
86         method =
87             prefix =
88             definition =
89             block = '';
90     };
91
92     appendResult();
93     for(var i = 0, ii = value.length; i<ii; i++) {
94         c = value[i];
95         if(mode === 0 && whitespace.indexOf( c ) > -1){
96             continue;
97         }
98         switch(c) {
99             case '"':
100                 if(!quote) {
101                     quote = c;
102                 }
103                 else if(quote === c) {
104                     quote = null;
105                 }
106                 break;
107
108             case '(':
109                 if(quote) { break; }
110                 else if(mode === 0) {
111                     mode = 1;
112                     block += c;
113                     continue;
114                 } else {
115                     numParen++;
116                 }
117                 break;
118
119             case ')':
120                 if(quote) { break; }
121                 else if(mode === 1) {
122                     if(numParen === 0) {
123                         mode = 0;
124                         block += c;
125                         appendResult();
126                         continue;
127                     } else {
128                         numParen--;
129                     }
130                 }
131                 break;
132
133             case ',':
134                 if(quote) { break; }
135                 else if(mode === 0) {
136                     appendResult();
137                     continue;
138                 }
139                 else if (mode === 1) {
140                     if(numParen === 0 && !method.match(/^url$/i)) {
141                         args.push(definition);
142                         definition = '';
143                         block += c;
144                         continue;
145                     }
146                 }
147                 break;
148         }
149
150         block += c;
151         if(mode === 0) { method += c; }
152         else { definition += c; }
153     }
154     appendResult();
155
156     return results;
157 };
158
159 _html2canvas.Util.Bounds = function (element) {
160   var clientRect, bounds = {};
161
162   if (element.getBoundingClientRect){
163     clientRect = element.getBoundingClientRect();
164
165     // TODO add scroll position to bounds, so no scrolling of window necessary
166     bounds.top = clientRect.top;
167     bounds.bottom = clientRect.bottom || (clientRect.top + clientRect.height);
168     bounds.left = clientRect.left;
169
170     bounds.width = element.offsetWidth;
171     bounds.height = element.offsetHeight;
172   }
173
174   return bounds;
175 };
176
177 // TODO ideally, we'd want everything to go through this function instead of Util.Bounds,
178 // but would require further work to calculate the correct positions for elements with offsetParents
179 _html2canvas.Util.OffsetBounds = function (element) {
180   var parent = element.offsetParent ? _html2canvas.Util.OffsetBounds(element.offsetParent) : {top: 0, left: 0};
181
182   return {
183     top: element.offsetTop + parent.top,
184     bottom: element.offsetTop + element.offsetHeight + parent.top,
185     left: element.offsetLeft + parent.left,
186     width: element.offsetWidth,
187     height: element.offsetHeight
188   };
189 };
190
191 function toPX(element, attribute, value ) {
192     var rsLeft = element.runtimeStyle && element.runtimeStyle[attribute],
193         left,
194         style = element.style;
195
196     // Check if we are not dealing with pixels, (Opera has issues with this)
197     // Ported from jQuery css.js
198     // From the awesome hack by Dean Edwards
199     // http://erik.eae.net/archives/2007/07/27/18.54.15/#comment-102291
200
201     // If we're not dealing with a regular pixel number
202     // but a number that has a weird ending, we need to convert it to pixels
203
204     if ( !/^-?[0-9]+\.?[0-9]*(?:px)?$/i.test( value ) && /^-?\d/.test(value) ) {
205         // Remember the original values
206         left = style.left;
207
208         // Put in the new values to get a computed value out
209         if (rsLeft) {
210             element.runtimeStyle.left = element.currentStyle.left;
211         }
212         style.left = attribute === "fontSize" ? "1em" : (value || 0);
213         value = style.pixelLeft + "px";
214
215         // Revert the changed values
216         style.left = left;
217         if (rsLeft) {
218             element.runtimeStyle.left = rsLeft;
219         }
220     }
221
222     if (!/^(thin|medium|thick)$/i.test(value)) {
223         return Math.round(parseFloat(value)) + "px";
224     }
225
226     return value;
227 }
228
229 function asInt(val) {
230     return parseInt(val, 10);
231 }
232
233 function isPercentage(value) {
234   return value.toString().indexOf("%") !== -1;
235 }
236
237 function parseBackgroundSizePosition(value, element, attribute, index) {
238     value = (value || '').split(',');
239     value = value[index || 0] || value[0] || 'auto';
240     value = _html2canvas.Util.trimText(value).split(' ');
241     if(attribute === 'backgroundSize' && (value[0] && value[0].match(/^(cover|contain|auto)$/))) {
242         return value;
243     } else {
244         value[0] = (value[0].indexOf( "%" ) === -1) ? toPX(element, attribute + "X", value[0]) : value[0];
245         if(value[1] === undefined) {
246             if(attribute === 'backgroundSize') {
247                 value[1] = 'auto';
248                 return value;
249             } else {
250                 // IE 9 doesn't return double digit always
251                 value[1] = value[0];
252             }
253         }
254         value[1] = (value[1].indexOf("%") === -1) ? toPX(element, attribute + "Y", value[1]) : value[1];
255     }
256     return value;
257 }
258
259 _html2canvas.Util.getCSS = function (element, attribute, index) {
260     if (previousElement !== element) {
261       computedCSS = document.defaultView.getComputedStyle(element, null);
262     }
263
264     var value = computedCSS[attribute];
265
266     if (/^background(Size|Position)$/.test(attribute)) {
267         return parseBackgroundSizePosition(value, element, attribute, index);
268     } else if (/border(Top|Bottom)(Left|Right)Radius/.test(attribute)) {
269       var arr = value.split(" ");
270       if (arr.length <= 1) {
271           arr[1] = arr[0];
272       }
273       return arr.map(asInt);
274     }
275
276   return value;
277 };
278
279 _html2canvas.Util.resizeBounds = function( current_width, current_height, target_width, target_height, stretch_mode ){
280   var target_ratio = target_width / target_height,
281     current_ratio = current_width / current_height,
282     output_width, output_height;
283
284   if(!stretch_mode || stretch_mode === 'auto') {
285     output_width = target_width;
286     output_height = target_height;
287   } else if(target_ratio < current_ratio ^ stretch_mode === 'contain') {
288     output_height = target_height;
289     output_width = target_height * current_ratio;
290   } else {
291     output_width = target_width;
292     output_height = target_width / current_ratio;
293   }
294
295   return {
296     width: output_width,
297     height: output_height
298   };
299 };
300
301 _html2canvas.Util.BackgroundPosition = function(element, bounds, image, imageIndex, backgroundSize ) {
302     var backgroundPosition =  _html2canvas.Util.getCSS(element, 'backgroundPosition', imageIndex),
303         leftPosition,
304         topPosition;
305     if (backgroundPosition.length === 1){
306         backgroundPosition = [backgroundPosition[0], backgroundPosition[0]];
307     }
308
309     if (isPercentage(backgroundPosition[0])){
310         leftPosition = (bounds.width - (backgroundSize || image).width) * (parseFloat(backgroundPosition[0]) / 100);
311     } else {
312         leftPosition = parseInt(backgroundPosition[0], 10);
313     }
314
315     if (backgroundPosition[1] === 'auto') {
316         topPosition = leftPosition / image.width * image.height;
317     } else if (isPercentage(backgroundPosition[1])){
318         topPosition =  (bounds.height - (backgroundSize || image).height) * parseFloat(backgroundPosition[1]) / 100;
319     } else {
320         topPosition = parseInt(backgroundPosition[1], 10);
321     }
322
323     if (backgroundPosition[0] === 'auto') {
324         leftPosition = topPosition / image.height * image.width;
325     }
326
327     return {left: leftPosition, top: topPosition};
328 };
329
330 _html2canvas.Util.BackgroundSize = function(element, bounds, image, imageIndex) {
331   var backgroundSize =  _html2canvas.Util.getCSS(element, 'backgroundSize', imageIndex), width, height;
332
333   if (backgroundSize.length === 1) {
334     backgroundSize = [backgroundSize[0], backgroundSize[0]];
335   }
336
337   if (isPercentage(backgroundSize[0])) {
338     width = bounds.width * parseFloat(backgroundSize[0]) / 100;
339   } else if (/contain|cover/.test(backgroundSize[0])) {
340     return _html2canvas.Util.resizeBounds(image.width, image.height, bounds.width, bounds.height, backgroundSize[0]);
341   } else {
342     width = parseInt(backgroundSize[0], 10);
343   }
344
345   if (backgroundSize[0] === 'auto' && backgroundSize[1] === 'auto') {
346     height = image.height;
347   } else if (backgroundSize[1] === 'auto') {
348     height = width / image.width * image.height;
349   } else if (isPercentage(backgroundSize[1])) {
350     height =  bounds.height * parseFloat(backgroundSize[1]) / 100;
351   } else {
352     height = parseInt(backgroundSize[1], 10);
353   }
354
355   if (backgroundSize[0] === 'auto') {
356     width = height / image.height * image.width;
357   }
358
359   return {width: width, height: height};
360 };
361
362 _html2canvas.Util.BackgroundRepeat = function(element, imageIndex) {
363   var backgroundRepeat = _html2canvas.Util.getCSS(element, "backgroundRepeat").split(",").map(_html2canvas.Util.trimText);
364   return backgroundRepeat[imageIndex] || backgroundRepeat[0];
365 };
366
367 _html2canvas.Util.Extend = function (options, defaults) {
368   for (var key in options) {
369     if (options.hasOwnProperty(key)) {
370       defaults[key] = options[key];
371     }
372   }
373   return defaults;
374 };
375
376
377 /*
378  * Derived from jQuery.contents()
379  * Copyright 2010, John Resig
380  * Dual licensed under the MIT or GPL Version 2 licenses.
381  * http://jquery.org/license
382  */
383 _html2canvas.Util.Children = function( elem ) {
384   var children;
385   try {
386     children = (elem.nodeName && elem.nodeName.toUpperCase() === "IFRAME") ? elem.contentDocument || elem.contentWindow.document : (function(array) {
387       var ret = [];
388       if (array !== null) {
389         (function(first, second ) {
390           var i = first.length,
391           j = 0;
392
393           if (typeof second.length === "number") {
394             for (var l = second.length; j < l; j++) {
395               first[i++] = second[j];
396             }
397           } else {
398             while (second[j] !== undefined) {
399               first[i++] = second[j++];
400             }
401           }
402
403           first.length = i;
404
405           return first;
406         })(ret, array);
407       }
408       return ret;
409     })(elem.childNodes);
410
411   } catch (ex) {
412     _html2canvas.Util.log("html2canvas.Util.Children failed with exception: " + ex.message);
413     children = [];
414   }
415   return children;
416 };
417
418 _html2canvas.Util.isTransparent = function(backgroundColor) {
419   return (!backgroundColor || backgroundColor === "transparent" || backgroundColor === "rgba(0, 0, 0, 0)");
420 };
421
422 _html2canvas.Util.Font = (function () {
423
424   var fontData = {};
425
426   return function(font, fontSize, doc) {
427     if (fontData[font + "-" + fontSize] !== undefined) {
428       return fontData[font + "-" + fontSize];
429     }
430
431     var container = doc.createElement('div'),
432     img = doc.createElement('img'),
433     span = doc.createElement('span'),
434     sampleText = 'Hidden Text',
435     baseline,
436     middle,
437     metricsObj;
438
439     container.style.visibility = "hidden";
440     container.style.fontFamily = font;
441     container.style.fontSize = fontSize;
442     container.style.margin = 0;
443     container.style.padding = 0;
444
445     doc.body.appendChild(container);
446
447     // http://probablyprogramming.com/2009/03/15/the-tiniest-gif-ever (handtinywhite.gif)
448     img.src = "data:image/gif;base64,R0lGODlhAQABAIABAP///wAAACwAAAAAAQABAAACAkQBADs=";
449     img.width = 1;
450     img.height = 1;
451
452     img.style.margin = 0;
453     img.style.padding = 0;
454     img.style.verticalAlign = "baseline";
455
456     span.style.fontFamily = font;
457     span.style.fontSize = fontSize;
458     span.style.margin = 0;
459     span.style.padding = 0;
460
461     span.appendChild(doc.createTextNode(sampleText));
462     container.appendChild(span);
463     container.appendChild(img);
464     baseline = (img.offsetTop - span.offsetTop) + 1;
465
466     container.removeChild(span);
467     container.appendChild(doc.createTextNode(sampleText));
468
469     container.style.lineHeight = "normal";
470     img.style.verticalAlign = "super";
471
472     middle = (img.offsetTop-container.offsetTop) + 1;
473     metricsObj = {
474       baseline: baseline,
475       lineWidth: 1,
476       middle: middle
477     };
478
479     fontData[font + "-" + fontSize] = metricsObj;
480
481     doc.body.removeChild(container);
482
483     return metricsObj;
484   };
485 })();
486
487 (function(){
488   var Util = _html2canvas.Util,
489     Generate = {};
490
491   _html2canvas.Generate = Generate;
492
493   var reGradients = [
494   /^(-webkit-linear-gradient)\(([a-z\s]+)([\w\d\.\s,%\(\)]+)\)$/,
495   /^(-o-linear-gradient)\(([a-z\s]+)([\w\d\.\s,%\(\)]+)\)$/,
496   /^(-webkit-gradient)\((linear|radial),\s((?:\d{1,3}%?)\s(?:\d{1,3}%?),\s(?:\d{1,3}%?)\s(?:\d{1,3}%?))([\w\d\.\s,%\(\)\-]+)\)$/,
497   /^(-moz-linear-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?))([\w\d\.\s,%\(\)]+)\)$/,
498   /^(-webkit-radial-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?)),\s(\w+)\s([a-z\-]+)([\w\d\.\s,%\(\)]+)\)$/,
499   /^(-moz-radial-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?)),\s(\w+)\s?([a-z\-]*)([\w\d\.\s,%\(\)]+)\)$/,
500   /^(-o-radial-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?)),\s(\w+)\s([a-z\-]+)([\w\d\.\s,%\(\)]+)\)$/
501   ];
502
503   /*
504  * TODO: Add IE10 vendor prefix (-ms) support
505  * TODO: Add W3C gradient (linear-gradient) support
506  * TODO: Add old Webkit -webkit-gradient(radial, ...) support
507  * TODO: Maybe some RegExp optimizations are possible ;o)
508  */
509   Generate.parseGradient = function(css, bounds) {
510     var gradient, i, len = reGradients.length, m1, stop, m2, m2Len, step, m3, tl,tr,br,bl;
511
512     for(i = 0; i < len; i+=1){
513       m1 = css.match(reGradients[i]);
514       if(m1) {
515         break;
516       }
517     }
518
519     if(m1) {
520       switch(m1[1]) {
521         case '-webkit-linear-gradient':
522         case '-o-linear-gradient':
523
524           gradient = {
525             type: 'linear',
526             x0: null,
527             y0: null,
528             x1: null,
529             y1: null,
530             colorStops: []
531           };
532
533           // get coordinates
534           m2 = m1[2].match(/\w+/g);
535           if(m2){
536             m2Len = m2.length;
537             for(i = 0; i < m2Len; i+=1){
538               switch(m2[i]) {
539                 case 'top':
540                   gradient.y0 = 0;
541                   gradient.y1 = bounds.height;
542                   break;
543
544                 case 'right':
545                   gradient.x0 = bounds.width;
546                   gradient.x1 = 0;
547                   break;
548
549                 case 'bottom':
550                   gradient.y0 = bounds.height;
551                   gradient.y1 = 0;
552                   break;
553
554                 case 'left':
555                   gradient.x0 = 0;
556                   gradient.x1 = bounds.width;
557                   break;
558               }
559             }
560           }
561           if(gradient.x0 === null && gradient.x1 === null){ // center
562             gradient.x0 = gradient.x1 = bounds.width / 2;
563           }
564           if(gradient.y0 === null && gradient.y1 === null){ // center
565             gradient.y0 = gradient.y1 = bounds.height / 2;
566           }
567
568           // get colors and stops
569           m2 = m1[3].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\)(?:\s\d{1,3}(?:%|px))?)+/g);
570           if(m2){
571             m2Len = m2.length;
572             step = 1 / Math.max(m2Len - 1, 1);
573             for(i = 0; i < m2Len; i+=1){
574               m3 = m2[i].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\))\s*(\d{1,3})?(%|px)?/);
575               if(m3[2]){
576                 stop = parseFloat(m3[2]);
577                 if(m3[3] === '%'){
578                   stop /= 100;
579                 } else { // px - stupid opera
580                   stop /= bounds.width;
581                 }
582               } else {
583                 stop = i * step;
584               }
585               gradient.colorStops.push({
586                 color: m3[1],
587                 stop: stop
588               });
589             }
590           }
591           break;
592
593         case '-webkit-gradient':
594
595           gradient = {
596             type: m1[2] === 'radial' ? 'circle' : m1[2], // TODO: Add radial gradient support for older mozilla definitions
597             x0: 0,
598             y0: 0,
599             x1: 0,
600             y1: 0,
601             colorStops: []
602           };
603
604           // get coordinates
605           m2 = m1[3].match(/(\d{1,3})%?\s(\d{1,3})%?,\s(\d{1,3})%?\s(\d{1,3})%?/);
606           if(m2){
607             gradient.x0 = (m2[1] * bounds.width) / 100;
608             gradient.y0 = (m2[2] * bounds.height) / 100;
609             gradient.x1 = (m2[3] * bounds.width) / 100;
610             gradient.y1 = (m2[4] * bounds.height) / 100;
611           }
612
613           // get colors and stops
614           m2 = m1[4].match(/((?:from|to|color-stop)\((?:[0-9\.]+,\s)?(?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\)\))+/g);
615           if(m2){
616             m2Len = m2.length;
617             for(i = 0; i < m2Len; i+=1){
618               m3 = m2[i].match(/(from|to|color-stop)\(([0-9\.]+)?(?:,\s)?((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\))\)/);
619               stop = parseFloat(m3[2]);
620               if(m3[1] === 'from') {
621                 stop = 0.0;
622               }
623               if(m3[1] === 'to') {
624                 stop = 1.0;
625               }
626               gradient.colorStops.push({
627                 color: m3[3],
628                 stop: stop
629               });
630             }
631           }
632           break;
633
634         case '-moz-linear-gradient':
635
636           gradient = {
637             type: 'linear',
638             x0: 0,
639             y0: 0,
640             x1: 0,
641             y1: 0,
642             colorStops: []
643           };
644
645           // get coordinates
646           m2 = m1[2].match(/(\d{1,3})%?\s(\d{1,3})%?/);
647
648           // m2[1] == 0%   -> left
649           // m2[1] == 50%  -> center
650           // m2[1] == 100% -> right
651
652           // m2[2] == 0%   -> top
653           // m2[2] == 50%  -> center
654           // m2[2] == 100% -> bottom
655
656           if(m2){
657             gradient.x0 = (m2[1] * bounds.width) / 100;
658             gradient.y0 = (m2[2] * bounds.height) / 100;
659             gradient.x1 = bounds.width - gradient.x0;
660             gradient.y1 = bounds.height - gradient.y0;
661           }
662
663           // get colors and stops
664           m2 = m1[3].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\)(?:\s\d{1,3}%)?)+/g);
665           if(m2){
666             m2Len = m2.length;
667             step = 1 / Math.max(m2Len - 1, 1);
668             for(i = 0; i < m2Len; i+=1){
669               m3 = m2[i].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\))\s*(\d{1,3})?(%)?/);
670               if(m3[2]){
671                 stop = parseFloat(m3[2]);
672                 if(m3[3]){ // percentage
673                   stop /= 100;
674                 }
675               } else {
676                 stop = i * step;
677               }
678               gradient.colorStops.push({
679                 color: m3[1],
680                 stop: stop
681               });
682             }
683           }
684           break;
685
686         case '-webkit-radial-gradient':
687         case '-moz-radial-gradient':
688         case '-o-radial-gradient':
689
690           gradient = {
691             type: 'circle',
692             x0: 0,
693             y0: 0,
694             x1: bounds.width,
695             y1: bounds.height,
696             cx: 0,
697             cy: 0,
698             rx: 0,
699             ry: 0,
700             colorStops: []
701           };
702
703           // center
704           m2 = m1[2].match(/(\d{1,3})%?\s(\d{1,3})%?/);
705           if(m2){
706             gradient.cx = (m2[1] * bounds.width) / 100;
707             gradient.cy = (m2[2] * bounds.height) / 100;
708           }
709
710           // size
711           m2 = m1[3].match(/\w+/);
712           m3 = m1[4].match(/[a-z\-]*/);
713           if(m2 && m3){
714             switch(m3[0]){
715               case 'farthest-corner':
716               case 'cover': // is equivalent to farthest-corner
717               case '': // mozilla removes "cover" from definition :(
718                 tl = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.cy, 2));
719                 tr = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2));
720                 br = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2));
721                 bl = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.cy, 2));
722                 gradient.rx = gradient.ry = Math.max(tl, tr, br, bl);
723                 break;
724               case 'closest-corner':
725                 tl = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.cy, 2));
726                 tr = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2));
727                 br = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2));
728                 bl = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.cy, 2));
729                 gradient.rx = gradient.ry = Math.min(tl, tr, br, bl);
730                 break;
731               case 'farthest-side':
732                 if(m2[0] === 'circle'){
733                   gradient.rx = gradient.ry = Math.max(
734                     gradient.cx,
735                     gradient.cy,
736                     gradient.x1 - gradient.cx,
737                     gradient.y1 - gradient.cy
738                     );
739                 } else { // ellipse
740
741                   gradient.type = m2[0];
742
743                   gradient.rx = Math.max(
744                     gradient.cx,
745                     gradient.x1 - gradient.cx
746                     );
747                   gradient.ry = Math.max(
748                     gradient.cy,
749                     gradient.y1 - gradient.cy
750                     );
751                 }
752                 break;
753               case 'closest-side':
754               case 'contain': // is equivalent to closest-side
755                 if(m2[0] === 'circle'){
756                   gradient.rx = gradient.ry = Math.min(
757                     gradient.cx,
758                     gradient.cy,
759                     gradient.x1 - gradient.cx,
760                     gradient.y1 - gradient.cy
761                     );
762                 } else { // ellipse
763
764                   gradient.type = m2[0];
765
766                   gradient.rx = Math.min(
767                     gradient.cx,
768                     gradient.x1 - gradient.cx
769                     );
770                   gradient.ry = Math.min(
771                     gradient.cy,
772                     gradient.y1 - gradient.cy
773                     );
774                 }
775                 break;
776
777             // TODO: add support for "30px 40px" sizes (webkit only)
778             }
779           }
780
781           // color stops
782           m2 = m1[5].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\)(?:\s\d{1,3}(?:%|px))?)+/g);
783           if(m2){
784             m2Len = m2.length;
785             step = 1 / Math.max(m2Len - 1, 1);
786             for(i = 0; i < m2Len; i+=1){
787               m3 = m2[i].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\))\s*(\d{1,3})?(%|px)?/);
788               if(m3[2]){
789                 stop = parseFloat(m3[2]);
790                 if(m3[3] === '%'){
791                   stop /= 100;
792                 } else { // px - stupid opera
793                   stop /= bounds.width;
794                 }
795               } else {
796                 stop = i * step;
797               }
798               gradient.colorStops.push({
799                 color: m3[1],
800                 stop: stop
801               });
802             }
803           }
804           break;
805       }
806     }
807
808     return gradient;
809   };
810
811   function addScrollStops(grad) {
812     return function(colorStop) {
813       try {
814         grad.addColorStop(colorStop.stop, colorStop.color);
815       }
816       catch(e) {
817         Util.log(['failed to add color stop: ', e, '; tried to add: ', colorStop]);
818       }
819     };
820   }
821
822   Generate.Gradient = function(src, bounds) {
823     if(bounds.width === 0 || bounds.height === 0) {
824       return;
825     }
826
827     var canvas = document.createElement('canvas'),
828     ctx = canvas.getContext('2d'),
829     gradient, grad;
830
831     canvas.width = bounds.width;
832     canvas.height = bounds.height;
833
834     // TODO: add support for multi defined background gradients
835     gradient = _html2canvas.Generate.parseGradient(src, bounds);
836
837     if(gradient) {
838       switch(gradient.type) {
839         case 'linear':
840           grad = ctx.createLinearGradient(gradient.x0, gradient.y0, gradient.x1, gradient.y1);
841           gradient.colorStops.forEach(addScrollStops(grad));
842           ctx.fillStyle = grad;
843           ctx.fillRect(0, 0, bounds.width, bounds.height);
844           break;
845
846         case 'circle':
847           grad = ctx.createRadialGradient(gradient.cx, gradient.cy, 0, gradient.cx, gradient.cy, gradient.rx);
848           gradient.colorStops.forEach(addScrollStops(grad));
849           ctx.fillStyle = grad;
850           ctx.fillRect(0, 0, bounds.width, bounds.height);
851           break;
852
853         case 'ellipse':
854           var canvasRadial = document.createElement('canvas'),
855             ctxRadial = canvasRadial.getContext('2d'),
856             ri = Math.max(gradient.rx, gradient.ry),
857             di = ri * 2;
858
859           canvasRadial.width = canvasRadial.height = di;
860
861           grad = ctxRadial.createRadialGradient(gradient.rx, gradient.ry, 0, gradient.rx, gradient.ry, ri);
862           gradient.colorStops.forEach(addScrollStops(grad));
863
864           ctxRadial.fillStyle = grad;
865           ctxRadial.fillRect(0, 0, di, di);
866
867           ctx.fillStyle = gradient.colorStops[gradient.colorStops.length - 1].color;
868           ctx.fillRect(0, 0, canvas.width, canvas.height);
869           ctx.drawImage(canvasRadial, gradient.cx - gradient.rx, gradient.cy - gradient.ry, 2 * gradient.rx, 2 * gradient.ry);
870           break;
871       }
872     }
873
874     return canvas;
875   };
876
877   Generate.ListAlpha = function(number) {
878     var tmp = "",
879     modulus;
880
881     do {
882       modulus = number % 26;
883       tmp = String.fromCharCode((modulus) + 64) + tmp;
884       number = number / 26;
885     }while((number*26) > 26);
886
887     return tmp;
888   };
889
890   Generate.ListRoman = function(number) {
891     var romanArray = ["M", "CM", "D", "CD", "C", "XC", "L", "XL", "X", "IX", "V", "IV", "I"],
892     decimal = [1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1],
893     roman = "",
894     v,
895     len = romanArray.length;
896
897     if (number <= 0 || number >= 4000) {
898       return number;
899     }
900
901     for (v=0; v < len; v+=1) {
902       while (number >= decimal[v]) {
903         number -= decimal[v];
904         roman += romanArray[v];
905       }
906     }
907
908     return roman;
909   };
910 })();
911 function h2cRenderContext(width, height) {
912   var storage = [];
913   return {
914     storage: storage,
915     width: width,
916     height: height,
917     clip: function() {
918       storage.push({
919         type: "function",
920         name: "clip",
921         'arguments': arguments
922       });
923     },
924     translate: function() {
925       storage.push({
926         type: "function",
927         name: "translate",
928         'arguments': arguments
929       });
930     },
931     fill: function() {
932       storage.push({
933         type: "function",
934         name: "fill",
935         'arguments': arguments
936       });
937     },
938     save: function() {
939       storage.push({
940         type: "function",
941         name: "save",
942         'arguments': arguments
943       });
944     },
945     restore: function() {
946       storage.push({
947         type: "function",
948         name: "restore",
949         'arguments': arguments
950       });
951     },
952     fillRect: function () {
953       storage.push({
954         type: "function",
955         name: "fillRect",
956         'arguments': arguments
957       });
958     },
959     createPattern: function() {
960       storage.push({
961         type: "function",
962         name: "createPattern",
963         'arguments': arguments
964       });
965     },
966     drawShape: function() {
967
968       var shape = [];
969
970       storage.push({
971         type: "function",
972         name: "drawShape",
973         'arguments': shape
974       });
975
976       return {
977         moveTo: function() {
978           shape.push({
979             name: "moveTo",
980             'arguments': arguments
981           });
982         },
983         lineTo: function() {
984           shape.push({
985             name: "lineTo",
986             'arguments': arguments
987           });
988         },
989         arcTo: function() {
990           shape.push({
991             name: "arcTo",
992             'arguments': arguments
993           });
994         },
995         bezierCurveTo: function() {
996           shape.push({
997             name: "bezierCurveTo",
998             'arguments': arguments
999           });
1000         },
1001         quadraticCurveTo: function() {
1002           shape.push({
1003             name: "quadraticCurveTo",
1004             'arguments': arguments
1005           });
1006         }
1007       };
1008
1009     },
1010     drawImage: function () {
1011       storage.push({
1012         type: "function",
1013         name: "drawImage",
1014         'arguments': arguments
1015       });
1016     },
1017     fillText: function () {
1018       storage.push({
1019         type: "function",
1020         name: "fillText",
1021         'arguments': arguments
1022       });
1023     },
1024     setVariable: function (variable, value) {
1025       storage.push({
1026         type: "variable",
1027         name: variable,
1028         'arguments': value
1029       });
1030       return value;
1031     }
1032   };
1033 }
1034 _html2canvas.Parse = function (images, options, cb) {
1035   window.scroll(0,0);
1036
1037   var element = (( options.elements === undefined ) ? document.body : options.elements[0]), // select body by default
1038   numDraws = 0,
1039   doc = element.ownerDocument,
1040   Util = _html2canvas.Util,
1041   support = Util.Support(options, doc),
1042   ignoreElementsRegExp = new RegExp("(" + options.ignoreElements + ")"),
1043   body = doc.body,
1044   getCSS = Util.getCSS,
1045   pseudoHide = "___html2canvas___pseudoelement",
1046   hidePseudoElementsStyles = doc.createElement('style');
1047
1048   hidePseudoElementsStyles.innerHTML = '.' + pseudoHide +
1049   '-parent:before { content: "" !important; display: none !important; }' +
1050   '.' + pseudoHide + '-parent:after { content: "" !important; display: none !important; }';
1051
1052   body.appendChild(hidePseudoElementsStyles);
1053
1054   images = images || {};
1055
1056   init();
1057
1058   function init() {
1059     var background = getCSS(document.documentElement, "backgroundColor"),
1060       transparentBackground = (Util.isTransparent(background) && element === document.body),
1061       stack = renderElement(element, null, false, transparentBackground);
1062
1063     // create pseudo elements in a single pass to prevent synchronous layouts
1064     addPseudoElements(element);
1065
1066     parseChildren(element, stack, function() {
1067       if (transparentBackground) {
1068         background = stack.backgroundColor;
1069       }
1070
1071       removePseudoElements();
1072
1073       Util.log('Done parsing, moving to Render.');
1074
1075       cb({
1076         backgroundColor: background,
1077         stack: stack
1078       });
1079     });
1080   }
1081
1082   // Given a root element, find all pseudo elements below, create elements mocking pseudo element styles
1083   // so we can process them as normal elements, and hide the original pseudo elements so they don't interfere
1084   // with layout.
1085   function addPseudoElements(el) {
1086     // These are done in discrete steps to prevent a relayout loop caused by addClass() invalidating
1087     // layouts & getPseudoElement calling getComputedStyle.
1088     var jobs = [], classes = [];
1089     getPseudoElementClasses();
1090     findPseudoElements(el);
1091     runJobs();
1092
1093     function getPseudoElementClasses(){
1094       var findPsuedoEls = /:before|:after/;
1095       var sheets = document.styleSheets;
1096       for (var i = 0, j = sheets.length; i < j; i++) {
1097         try {
1098           var rules = sheets[i].cssRules;
1099           for (var k = 0, l = rules.length; k < l; k++) {
1100             if(findPsuedoEls.test(rules[k].selectorText)) {
1101               classes.push(rules[k].selectorText);
1102             }
1103           }
1104         }
1105         catch(e) { // will throw security exception for style sheets loaded from external domains
1106         }
1107       }
1108
1109       // Trim off the :after and :before (or ::after and ::before)
1110       for (i = 0, j = classes.length; i < j; i++) {
1111         classes[i] = classes[i].match(/(^[^:]*)/)[1];
1112       }
1113     }
1114
1115     // Using the list of elements we know how pseudo el styles, create fake pseudo elements.
1116     function findPseudoElements(el) {
1117       var els = document.querySelectorAll(classes.join(','));
1118       for(var i = 0, j = els.length; i < j; i++) {
1119         createPseudoElements(els[i]);
1120       }
1121     }
1122
1123     // Create pseudo elements & add them to a job queue.
1124     function createPseudoElements(el) {
1125       var before = getPseudoElement(el, ':before'),
1126       after = getPseudoElement(el, ':after');
1127
1128       if(before) {
1129         jobs.push({type: 'before', pseudo: before, el: el});
1130       }
1131
1132       if (after) {
1133         jobs.push({type: 'after', pseudo: after, el: el});
1134       }
1135     }
1136
1137     // Adds a class to the pseudo's parent to prevent the original before/after from messing
1138     // with layouts.
1139     // Execute the inserts & addClass() calls in a batch to prevent relayouts.
1140     function runJobs() {
1141       // Add Class
1142       jobs.forEach(function(job){
1143         addClass(job.el, pseudoHide + "-parent");
1144       });
1145
1146       // Insert el
1147       jobs.forEach(function(job){
1148         if(job.type === 'before'){
1149           job.el.insertBefore(job.pseudo, job.el.firstChild);
1150         } else {
1151           job.el.appendChild(job.pseudo);
1152         }
1153       });
1154     }
1155   }
1156
1157
1158
1159   // Delete our fake pseudo elements from the DOM. This will remove those actual elements
1160   // and the classes on their parents that hide the actual pseudo elements.
1161   // Note that NodeLists are 'live' collections so you can't use a for loop here. They are
1162   // actually deleted from the NodeList after each iteration.
1163   function removePseudoElements(){
1164     // delete pseudo elements
1165     body.removeChild(hidePseudoElementsStyles);
1166     var pseudos = document.getElementsByClassName(pseudoHide + "-element");
1167     while (pseudos.length) {
1168       pseudos[0].parentNode.removeChild(pseudos[0]);
1169     }
1170
1171     // Remove pseudo hiding classes
1172     var parents = document.getElementsByClassName(pseudoHide + "-parent");
1173     while(parents.length) {
1174       removeClass(parents[0], pseudoHide + "-parent");
1175     }
1176   }
1177
1178   function addClass (el, className) {
1179     if (el.classList) {
1180       el.classList.add(className);
1181     } else {
1182       el.className = el.className + " " + className;
1183     }
1184   }
1185
1186   function removeClass (el, className) {
1187     if (el.classList) {
1188       el.classList.remove(className);
1189     } else {
1190       el.className = el.className.replace(className, "").trim();
1191     }
1192   }
1193
1194   function hasClass (el, className) {
1195     return el.className.indexOf(className) > -1;
1196   }
1197
1198   // Note that this doesn't work in < IE8, but we don't support that anyhow
1199   function nodeListToArray (nodeList) {
1200     return Array.prototype.slice.call(nodeList);
1201   }
1202
1203   function documentWidth () {
1204     return Math.max(
1205       Math.max(doc.body.scrollWidth, doc.documentElement.scrollWidth),
1206       Math.max(doc.body.offsetWidth, doc.documentElement.offsetWidth),
1207       Math.max(doc.body.clientWidth, doc.documentElement.clientWidth)
1208       );
1209   }
1210
1211   function documentHeight () {
1212     return Math.max(
1213       Math.max(doc.body.scrollHeight, doc.documentElement.scrollHeight),
1214       Math.max(doc.body.offsetHeight, doc.documentElement.offsetHeight),
1215       Math.max(doc.body.clientHeight, doc.documentElement.clientHeight)
1216       );
1217   }
1218
1219   function getCSSInt(element, attribute) {
1220     var val = parseInt(getCSS(element, attribute), 10);
1221     return (isNaN(val)) ? 0 : val; // borders in old IE are throwing 'medium' for demo.html
1222   }
1223
1224   function renderRect (ctx, x, y, w, h, bgcolor) {
1225     if (bgcolor !== "transparent"){
1226       ctx.setVariable("fillStyle", bgcolor);
1227       ctx.fillRect(x, y, w, h);
1228       numDraws+=1;
1229     }
1230   }
1231
1232   function capitalize(m, p1, p2) {
1233     if (m.length > 0) {
1234       return p1 + p2.toUpperCase();
1235     }
1236   }
1237
1238   function textTransform (text, transform) {
1239     switch(transform){
1240       case "lowercase":
1241         return text.toLowerCase();
1242       case "capitalize":
1243         return text.replace( /(^|\s|:|-|\(|\))([a-z])/g, capitalize);
1244       case "uppercase":
1245         return text.toUpperCase();
1246       default:
1247         return text;
1248     }
1249   }
1250
1251   function noLetterSpacing(letter_spacing) {
1252     return (/^(normal|none|0px)$/.test(letter_spacing));
1253   }
1254
1255   function drawText(currentText, x, y, ctx){
1256     if (currentText !== null && Util.trimText(currentText).length > 0) {
1257       ctx.fillText(currentText, x, y);
1258       numDraws+=1;
1259     }
1260   }
1261
1262   function setTextVariables(ctx, el, text_decoration, color) {
1263     var align = false,
1264     bold = getCSS(el, "fontWeight"),
1265     family = getCSS(el, "fontFamily"),
1266     size = getCSS(el, "fontSize"),
1267     shadows = Util.parseTextShadows(getCSS(el, "textShadow"));
1268
1269     switch(parseInt(bold, 10)){
1270       case 401:
1271         bold = "bold";
1272         break;
1273       case 400:
1274         bold = "normal";
1275         break;
1276     }
1277
1278     ctx.setVariable("fillStyle", color);
1279     ctx.setVariable("font", [getCSS(el, "fontStyle"), getCSS(el, "fontVariant"), bold, size, family].join(" "));
1280     ctx.setVariable("textAlign", (align) ? "right" : "left");
1281
1282     if (shadows.length) {
1283       // TODO: support multiple text shadows
1284       // apply the first text shadow
1285       ctx.setVariable("shadowColor", shadows[0].color);
1286       ctx.setVariable("shadowOffsetX", shadows[0].offsetX);
1287       ctx.setVariable("shadowOffsetY", shadows[0].offsetY);
1288       ctx.setVariable("shadowBlur", shadows[0].blur);
1289     }
1290
1291     if (text_decoration !== "none"){
1292       return Util.Font(family, size, doc);
1293     }
1294   }
1295
1296   function renderTextDecoration(ctx, text_decoration, bounds, metrics, color) {
1297     switch(text_decoration) {
1298       case "underline":
1299         // Draws a line at the baseline of the font
1300         // TODO As some browsers display the line as more than 1px if the font-size is big, need to take that into account both in position and size
1301         renderRect(ctx, bounds.left, Math.round(bounds.top + metrics.baseline + metrics.lineWidth), bounds.width, 1, color);
1302         break;
1303       case "overline":
1304         renderRect(ctx, bounds.left, Math.round(bounds.top), bounds.width, 1, color);
1305         break;
1306       case "line-through":
1307         // TODO try and find exact position for line-through
1308         renderRect(ctx, bounds.left, Math.ceil(bounds.top + metrics.middle + metrics.lineWidth), bounds.width, 1, color);
1309         break;
1310     }
1311   }
1312
1313   function getTextBounds(state, text, textDecoration, isLast, transform) {
1314     var bounds;
1315     if (support.rangeBounds && !transform) {
1316       if (textDecoration !== "none" || Util.trimText(text).length !== 0) {
1317         bounds = textRangeBounds(text, state.node, state.textOffset);
1318       }
1319       state.textOffset += text.length;
1320     } else if (state.node && typeof state.node.nodeValue === "string" ){
1321       var newTextNode = (isLast) ? state.node.splitText(text.length) : null;
1322       bounds = textWrapperBounds(state.node, transform);
1323       state.node = newTextNode;
1324     }
1325     return bounds;
1326   }
1327
1328   function textRangeBounds(text, textNode, textOffset) {
1329     var range = doc.createRange();
1330     range.setStart(textNode, textOffset);
1331     range.setEnd(textNode, textOffset + text.length);
1332     return range.getBoundingClientRect();
1333   }
1334
1335   function textWrapperBounds(oldTextNode, transform) {
1336     var parent = oldTextNode.parentNode,
1337     wrapElement = doc.createElement('wrapper'),
1338     backupText = oldTextNode.cloneNode(true);
1339
1340     wrapElement.appendChild(oldTextNode.cloneNode(true));
1341     parent.replaceChild(wrapElement, oldTextNode);
1342
1343     var bounds = transform ? Util.OffsetBounds(wrapElement) : Util.Bounds(wrapElement);
1344     parent.replaceChild(backupText, wrapElement);
1345     return bounds;
1346   }
1347
1348   function renderText(el, textNode, stack) {
1349     var ctx = stack.ctx,
1350     color = getCSS(el, "color"),
1351     textDecoration = getCSS(el, "textDecoration"),
1352     textAlign = getCSS(el, "textAlign"),
1353     metrics,
1354     textList,
1355     state = {
1356       node: textNode,
1357       textOffset: 0
1358     };
1359
1360     if (Util.trimText(textNode.nodeValue).length > 0) {
1361       textNode.nodeValue = textTransform(textNode.nodeValue, getCSS(el, "textTransform"));
1362       textAlign = textAlign.replace(["-webkit-auto"],["auto"]);
1363
1364       textList = (!options.letterRendering && /^(left|right|justify|auto)$/.test(textAlign) && noLetterSpacing(getCSS(el, "letterSpacing"))) ?
1365       textNode.nodeValue.split(/(\b| )/)
1366       : textNode.nodeValue.split("");
1367
1368       metrics = setTextVariables(ctx, el, textDecoration, color);
1369
1370       if (options.chinese) {
1371         textList.forEach(function(word, index) {
1372           if (/.*[\u4E00-\u9FA5].*$/.test(word)) {
1373             word = word.split("");
1374             word.unshift(index, 1);
1375             textList.splice.apply(textList, word);
1376           }
1377         });
1378       }
1379
1380       textList.forEach(function(text, index) {
1381         var bounds = getTextBounds(state, text, textDecoration, (index < textList.length - 1), stack.transform.matrix);
1382         if (bounds) {
1383           drawText(text, bounds.left, bounds.bottom, ctx);
1384           renderTextDecoration(ctx, textDecoration, bounds, metrics, color);
1385         }
1386       });
1387     }
1388   }
1389
1390   function listPosition (element, val) {
1391     var boundElement = doc.createElement( "boundelement" ),
1392     originalType,
1393     bounds;
1394
1395     boundElement.style.display = "inline";
1396
1397     originalType = element.style.listStyleType;
1398     element.style.listStyleType = "none";
1399
1400     boundElement.appendChild(doc.createTextNode(val));
1401
1402     element.insertBefore(boundElement, element.firstChild);
1403
1404     bounds = Util.Bounds(boundElement);
1405     element.removeChild(boundElement);
1406     element.style.listStyleType = originalType;
1407     return bounds;
1408   }
1409
1410   function elementIndex(el) {
1411     var i = -1,
1412     count = 1,
1413     childs = el.parentNode.childNodes;
1414
1415     if (el.parentNode) {
1416       while(childs[++i] !== el) {
1417         if (childs[i].nodeType === 1) {
1418           count++;
1419         }
1420       }
1421       return count;
1422     } else {
1423       return -1;
1424     }
1425   }
1426
1427   function listItemText(element, type) {
1428     var currentIndex = elementIndex(element), text;
1429     switch(type){
1430       case "decimal":
1431         text = currentIndex;
1432         break;
1433       case "decimal-leading-zero":
1434         text = (currentIndex.toString().length === 1) ? currentIndex = "0" + currentIndex.toString() : currentIndex.toString();
1435         break;
1436       case "upper-roman":
1437         text = _html2canvas.Generate.ListRoman( currentIndex );
1438         break;
1439       case "lower-roman":
1440         text = _html2canvas.Generate.ListRoman( currentIndex ).toLowerCase();
1441         break;
1442       case "lower-alpha":
1443         text = _html2canvas.Generate.ListAlpha( currentIndex ).toLowerCase();
1444         break;
1445       case "upper-alpha":
1446         text = _html2canvas.Generate.ListAlpha( currentIndex );
1447         break;
1448     }
1449
1450     return text + ". ";
1451   }
1452
1453   function renderListItem(element, stack, elBounds) {
1454     var x,
1455     text,
1456     ctx = stack.ctx,
1457     type = getCSS(element, "listStyleType"),
1458     listBounds;
1459
1460     if (/^(decimal|decimal-leading-zero|upper-alpha|upper-latin|upper-roman|lower-alpha|lower-greek|lower-latin|lower-roman)$/i.test(type)) {
1461       text = listItemText(element, type);
1462       listBounds = listPosition(element, text);
1463       setTextVariables(ctx, element, "none", getCSS(element, "color"));
1464
1465       if (getCSS(element, "listStylePosition") === "inside") {
1466         ctx.setVariable("textAlign", "left");
1467         x = elBounds.left;
1468       } else {
1469         return;
1470       }
1471
1472       drawText(text, x, listBounds.bottom, ctx);
1473     }
1474   }
1475
1476   function loadImage (src){
1477     var img = images[src];
1478     return (img && img.succeeded === true) ? img.img : false;
1479   }
1480
1481   function clipBounds(src, dst){
1482     var x = Math.max(src.left, dst.left),
1483     y = Math.max(src.top, dst.top),
1484     x2 = Math.min((src.left + src.width), (dst.left + dst.width)),
1485     y2 = Math.min((src.top + src.height), (dst.top + dst.height));
1486
1487     return {
1488       left:x,
1489       top:y,
1490       width:x2-x,
1491       height:y2-y
1492     };
1493   }
1494
1495   function setZ(element, stack, parentStack){
1496     var newContext,
1497     isPositioned = stack.cssPosition !== 'static',
1498     zIndex = isPositioned ? getCSS(element, 'zIndex') : 'auto',
1499     opacity = getCSS(element, 'opacity'),
1500     isFloated = getCSS(element, 'cssFloat') !== 'none';
1501
1502     // https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Understanding_z_index/The_stacking_context
1503     // When a new stacking context should be created:
1504     // the root element (HTML),
1505     // positioned (absolutely or relatively) with a z-index value other than "auto",
1506     // elements with an opacity value less than 1. (See the specification for opacity),
1507     // on mobile WebKit and Chrome 22+, position: fixed always creates a new stacking context, even when z-index is "auto" (See this post)
1508
1509     stack.zIndex = newContext = h2czContext(zIndex);
1510     newContext.isPositioned = isPositioned;
1511     newContext.isFloated = isFloated;
1512     newContext.opacity = opacity;
1513     newContext.ownStacking = (zIndex !== 'auto' || opacity < 1);
1514     newContext.depth = parentStack ? (parentStack.zIndex.depth + 1) : 0;
1515
1516     if (parentStack) {
1517       parentStack.zIndex.children.push(stack);
1518     }
1519   }
1520
1521   function h2czContext(zindex) {
1522     return {
1523       depth: 0,
1524       zindex: zindex,
1525       children: []
1526     };
1527   }
1528
1529   function renderImage(ctx, element, image, bounds, borders) {
1530
1531     var paddingLeft = getCSSInt(element, 'paddingLeft'),
1532     paddingTop = getCSSInt(element, 'paddingTop'),
1533     paddingRight = getCSSInt(element, 'paddingRight'),
1534     paddingBottom = getCSSInt(element, 'paddingBottom');
1535
1536     drawImage(
1537       ctx,
1538       image,
1539       0, //sx
1540       0, //sy
1541       image.width, //sw
1542       image.height, //sh
1543       bounds.left + paddingLeft + borders[3].width, //dx
1544       bounds.top + paddingTop + borders[0].width, // dy
1545       bounds.width - (borders[1].width + borders[3].width + paddingLeft + paddingRight), //dw
1546       bounds.height - (borders[0].width + borders[2].width + paddingTop + paddingBottom) //dh
1547       );
1548   }
1549
1550   function getBorderData(element) {
1551     return ["Top", "Right", "Bottom", "Left"].map(function(side) {
1552       return {
1553         width: getCSSInt(element, 'border' + side + 'Width'),
1554         color: getCSS(element, 'border' + side + 'Color')
1555       };
1556     });
1557   }
1558
1559   function getBorderRadiusData(element) {
1560     return ["TopLeft", "TopRight", "BottomRight", "BottomLeft"].map(function(side) {
1561       return getCSS(element, 'border' + side + 'Radius');
1562     });
1563   }
1564
1565   function getCurvePoints(x, y, r1, r2) {
1566     var kappa = 4 * ((Math.sqrt(2) - 1) / 3);
1567     var ox = (r1) * kappa, // control point offset horizontal
1568     oy = (r2) * kappa, // control point offset vertical
1569     xm = x + r1, // x-middle
1570     ym = y + r2; // y-middle
1571     return {
1572       topLeft: bezierCurve({
1573         x:x,
1574         y:ym
1575       }, {
1576         x:x,
1577         y:ym - oy
1578       }, {
1579         x:xm - ox,
1580         y:y
1581       }, {
1582         x:xm,
1583         y:y
1584       }),
1585       topRight: bezierCurve({
1586         x:x,
1587         y:y
1588       }, {
1589         x:x + ox,
1590         y:y
1591       }, {
1592         x:xm,
1593         y:ym - oy
1594       }, {
1595         x:xm,
1596         y:ym
1597       }),
1598       bottomRight: bezierCurve({
1599         x:xm,
1600         y:y
1601       }, {
1602         x:xm,
1603         y:y + oy
1604       }, {
1605         x:x + ox,
1606         y:ym
1607       }, {
1608         x:x,
1609         y:ym
1610       }),
1611       bottomLeft: bezierCurve({
1612         x:xm,
1613         y:ym
1614       }, {
1615         x:xm - ox,
1616         y:ym
1617       }, {
1618         x:x,
1619         y:y + oy
1620       }, {
1621         x:x,
1622         y:y
1623       })
1624     };
1625   }
1626
1627   function bezierCurve(start, startControl, endControl, end) {
1628
1629     var lerp = function (a, b, t) {
1630       return {
1631         x:a.x + (b.x - a.x) * t,
1632         y:a.y + (b.y - a.y) * t
1633       };
1634     };
1635
1636     return {
1637       start: start,
1638       startControl: startControl,
1639       endControl: endControl,
1640       end: end,
1641       subdivide: function(t) {
1642         var ab = lerp(start, startControl, t),
1643         bc = lerp(startControl, endControl, t),
1644         cd = lerp(endControl, end, t),
1645         abbc = lerp(ab, bc, t),
1646         bccd = lerp(bc, cd, t),
1647         dest = lerp(abbc, bccd, t);
1648         return [bezierCurve(start, ab, abbc, dest), bezierCurve(dest, bccd, cd, end)];
1649       },
1650       curveTo: function(borderArgs) {
1651         borderArgs.push(["bezierCurve", startControl.x, startControl.y, endControl.x, endControl.y, end.x, end.y]);
1652       },
1653       curveToReversed: function(borderArgs) {
1654         borderArgs.push(["bezierCurve", endControl.x, endControl.y, startControl.x, startControl.y, start.x, start.y]);
1655       }
1656     };
1657   }
1658
1659   function parseCorner(borderArgs, radius1, radius2, corner1, corner2, x, y) {
1660     if (radius1[0] > 0 || radius1[1] > 0) {
1661       borderArgs.push(["line", corner1[0].start.x, corner1[0].start.y]);
1662       corner1[0].curveTo(borderArgs);
1663       corner1[1].curveTo(borderArgs);
1664     } else {
1665       borderArgs.push(["line", x, y]);
1666     }
1667
1668     if (radius2[0] > 0 || radius2[1] > 0) {
1669       borderArgs.push(["line", corner2[0].start.x, corner2[0].start.y]);
1670     }
1671   }
1672
1673   function drawSide(borderData, radius1, radius2, outer1, inner1, outer2, inner2) {
1674     var borderArgs = [];
1675
1676     if (radius1[0] > 0 || radius1[1] > 0) {
1677       borderArgs.push(["line", outer1[1].start.x, outer1[1].start.y]);
1678       outer1[1].curveTo(borderArgs);
1679     } else {
1680       borderArgs.push([ "line", borderData.c1[0], borderData.c1[1]]);
1681     }
1682
1683     if (radius2[0] > 0 || radius2[1] > 0) {
1684       borderArgs.push(["line", outer2[0].start.x, outer2[0].start.y]);
1685       outer2[0].curveTo(borderArgs);
1686       borderArgs.push(["line", inner2[0].end.x, inner2[0].end.y]);
1687       inner2[0].curveToReversed(borderArgs);
1688     } else {
1689       borderArgs.push([ "line", borderData.c2[0], borderData.c2[1]]);
1690       borderArgs.push([ "line", borderData.c3[0], borderData.c3[1]]);
1691     }
1692
1693     if (radius1[0] > 0 || radius1[1] > 0) {
1694       borderArgs.push(["line", inner1[1].end.x, inner1[1].end.y]);
1695       inner1[1].curveToReversed(borderArgs);
1696     } else {
1697       borderArgs.push([ "line", borderData.c4[0], borderData.c4[1]]);
1698     }
1699
1700     return borderArgs;
1701   }
1702
1703   function calculateCurvePoints(bounds, borderRadius, borders) {
1704
1705     var x = bounds.left,
1706     y = bounds.top,
1707     width = bounds.width,
1708     height = bounds.height,
1709
1710     tlh = borderRadius[0][0],
1711     tlv = borderRadius[0][1],
1712     trh = borderRadius[1][0],
1713     trv = borderRadius[1][1],
1714     brh = borderRadius[2][0],
1715     brv = borderRadius[2][1],
1716     blh = borderRadius[3][0],
1717     blv = borderRadius[3][1],
1718
1719     topWidth = width - trh,
1720     rightHeight = height - brv,
1721     bottomWidth = width - brh,
1722     leftHeight = height - blv;
1723
1724     return {
1725       topLeftOuter: getCurvePoints(
1726         x,
1727         y,
1728         tlh,
1729         tlv
1730         ).topLeft.subdivide(0.5),
1731
1732       topLeftInner: getCurvePoints(
1733         x + borders[3].width,
1734         y + borders[0].width,
1735         Math.max(0, tlh - borders[3].width),
1736         Math.max(0, tlv - borders[0].width)
1737         ).topLeft.subdivide(0.5),
1738
1739       topRightOuter: getCurvePoints(
1740         x + topWidth,
1741         y,
1742         trh,
1743         trv
1744         ).topRight.subdivide(0.5),
1745
1746       topRightInner: getCurvePoints(
1747         x + Math.min(topWidth, width + borders[3].width),
1748         y + borders[0].width,
1749         (topWidth > width + borders[3].width) ? 0 :trh - borders[3].width,
1750         trv - borders[0].width
1751         ).topRight.subdivide(0.5),
1752
1753       bottomRightOuter: getCurvePoints(
1754         x + bottomWidth,
1755         y + rightHeight,
1756         brh,
1757         brv
1758         ).bottomRight.subdivide(0.5),
1759
1760       bottomRightInner: getCurvePoints(
1761         x + Math.min(bottomWidth, width + borders[3].width),
1762         y + Math.min(rightHeight, height + borders[0].width),
1763         Math.max(0, brh - borders[1].width),
1764         Math.max(0, brv - borders[2].width)
1765         ).bottomRight.subdivide(0.5),
1766
1767       bottomLeftOuter: getCurvePoints(
1768         x,
1769         y + leftHeight,
1770         blh,
1771         blv
1772         ).bottomLeft.subdivide(0.5),
1773
1774       bottomLeftInner: getCurvePoints(
1775         x + borders[3].width,
1776         y + leftHeight,
1777         Math.max(0, blh - borders[3].width),
1778         Math.max(0, blv - borders[2].width)
1779         ).bottomLeft.subdivide(0.5)
1780     };
1781   }
1782
1783   function getBorderClip(element, borderPoints, borders, radius, bounds) {
1784     var backgroundClip = getCSS(element, 'backgroundClip'),
1785     borderArgs = [];
1786
1787     switch(backgroundClip) {
1788       case "content-box":
1789       case "padding-box":
1790         parseCorner(borderArgs, radius[0], radius[1], borderPoints.topLeftInner, borderPoints.topRightInner, bounds.left + borders[3].width, bounds.top + borders[0].width);
1791         parseCorner(borderArgs, radius[1], radius[2], borderPoints.topRightInner, borderPoints.bottomRightInner, bounds.left + bounds.width - borders[1].width, bounds.top + borders[0].width);
1792         parseCorner(borderArgs, radius[2], radius[3], borderPoints.bottomRightInner, borderPoints.bottomLeftInner, bounds.left + bounds.width - borders[1].width, bounds.top + bounds.height - borders[2].width);
1793         parseCorner(borderArgs, radius[3], radius[0], borderPoints.bottomLeftInner, borderPoints.topLeftInner, bounds.left + borders[3].width, bounds.top + bounds.height - borders[2].width);
1794         break;
1795
1796       default:
1797         parseCorner(borderArgs, radius[0], radius[1], borderPoints.topLeftOuter, borderPoints.topRightOuter, bounds.left, bounds.top);
1798         parseCorner(borderArgs, radius[1], radius[2], borderPoints.topRightOuter, borderPoints.bottomRightOuter, bounds.left + bounds.width, bounds.top);
1799         parseCorner(borderArgs, radius[2], radius[3], borderPoints.bottomRightOuter, borderPoints.bottomLeftOuter, bounds.left + bounds.width, bounds.top + bounds.height);
1800         parseCorner(borderArgs, radius[3], radius[0], borderPoints.bottomLeftOuter, borderPoints.topLeftOuter, bounds.left, bounds.top + bounds.height);
1801         break;
1802     }
1803
1804     return borderArgs;
1805   }
1806
1807   function parseBorders(element, bounds, borders){
1808     var x = bounds.left,
1809     y = bounds.top,
1810     width = bounds.width,
1811     height = bounds.height,
1812     borderSide,
1813     bx,
1814     by,
1815     bw,
1816     bh,
1817     borderArgs,
1818     // http://www.w3.org/TR/css3-background/#the-border-radius
1819     borderRadius = getBorderRadiusData(element),
1820     borderPoints = calculateCurvePoints(bounds, borderRadius, borders),
1821     borderData = {
1822       clip: getBorderClip(element, borderPoints, borders, borderRadius, bounds),
1823       borders: []
1824     };
1825
1826     for (borderSide = 0; borderSide < 4; borderSide++) {
1827
1828       if (borders[borderSide].width > 0) {
1829         bx = x;
1830         by = y;
1831         bw = width;
1832         bh = height - (borders[2].width);
1833
1834         switch(borderSide) {
1835           case 0:
1836             // top border
1837             bh = borders[0].width;
1838
1839             borderArgs = drawSide({
1840               c1: [bx, by],
1841               c2: [bx + bw, by],
1842               c3: [bx + bw - borders[1].width, by + bh],
1843               c4: [bx + borders[3].width, by + bh]
1844             }, borderRadius[0], borderRadius[1],
1845             borderPoints.topLeftOuter, borderPoints.topLeftInner, borderPoints.topRightOuter, borderPoints.topRightInner);
1846             break;
1847           case 1:
1848             // right border
1849             bx = x + width - (borders[1].width);
1850             bw = borders[1].width;
1851
1852             borderArgs = drawSide({
1853               c1: [bx + bw, by],
1854               c2: [bx + bw, by + bh + borders[2].width],
1855               c3: [bx, by + bh],
1856               c4: [bx, by + borders[0].width]
1857             }, borderRadius[1], borderRadius[2],
1858             borderPoints.topRightOuter, borderPoints.topRightInner, borderPoints.bottomRightOuter, borderPoints.bottomRightInner);
1859             break;
1860           case 2:
1861             // bottom border
1862             by = (by + height) - (borders[2].width);
1863             bh = borders[2].width;
1864
1865             borderArgs = drawSide({
1866               c1: [bx + bw, by + bh],
1867               c2: [bx, by + bh],
1868               c3: [bx + borders[3].width, by],
1869               c4: [bx + bw - borders[3].width, by]
1870             }, borderRadius[2], borderRadius[3],
1871             borderPoints.bottomRightOuter, borderPoints.bottomRightInner, borderPoints.bottomLeftOuter, borderPoints.bottomLeftInner);
1872             break;
1873           case 3:
1874             // left border
1875             bw = borders[3].width;
1876
1877             borderArgs = drawSide({
1878               c1: [bx, by + bh + borders[2].width],
1879               c2: [bx, by],
1880               c3: [bx + bw, by + borders[0].width],
1881               c4: [bx + bw, by + bh]
1882             }, borderRadius[3], borderRadius[0],
1883             borderPoints.bottomLeftOuter, borderPoints.bottomLeftInner, borderPoints.topLeftOuter, borderPoints.topLeftInner);
1884             break;
1885         }
1886
1887         borderData.borders.push({
1888           args: borderArgs,
1889           color: borders[borderSide].color
1890         });
1891
1892       }
1893     }
1894
1895     return borderData;
1896   }
1897
1898   function createShape(ctx, args) {
1899     var shape = ctx.drawShape();
1900     args.forEach(function(border, index) {
1901       shape[(index === 0) ? "moveTo" : border[0] + "To" ].apply(null, border.slice(1));
1902     });
1903     return shape;
1904   }
1905
1906   function renderBorders(ctx, borderArgs, color) {
1907     if (color !== "transparent") {
1908       ctx.setVariable( "fillStyle", color);
1909       createShape(ctx, borderArgs);
1910       ctx.fill();
1911       numDraws+=1;
1912     }
1913   }
1914
1915   function renderFormValue (el, bounds, stack){
1916
1917     var valueWrap = doc.createElement('valuewrap'),
1918     cssPropertyArray = ['lineHeight','textAlign','fontFamily','color','fontSize','paddingLeft','paddingTop','width','height','border','borderLeftWidth','borderTopWidth'],
1919     textValue,
1920     textNode;
1921
1922     cssPropertyArray.forEach(function(property) {
1923       try {
1924         valueWrap.style[property] = getCSS(el, property);
1925       } catch(e) {
1926         // Older IE has issues with "border"
1927         Util.log("html2canvas: Parse: Exception caught in renderFormValue: " + e.message);
1928       }
1929     });
1930
1931     valueWrap.style.borderColor = "black";
1932     valueWrap.style.borderStyle = "solid";
1933     valueWrap.style.display = "block";
1934     valueWrap.style.position = "absolute";
1935
1936     if (/^(submit|reset|button|text|password)$/.test(el.type) || el.nodeName === "SELECT"){
1937       valueWrap.style.lineHeight = getCSS(el, "height");
1938     }
1939
1940     valueWrap.style.top = bounds.top + "px";
1941     valueWrap.style.left = bounds.left + "px";
1942
1943     textValue = (el.nodeName === "SELECT") ? (el.options[el.selectedIndex] || 0).text : el.value;
1944     if(!textValue) {
1945       textValue = el.placeholder;
1946     }
1947
1948     textNode = doc.createTextNode(textValue);
1949
1950     valueWrap.appendChild(textNode);
1951     body.appendChild(valueWrap);
1952
1953     renderText(el, textNode, stack);
1954     body.removeChild(valueWrap);
1955   }
1956
1957   function drawImage (ctx) {
1958     ctx.drawImage.apply(ctx, Array.prototype.slice.call(arguments, 1));
1959     numDraws+=1;
1960   }
1961
1962   function getPseudoElement(el, which) {
1963     var elStyle = window.getComputedStyle(el, which);
1964     var parentStyle = window.getComputedStyle(el);
1965     // If no content attribute is present, the pseudo element is hidden,
1966     // or the parent has a content property equal to the content on the pseudo element,
1967     // move along.
1968     if(!elStyle || !elStyle.content || elStyle.content === "none" || elStyle.content === "-moz-alt-content" ||
1969        elStyle.display === "none" || parentStyle.content === elStyle.content) {
1970       return;
1971     }
1972     var content = elStyle.content + '';
1973
1974     // Strip inner quotes
1975     if(content[0] === "'" || content[0] === "\"") {
1976       content = content.replace(/(^['"])|(['"]$)/g, '');
1977     }
1978
1979     var isImage = content.substr( 0, 3 ) === 'url',
1980     elps = document.createElement( isImage ? 'img' : 'span' );
1981
1982     elps.className = pseudoHide + "-element ";
1983
1984     Object.keys(elStyle).filter(indexedProperty).forEach(function(prop) {
1985       // Prevent assigning of read only CSS Rules, ex. length, parentRule
1986       try {
1987         elps.style[prop] = elStyle[prop];
1988       } catch (e) {
1989         Util.log(['Tried to assign readonly property ', prop, 'Error:', e]);
1990       }
1991     });
1992
1993     if(isImage) {
1994       elps.src = Util.parseBackgroundImage(content)[0].args[0];
1995     } else {
1996       elps.innerHTML = content;
1997     }
1998     return elps;
1999   }
2000
2001   function indexedProperty(property) {
2002     return (isNaN(window.parseInt(property, 10)));
2003   }
2004
2005   function renderBackgroundRepeat(ctx, image, backgroundPosition, bounds) {
2006     var offsetX = Math.round(bounds.left + backgroundPosition.left),
2007     offsetY = Math.round(bounds.top + backgroundPosition.top);
2008
2009     ctx.createPattern(image);
2010     ctx.translate(offsetX, offsetY);
2011     ctx.fill();
2012     ctx.translate(-offsetX, -offsetY);
2013   }
2014
2015   function backgroundRepeatShape(ctx, image, backgroundPosition, bounds, left, top, width, height) {
2016     var args = [];
2017     args.push(["line", Math.round(left), Math.round(top)]);
2018     args.push(["line", Math.round(left + width), Math.round(top)]);
2019     args.push(["line", Math.round(left + width), Math.round(height + top)]);
2020     args.push(["line", Math.round(left), Math.round(height + top)]);
2021     createShape(ctx, args);
2022     ctx.save();
2023     ctx.clip();
2024     renderBackgroundRepeat(ctx, image, backgroundPosition, bounds);
2025     ctx.restore();
2026   }
2027
2028   function renderBackgroundColor(ctx, backgroundBounds, bgcolor) {
2029     renderRect(
2030       ctx,
2031       backgroundBounds.left,
2032       backgroundBounds.top,
2033       backgroundBounds.width,
2034       backgroundBounds.height,
2035       bgcolor
2036       );
2037   }
2038
2039   function renderBackgroundRepeating(el, bounds, ctx, image, imageIndex) {
2040     var backgroundSize = Util.BackgroundSize(el, bounds, image, imageIndex),
2041     backgroundPosition = Util.BackgroundPosition(el, bounds, image, imageIndex, backgroundSize),
2042     backgroundRepeat = Util.BackgroundRepeat(el, imageIndex);
2043
2044     image = resizeImage(image, backgroundSize);
2045
2046     switch (backgroundRepeat) {
2047       case "repeat-x":
2048       case "repeat no-repeat":
2049         backgroundRepeatShape(ctx, image, backgroundPosition, bounds,
2050           bounds.left, bounds.top + backgroundPosition.top, 99999, image.height);
2051         break;
2052       case "repeat-y":
2053       case "no-repeat repeat":
2054         backgroundRepeatShape(ctx, image, backgroundPosition, bounds,
2055           bounds.left + backgroundPosition.left, bounds.top, image.width, 99999);
2056         break;
2057       case "no-repeat":
2058         backgroundRepeatShape(ctx, image, backgroundPosition, bounds,
2059           bounds.left + backgroundPosition.left, bounds.top + backgroundPosition.top, image.width, image.height);
2060         break;
2061       default:
2062         renderBackgroundRepeat(ctx, image, backgroundPosition, {
2063           top: bounds.top,
2064           left: bounds.left,
2065           width: image.width,
2066           height: image.height
2067         });
2068         break;
2069     }
2070   }
2071
2072   function renderBackgroundImage(element, bounds, ctx) {
2073     var backgroundImage = getCSS(element, "backgroundImage"),
2074     backgroundImages = Util.parseBackgroundImage(backgroundImage),
2075     image,
2076     imageIndex = backgroundImages.length;
2077
2078     while(imageIndex--) {
2079       backgroundImage = backgroundImages[imageIndex];
2080
2081       if (!backgroundImage.args || backgroundImage.args.length === 0) {
2082         continue;
2083       }
2084
2085       var key = backgroundImage.method === 'url' ?
2086       backgroundImage.args[0] :
2087       backgroundImage.value;
2088
2089       image = loadImage(key);
2090
2091       // TODO add support for background-origin
2092       if (image) {
2093         renderBackgroundRepeating(element, bounds, ctx, image, imageIndex);
2094       } else {
2095         Util.log("html2canvas: Error loading background:", backgroundImage);
2096       }
2097     }
2098   }
2099
2100   function resizeImage(image, bounds) {
2101     if(image.width === bounds.width && image.height === bounds.height) {
2102       return image;
2103     }
2104
2105     var ctx, canvas = doc.createElement('canvas');
2106     canvas.width = bounds.width;
2107     canvas.height = bounds.height;
2108     ctx = canvas.getContext("2d");
2109     drawImage(ctx, image, 0, 0, image.width, image.height, 0, 0, bounds.width, bounds.height );
2110     return canvas;
2111   }
2112
2113   function setOpacity(ctx, element, parentStack) {
2114     return ctx.setVariable("globalAlpha", getCSS(element, "opacity") * ((parentStack) ? parentStack.opacity : 1));
2115   }
2116
2117   function removePx(str) {
2118     return str.replace("px", "");
2119   }
2120
2121   function getTransform(element, parentStack) {
2122     var transformRegExp = /(matrix)\((.+)\)/;
2123     var transform = getCSS(element, "transform") || getCSS(element, "-webkit-transform") || getCSS(element, "-moz-transform") || getCSS(element, "-ms-transform") || getCSS(element, "-o-transform");
2124     var transformOrigin = getCSS(element, "transform-origin") || getCSS(element, "-webkit-transform-origin") || getCSS(element, "-moz-transform-origin") || getCSS(element, "-ms-transform-origin") || getCSS(element, "-o-transform-origin") || "0px 0px";
2125
2126     transformOrigin = transformOrigin.split(" ").map(removePx).map(Util.asFloat);
2127
2128     var matrix;
2129     if (transform && transform !== "none") {
2130       var match = transform.match(transformRegExp);
2131       if (match) {
2132         switch(match[1]) {
2133           case "matrix":
2134             matrix = match[2].split(",").map(Util.trimText).map(Util.asFloat);
2135             break;
2136         }
2137       }
2138     }
2139
2140     return {
2141       origin: transformOrigin,
2142       matrix: matrix
2143     };
2144   }
2145
2146   function createStack(element, parentStack, bounds, transform) {
2147     var ctx = h2cRenderContext((!parentStack) ? documentWidth() : bounds.width , (!parentStack) ? documentHeight() : bounds.height),
2148     stack = {
2149       ctx: ctx,
2150       opacity: setOpacity(ctx, element, parentStack),
2151       cssPosition: getCSS(element, "position"),
2152       borders: getBorderData(element),
2153       transform: transform,
2154       clip: (parentStack && parentStack.clip) ? Util.Extend( {}, parentStack.clip ) : null
2155     };
2156
2157     setZ(element, stack, parentStack);
2158
2159     // TODO correct overflow for absolute content residing under a static position
2160     if (options.useOverflow === true && /(hidden|scroll|auto)/.test(getCSS(element, "overflow")) === true && /(BODY)/i.test(element.nodeName) === false){
2161       stack.clip = (stack.clip) ? clipBounds(stack.clip, bounds) : bounds;
2162     }
2163
2164     return stack;
2165   }
2166
2167   function getBackgroundBounds(borders, bounds, clip) {
2168     var backgroundBounds = {
2169       left: bounds.left + borders[3].width,
2170       top: bounds.top + borders[0].width,
2171       width: bounds.width - (borders[1].width + borders[3].width),
2172       height: bounds.height - (borders[0].width + borders[2].width)
2173     };
2174
2175     if (clip) {
2176       backgroundBounds = clipBounds(backgroundBounds, clip);
2177     }
2178
2179     return backgroundBounds;
2180   }
2181
2182   function getBounds(element, transform) {
2183     var bounds = (transform.matrix) ? Util.OffsetBounds(element) : Util.Bounds(element);
2184     transform.origin[0] += bounds.left;
2185     transform.origin[1] += bounds.top;
2186     return bounds;
2187   }
2188
2189   function renderElement(element, parentStack, ignoreBackground) {
2190     var transform = getTransform(element, parentStack),
2191     bounds = getBounds(element, transform),
2192     image,
2193     stack = createStack(element, parentStack, bounds, transform),
2194     borders = stack.borders,
2195     ctx = stack.ctx,
2196     backgroundBounds = getBackgroundBounds(borders, bounds, stack.clip),
2197     borderData = parseBorders(element, bounds, borders),
2198     backgroundColor = (ignoreElementsRegExp.test(element.nodeName)) ? "#efefef" : getCSS(element, "backgroundColor");
2199
2200
2201     createShape(ctx, borderData.clip);
2202
2203     ctx.save();
2204     ctx.clip();
2205
2206     if (backgroundBounds.height > 0 && backgroundBounds.width > 0 && !ignoreBackground) {
2207       renderBackgroundColor(ctx, bounds, backgroundColor);
2208       renderBackgroundImage(element, backgroundBounds, ctx);
2209     } else if (ignoreBackground) {
2210       stack.backgroundColor =  backgroundColor;
2211     }
2212
2213     ctx.restore();
2214
2215     borderData.borders.forEach(function(border) {
2216       renderBorders(ctx, border.args, border.color);
2217     });
2218
2219     switch(element.nodeName){
2220       case "IMG":
2221         if ((image = loadImage(element.getAttribute('src')))) {
2222           renderImage(ctx, element, image, bounds, borders);
2223         } else {
2224           Util.log("html2canvas: Error loading <img>:" + element.getAttribute('src'));
2225         }
2226         break;
2227       case "INPUT":
2228         // TODO add all relevant type's, i.e. HTML5 new stuff
2229         // todo add support for placeholder attribute for browsers which support it
2230         if (/^(text|url|email|submit|button|reset)$/.test(element.type) && (element.value || element.placeholder || "").length > 0){
2231           renderFormValue(element, bounds, stack);
2232         }
2233         break;
2234       case "TEXTAREA":
2235         if ((element.value || element.placeholder || "").length > 0){
2236           renderFormValue(element, bounds, stack);
2237         }
2238         break;
2239       case "SELECT":
2240         if ((element.options||element.placeholder || "").length > 0){
2241           renderFormValue(element, bounds, stack);
2242         }
2243         break;
2244       case "LI":
2245         renderListItem(element, stack, backgroundBounds);
2246         break;
2247       case "CANVAS":
2248         renderImage(ctx, element, element, bounds, borders);
2249         break;
2250     }
2251
2252     return stack;
2253   }
2254
2255   function isElementVisible(element) {
2256     return (getCSS(element, 'display') !== "none" && getCSS(element, 'visibility') !== "hidden" && !element.hasAttribute("data-html2canvas-ignore"));
2257   }
2258
2259   function parseElement (element, stack, cb) {
2260     if (!cb) {
2261       cb = function(){};
2262     }
2263     if (isElementVisible(element)) {
2264       stack = renderElement(element, stack, false) || stack;
2265       if (!ignoreElementsRegExp.test(element.nodeName)) {
2266         return parseChildren(element, stack, cb);
2267       }
2268     }
2269     cb();
2270   }
2271
2272   function parseChildren(element, stack, cb) {
2273     var children = Util.Children(element);
2274     // After all nodes have processed, finished() will call the cb.
2275     // We add one and kick it off so this will still work when children.length === 0.
2276     // Note that unless async is true, this will happen synchronously, just will callbacks.
2277     var jobs = children.length + 1;
2278     finished();
2279
2280     if (options.async) {
2281       children.forEach(function(node) {
2282         // Don't block the page from rendering
2283         setTimeout(function(){ parseNode(node); }, 0);
2284       });
2285     } else {
2286       children.forEach(parseNode);
2287     }
2288
2289     function parseNode(node) {
2290       if (node.nodeType === node.ELEMENT_NODE) {
2291         parseElement(node, stack, finished);
2292       } else if (node.nodeType === node.TEXT_NODE) {
2293         renderText(element, node, stack);
2294         finished();
2295       } else {
2296         finished();
2297       }
2298     }
2299     function finished(el) {
2300       if (--jobs <= 0){
2301         Util.log("finished rendering " + children.length + " children.");
2302         cb();
2303       }
2304     }
2305   }
2306 };
2307 _html2canvas.Preload = function( options ) {
2308
2309   var images = {
2310     numLoaded: 0,   // also failed are counted here
2311     numFailed: 0,
2312     numTotal: 0,
2313     cleanupDone: false
2314   },
2315   pageOrigin,
2316   Util = _html2canvas.Util,
2317   methods,
2318   i,
2319   count = 0,
2320   element = options.elements[0] || document.body,
2321   doc = element.ownerDocument,
2322   domImages = element.getElementsByTagName('img'), // Fetch images of the present element only
2323   imgLen = domImages.length,
2324   link = doc.createElement("a"),
2325   supportCORS = (function( img ){
2326     return (img.crossOrigin !== undefined);
2327   })(new Image()),
2328   timeoutTimer;
2329
2330   link.href = window.location.href;
2331   pageOrigin  = link.protocol + link.host;
2332
2333   function isSameOrigin(url){
2334     link.href = url;
2335     link.href = link.href; // YES, BELIEVE IT OR NOT, that is required for IE9 - http://jsfiddle.net/niklasvh/2e48b/
2336     var origin = link.protocol + link.host;
2337     return (origin === pageOrigin);
2338   }
2339
2340   function start(){
2341     Util.log("html2canvas: start: images: " + images.numLoaded + " / " + images.numTotal + " (failed: " + images.numFailed + ")");
2342     if (!images.firstRun && images.numLoaded >= images.numTotal){
2343       Util.log("Finished loading images: # " + images.numTotal + " (failed: " + images.numFailed + ")");
2344
2345       if (typeof options.complete === "function"){
2346         options.complete(images);
2347       }
2348
2349     }
2350   }
2351
2352   // TODO modify proxy to serve images with CORS enabled, where available
2353   function proxyGetImage(url, img, imageObj){
2354     var callback_name,
2355     scriptUrl = options.proxy,
2356     script;
2357
2358     link.href = url;
2359     url = link.href; // work around for pages with base href="" set - WARNING: this may change the url
2360
2361     callback_name = 'html2canvas_' + (count++);
2362     imageObj.callbackname = callback_name;
2363
2364     if (scriptUrl.indexOf("?") > -1) {
2365       scriptUrl += "&";
2366     } else {
2367       scriptUrl += "?";
2368     }
2369     scriptUrl += 'url=' + encodeURIComponent(url) + '&callback=' + callback_name;
2370     script = doc.createElement("script");
2371
2372     window[callback_name] = function(a){
2373       if (a.substring(0,6) === "error:"){
2374         imageObj.succeeded = false;
2375         images.numLoaded++;
2376         images.numFailed++;
2377         start();
2378       } else {
2379         setImageLoadHandlers(img, imageObj);
2380         img.src = a;
2381       }
2382       window[callback_name] = undefined; // to work with IE<9  // NOTE: that the undefined callback property-name still exists on the window object (for IE<9)
2383       try {
2384         delete window[callback_name];  // for all browser that support this
2385       } catch(ex) {}
2386       script.parentNode.removeChild(script);
2387       script = null;
2388       delete imageObj.script;
2389       delete imageObj.callbackname;
2390     };
2391
2392     script.setAttribute("type", "text/javascript");
2393     script.setAttribute("src", scriptUrl);
2394     imageObj.script = script;
2395     window.document.body.appendChild(script);
2396
2397   }
2398
2399   function loadPseudoElement(element, type) {
2400     var style = window.getComputedStyle(element, type),
2401     content = style.content;
2402     if (content.substr(0, 3) === 'url') {
2403       methods.loadImage(_html2canvas.Util.parseBackgroundImage(content)[0].args[0]);
2404     }
2405     loadBackgroundImages(style.backgroundImage, element);
2406   }
2407
2408   function loadPseudoElementImages(element) {
2409     loadPseudoElement(element, ":before");
2410     loadPseudoElement(element, ":after");
2411   }
2412
2413   function loadGradientImage(backgroundImage, bounds) {
2414     var img = _html2canvas.Generate.Gradient(backgroundImage, bounds);
2415
2416     if (img !== undefined){
2417       images[backgroundImage] = {
2418         img: img,
2419         succeeded: true
2420       };
2421       images.numTotal++;
2422       images.numLoaded++;
2423       start();
2424     }
2425   }
2426
2427   function invalidBackgrounds(background_image) {
2428     return (background_image && background_image.method && background_image.args && background_image.args.length > 0 );
2429   }
2430
2431   function loadBackgroundImages(background_image, el) {
2432     var bounds;
2433
2434     _html2canvas.Util.parseBackgroundImage(background_image).filter(invalidBackgrounds).forEach(function(background_image) {
2435       if (background_image.method === 'url') {
2436         methods.loadImage(background_image.args[0]);
2437       } else if(background_image.method.match(/\-?gradient$/)) {
2438         if(bounds === undefined) {
2439           bounds = _html2canvas.Util.Bounds(el);
2440         }
2441         loadGradientImage(background_image.value, bounds);
2442       }
2443     });
2444   }
2445
2446   function getImages (el) {
2447     var elNodeType = false;
2448
2449     // Firefox fails with permission denied on pages with iframes
2450     try {
2451       Util.Children(el).forEach(getImages);
2452     }
2453     catch( e ) {}
2454
2455     try {
2456       elNodeType = el.nodeType;
2457     } catch (ex) {
2458       elNodeType = false;
2459       Util.log("html2canvas: failed to access some element's nodeType - Exception: " + ex.message);
2460     }
2461
2462     if (elNodeType === 1 || elNodeType === undefined) {
2463       loadPseudoElementImages(el);
2464       try {
2465         loadBackgroundImages(Util.getCSS(el, 'backgroundImage'), el);
2466       } catch(e) {
2467         Util.log("html2canvas: failed to get background-image - Exception: " + e.message);
2468       }
2469       loadBackgroundImages(el);
2470     }
2471   }
2472
2473   function setImageLoadHandlers(img, imageObj) {
2474     img.onload = function() {
2475       if ( imageObj.timer !== undefined ) {
2476         // CORS succeeded
2477         window.clearTimeout( imageObj.timer );
2478       }
2479
2480       images.numLoaded++;
2481       imageObj.succeeded = true;
2482       img.onerror = img.onload = null;
2483       start();
2484     };
2485     img.onerror = function() {
2486       if (img.crossOrigin === "anonymous") {
2487         // CORS failed
2488         window.clearTimeout( imageObj.timer );
2489
2490         // let's try with proxy instead
2491         if ( options.proxy ) {
2492           var src = img.src;
2493           img = new Image();
2494           imageObj.img = img;
2495           img.src = src;
2496
2497           proxyGetImage( img.src, img, imageObj );
2498           return;
2499         }
2500       }
2501
2502       images.numLoaded++;
2503       images.numFailed++;
2504       imageObj.succeeded = false;
2505       img.onerror = img.onload = null;
2506       start();
2507     };
2508   }
2509
2510   methods = {
2511     loadImage: function( src ) {
2512       var img, imageObj;
2513       if ( src && images[src] === undefined ) {
2514         img = new Image();
2515         if ( src.match(/data:image\/.*;base64,/i) ) {
2516           img.src = src.replace(/url\(['"]{0,}|['"]{0,}\)$/ig, '');
2517           imageObj = images[src] = {
2518             img: img
2519           };
2520           images.numTotal++;
2521           setImageLoadHandlers(img, imageObj);
2522         } else if ( isSameOrigin( src ) || options.allowTaint ===  true ) {
2523           imageObj = images[src] = {
2524             img: img
2525           };
2526           images.numTotal++;
2527           setImageLoadHandlers(img, imageObj);
2528           img.src = src;
2529         } else if ( supportCORS && !options.allowTaint && options.useCORS ) {
2530           // attempt to load with CORS
2531
2532           img.crossOrigin = "anonymous";
2533           imageObj = images[src] = {
2534             img: img
2535           };
2536           images.numTotal++;
2537           setImageLoadHandlers(img, imageObj);
2538           img.src = src;
2539         } else if ( options.proxy ) {
2540           imageObj = images[src] = {
2541             img: img
2542           };
2543           images.numTotal++;
2544           proxyGetImage( src, img, imageObj );
2545         }
2546       }
2547
2548     },
2549     cleanupDOM: function(cause) {
2550       var img, src;
2551       if (!images.cleanupDone) {
2552         if (cause && typeof cause === "string") {
2553           Util.log("html2canvas: Cleanup because: " + cause);
2554         } else {
2555           Util.log("html2canvas: Cleanup after timeout: " + options.timeout + " ms.");
2556         }
2557
2558         for (src in images) {
2559           if (images.hasOwnProperty(src)) {
2560             img = images[src];
2561             if (typeof img === "object" && img.callbackname && img.succeeded === undefined) {
2562               // cancel proxy image request
2563               window[img.callbackname] = undefined; // to work with IE<9  // NOTE: that the undefined callback property-name still exists on the window object (for IE<9)
2564               try {
2565                 delete window[img.callbackname];  // for all browser that support this
2566               } catch(ex) {}
2567               if (img.script && img.script.parentNode) {
2568                 img.script.setAttribute("src", "about:blank");  // try to cancel running request
2569                 img.script.parentNode.removeChild(img.script);
2570               }
2571               images.numLoaded++;
2572               images.numFailed++;
2573               Util.log("html2canvas: Cleaned up failed img: '" + src + "' Steps: " + images.numLoaded + " / " + images.numTotal);
2574             }
2575           }
2576         }
2577
2578         // cancel any pending requests
2579         if(window.stop !== undefined) {
2580           window.stop();
2581         } else if(document.execCommand !== undefined) {
2582           document.execCommand("Stop", false);
2583         }
2584         if (document.close !== undefined) {
2585           document.close();
2586         }
2587         images.cleanupDone = true;
2588         if (!(cause && typeof cause === "string")) {
2589           start();
2590         }
2591       }
2592     },
2593
2594     renderingDone: function() {
2595       if (timeoutTimer) {
2596         window.clearTimeout(timeoutTimer);
2597       }
2598     }
2599   };
2600
2601   if (options.timeout > 0) {
2602     timeoutTimer = window.setTimeout(methods.cleanupDOM, options.timeout);
2603   }
2604
2605   Util.log('html2canvas: Preload starts: finding background-images');
2606   images.firstRun = true;
2607
2608   getImages(element);
2609
2610   Util.log('html2canvas: Preload: Finding images');
2611   // load <img> images
2612   for (i = 0; i < imgLen; i+=1){
2613     methods.loadImage( domImages[i].getAttribute( "src" ) );
2614   }
2615
2616   images.firstRun = false;
2617   Util.log('html2canvas: Preload: Done.');
2618   if (images.numTotal === images.numLoaded) {
2619     start();
2620   }
2621
2622   return methods;
2623 };
2624
2625 _html2canvas.Renderer = function(parseQueue, options){
2626   function sortZindex(a, b) {
2627     if (a === 'children') {
2628       return -1;
2629     } else if (b === 'children') {
2630       return 1;
2631     } else {
2632       return a - b;
2633     }
2634   }
2635
2636   // http://www.w3.org/TR/CSS21/zindex.html
2637   function createRenderQueue(parseQueue) {
2638     var queue = [],
2639     rootContext;
2640
2641     rootContext = (function buildStackingContext(rootNode) {
2642       var rootContext = {};
2643       function insert(context, node, specialParent) {
2644         var zi = (node.zIndex.zindex === 'auto') ? 0 : Number(node.zIndex.zindex),
2645         contextForChildren = context, // the stacking context for children
2646         isPositioned = node.zIndex.isPositioned,
2647         isFloated = node.zIndex.isFloated,
2648         stub = {node: node},
2649         childrenDest = specialParent; // where children without z-index should be pushed into
2650
2651         if (node.zIndex.ownStacking) {
2652           contextForChildren = stub.context = {
2653               children: [{node:node, children: []}]
2654           };
2655           childrenDest = undefined;
2656         } else if (isPositioned || isFloated) {
2657           childrenDest = stub.children = [];
2658         }
2659
2660         if (zi === 0 && specialParent) {
2661           specialParent.push(stub);
2662         } else {
2663           if (!context[zi]) { context[zi] = []; }
2664           context[zi].push(stub);
2665         }
2666
2667         node.zIndex.children.forEach(function(childNode) {
2668           insert(contextForChildren, childNode, childrenDest);
2669         });
2670       }
2671       insert(rootContext, rootNode);
2672       return rootContext;
2673     })(parseQueue);
2674
2675     function sortZ(context) {
2676       Object.keys(context).sort(sortZindex).forEach(function(zi) {
2677         var nonPositioned = [],
2678         floated = [],
2679         positioned = [],
2680         list = [];
2681
2682         // positioned after static
2683         context[zi].forEach(function(v) {
2684           if (v.node.zIndex.isPositioned || v.node.zIndex.opacity < 1) {
2685             // http://www.w3.org/TR/css3-color/#transparency
2686             // non-positioned element with opactiy < 1 should be stacked as if it were a positioned element with â€˜z-index: 0’ and â€˜opacity: 1’.
2687             positioned.push(v);
2688           } else if (v.node.zIndex.isFloated) {
2689             floated.push(v);
2690           } else {
2691             nonPositioned.push(v);
2692           }
2693         });
2694
2695         (function walk(arr) {
2696           arr.forEach(function(v) {
2697             list.push(v);
2698             if (v.children) { walk(v.children); }
2699           });
2700         })(nonPositioned.concat(floated, positioned));
2701
2702         list.forEach(function(v) {
2703           if (v.context) {
2704             sortZ(v.context);
2705           } else {
2706             queue.push(v.node);
2707           }
2708         });
2709       });
2710     }
2711
2712     sortZ(rootContext);
2713
2714     return queue;
2715   }
2716
2717   function getRenderer(rendererName) {
2718     var renderer;
2719
2720     if (typeof options.renderer === "string" && _html2canvas.Renderer[rendererName] !== undefined) {
2721       renderer = _html2canvas.Renderer[rendererName](options);
2722     } else if (typeof rendererName === "function") {
2723       renderer = rendererName(options);
2724     } else {
2725       throw new Error("Unknown renderer");
2726     }
2727
2728     if ( typeof renderer !== "function" ) {
2729       throw new Error("Invalid renderer defined");
2730     }
2731     return renderer;
2732   }
2733
2734   return getRenderer(options.renderer)(parseQueue, options, document, createRenderQueue(parseQueue.stack), _html2canvas);
2735 };
2736
2737 _html2canvas.Util.Support = function (options, doc) {
2738
2739   function supportSVGRendering() {
2740     var img = new Image(),
2741     canvas = doc.createElement("canvas"),
2742     ctx = (canvas.getContext === undefined) ? false : canvas.getContext("2d");
2743     if (ctx === false) {
2744       return false;
2745     }
2746     canvas.width = canvas.height = 10;
2747     img.src = [
2748     "data:image/svg+xml,",
2749     "<svg xmlns='http://www.w3.org/2000/svg' width='10' height='10'>",
2750     "<foreignObject width='10' height='10'>",
2751     "<div xmlns='http://www.w3.org/1999/xhtml' style='width:10;height:10;'>",
2752     "sup",
2753     "</div>",
2754     "</foreignObject>",
2755     "</svg>"
2756     ].join("");
2757     try {
2758       ctx.drawImage(img, 0, 0);
2759       canvas.toDataURL();
2760     } catch(e) {
2761       return false;
2762     }
2763     _html2canvas.Util.log('html2canvas: Parse: SVG powered rendering available');
2764     return true;
2765   }
2766
2767   // Test whether we can use ranges to measure bounding boxes
2768   // Opera doesn't provide valid bounds.height/bottom even though it supports the method.
2769
2770   function supportRangeBounds() {
2771     var r, testElement, rangeBounds, rangeHeight, support = false;
2772
2773     if (doc.createRange) {
2774       r = doc.createRange();
2775       if (r.getBoundingClientRect) {
2776         testElement = doc.createElement('boundtest');
2777         testElement.style.height = "123px";
2778         testElement.style.display = "block";
2779         doc.body.appendChild(testElement);
2780
2781         r.selectNode(testElement);
2782         rangeBounds = r.getBoundingClientRect();
2783         rangeHeight = rangeBounds.height;
2784
2785         if (rangeHeight === 123) {
2786           support = true;
2787         }
2788         doc.body.removeChild(testElement);
2789       }
2790     }
2791
2792     return support;
2793   }
2794
2795   return {
2796     rangeBounds: supportRangeBounds(),
2797     svgRendering: options.svgRendering && supportSVGRendering()
2798   };
2799 };
2800 window.html2canvas = function(elements, opts) {
2801   elements = (elements.length) ? elements : [elements];
2802   var queue,
2803   canvas,
2804   options = {
2805     // general
2806     logging: false,
2807     elements: elements,
2808     background: "#fff",
2809
2810     // preload options
2811     proxy: null,
2812     timeout: 0,    // no timeout
2813     useCORS: false, // try to load images as CORS (where available), before falling back to proxy
2814     allowTaint: false, // whether to allow images to taint the canvas, won't need proxy if set to true
2815
2816     // parse options
2817     svgRendering: false, // use svg powered rendering where available (FF11+)
2818     ignoreElements: "IFRAME|OBJECT|PARAM",
2819     useOverflow: true,
2820     letterRendering: false,
2821     chinese: false,
2822     async: false, // If true, parsing will not block, but if the user scrolls during parse the image can get weird
2823
2824     // render options
2825     width: null,
2826     height: null,
2827     taintTest: true, // do a taint test with all images before applying to canvas
2828     renderer: "Canvas"
2829   };
2830
2831   options = _html2canvas.Util.Extend(opts, options);
2832
2833   _html2canvas.logging = options.logging;
2834   options.complete = function( images ) {
2835
2836     if (typeof options.onpreloaded === "function") {
2837       if ( options.onpreloaded( images ) === false ) {
2838         return;
2839       }
2840     }
2841     _html2canvas.Parse( images, options, function(queue) {
2842       if (typeof options.onparsed === "function") {
2843         if ( options.onparsed( queue ) === false ) {
2844           return;
2845         }
2846       }
2847
2848       canvas = _html2canvas.Renderer( queue, options );
2849
2850       if (typeof options.onrendered === "function") {
2851         options.onrendered( canvas );
2852       }
2853     });
2854   };
2855
2856   // for pages without images, we still want this to be async, i.e. return methods before executing
2857   window.setTimeout( function(){
2858     _html2canvas.Preload( options );
2859   }, 0 );
2860
2861   return {
2862     render: function( queue, opts ) {
2863       return _html2canvas.Renderer( queue, _html2canvas.Util.Extend(opts, options) );
2864     },
2865     parse: function( images, opts ) {
2866       return _html2canvas.Parse( images, _html2canvas.Util.Extend(opts, options) );
2867     },
2868     preload: function( opts ) {
2869       return _html2canvas.Preload( _html2canvas.Util.Extend(opts, options) );
2870     },
2871     log: _html2canvas.Util.log
2872   };
2873 };
2874
2875 window.html2canvas.log = _html2canvas.Util.log; // for renderers
2876 window.html2canvas.Renderer = {
2877   Canvas: undefined // We are assuming this will be used
2878 };
2879 _html2canvas.Renderer.Canvas = function(options) {
2880   options = options || {};
2881
2882   var doc = document,
2883   safeImages = [],
2884   testCanvas = document.createElement("canvas"),
2885   testctx = testCanvas.getContext("2d"),
2886   Util = _html2canvas.Util,
2887   canvas = options.canvas || doc.createElement('canvas');
2888
2889   function createShape(ctx, args) {
2890     ctx.beginPath();
2891     args.forEach(function(arg) {
2892       ctx[arg.name].apply(ctx, arg['arguments']);
2893     });
2894     ctx.closePath();
2895   }
2896
2897   function safeImage(item) {
2898     if (safeImages.indexOf(item['arguments'][0].src) === -1) {
2899       testctx.drawImage(item['arguments'][0], 0, 0);
2900       try {
2901         testctx.getImageData(0, 0, 1, 1);
2902       } catch(e) {
2903         testCanvas = doc.createElement("canvas");
2904         testctx = testCanvas.getContext("2d");
2905         return false;
2906       }
2907       safeImages.push(item['arguments'][0].src);
2908     }
2909     return true;
2910   }
2911
2912   function renderItem(ctx, item) {
2913     switch(item.type){
2914       case "variable":
2915         ctx[item.name] = item['arguments'];
2916         break;
2917       case "function":
2918         switch(item.name) {
2919           case "createPattern":
2920             if (item['arguments'][0].width > 0 && item['arguments'][0].height > 0) {
2921               try {
2922                 ctx.fillStyle = ctx.createPattern(item['arguments'][0], "repeat");
2923               } catch(e) {
2924                 Util.log("html2canvas: Renderer: Error creating pattern", e.message);
2925               }
2926             }
2927             break;
2928           case "drawShape":
2929             createShape(ctx, item['arguments']);
2930             break;
2931           case "drawImage":
2932             if (item['arguments'][8] > 0 && item['arguments'][7] > 0) {
2933               if (!options.taintTest || (options.taintTest && safeImage(item))) {
2934                 ctx.drawImage.apply( ctx, item['arguments'] );
2935               }
2936             }
2937             break;
2938           default:
2939             ctx[item.name].apply(ctx, item['arguments']);
2940         }
2941         break;
2942     }
2943   }
2944
2945   return function(parsedData, options, document, queue, _html2canvas) {
2946     var ctx = canvas.getContext("2d"),
2947     newCanvas,
2948     bounds,
2949     fstyle,
2950     zStack = parsedData.stack;
2951
2952     canvas.width = canvas.style.width =  options.width || zStack.ctx.width;
2953     canvas.height = canvas.style.height = options.height || zStack.ctx.height;
2954
2955     fstyle = ctx.fillStyle;
2956     ctx.fillStyle = (Util.isTransparent(parsedData.backgroundColor) && options.background !== undefined) ? options.background : parsedData.backgroundColor;
2957     ctx.fillRect(0, 0, canvas.width, canvas.height);
2958     ctx.fillStyle = fstyle;
2959     queue.forEach(function(storageContext) {
2960       // set common settings for canvas
2961       ctx.textBaseline = "bottom";
2962       ctx.save();
2963
2964       if (storageContext.transform.matrix) {
2965         ctx.translate(storageContext.transform.origin[0], storageContext.transform.origin[1]);
2966         ctx.transform.apply(ctx, storageContext.transform.matrix);
2967         ctx.translate(-storageContext.transform.origin[0], -storageContext.transform.origin[1]);
2968       }
2969
2970       if (storageContext.clip){
2971         ctx.beginPath();
2972         ctx.rect(storageContext.clip.left, storageContext.clip.top, storageContext.clip.width, storageContext.clip.height);
2973         ctx.clip();
2974       }
2975
2976       if (storageContext.ctx.storage) {
2977         storageContext.ctx.storage.forEach(function(item) {
2978           renderItem(ctx, item);
2979         });
2980       }
2981
2982       ctx.restore();
2983     });
2984
2985     Util.log("html2canvas: Renderer: Canvas renderer done - returning canvas obj");
2986
2987     if (options.elements.length === 1) {
2988       if (typeof options.elements[0] === "object" && options.elements[0].nodeName !== "BODY") {
2989         // crop image to the bounds of selected (single) element
2990         bounds = _html2canvas.Util.Bounds(options.elements[0]);
2991         newCanvas = document.createElement('canvas');
2992         
2993
2994             newCanvas.width = Math.ceil(bounds.width);
2995         newCanvas.height = Math.ceil(bounds.height);
2996    
2997           ctx = newCanvas.getContext("2d");
2998       ctx.drawImage(canvas, bounds.left, bounds.top, bounds.width, bounds.height, 0, 0, bounds.width, bounds.height);
2999
3000                 
3001                 
3002                 canvas = null;
3003         return newCanvas;
3004       }
3005     }
3006
3007     return canvas;
3008   };
3009 };
3010 })(window,document);