| /* | 
|   html2canvas 0.4.1 <http://html2canvas.hertzen.com> | 
|   Copyright (c) 2013 Niklas von Hertzen | 
|   | 
|   Released under MIT License | 
| */ | 
|   | 
| (function(window, document, undefined){ | 
|   | 
| "use strict"; | 
|   | 
| var _html2canvas = {}, | 
| previousElement, | 
| computedCSS, | 
| html2canvas; | 
|   | 
| _html2canvas.Util = {}; | 
|   | 
| _html2canvas.Util.log = function(a) { | 
|   if (_html2canvas.logging && window.console && window.console.log) { | 
|     window.console.log(a); | 
|   } | 
| }; | 
|   | 
| _html2canvas.Util.trimText = (function(isNative){ | 
|   return function(input) { | 
|     return isNative ? isNative.apply(input) : ((input || '') + '').replace( /^\s+|\s+$/g , '' ); | 
|   }; | 
| })(String.prototype.trim); | 
|   | 
| _html2canvas.Util.asFloat = function(v) { | 
|   return parseFloat(v); | 
| }; | 
|   | 
| (function() { | 
|   // TODO: support all possible length values | 
|   var TEXT_SHADOW_PROPERTY = /((rgba|rgb)\([^\)]+\)(\s-?\d+px){0,})/g; | 
|   var TEXT_SHADOW_VALUES = /(-?\d+px)|(#.+)|(rgb\(.+\))|(rgba\(.+\))/g; | 
|   _html2canvas.Util.parseTextShadows = function (value) { | 
|     if (!value || value === 'none') { | 
|       return []; | 
|     } | 
|   | 
|     // find multiple shadow declarations | 
|     var shadows = value.match(TEXT_SHADOW_PROPERTY), | 
|       results = []; | 
|     for (var i = 0; shadows && (i < shadows.length); i++) { | 
|       var s = shadows[i].match(TEXT_SHADOW_VALUES); | 
|       results.push({ | 
|         color: s[0], | 
|         offsetX: s[1] ? s[1].replace('px', '') : 0, | 
|         offsetY: s[2] ? s[2].replace('px', '') : 0, | 
|         blur: s[3] ? s[3].replace('px', '') : 0 | 
|       }); | 
|     } | 
|     return results; | 
|   }; | 
| })(); | 
|   | 
|   | 
| _html2canvas.Util.parseBackgroundImage = function (value) { | 
|     var whitespace = ' \r\n\t', | 
|         method, definition, prefix, prefix_i, block, results = [], | 
|         c, mode = 0, numParen = 0, quote, args; | 
|   | 
|     var appendResult = function(){ | 
|         if(method) { | 
|             if(definition.substr( 0, 1 ) === '"') { | 
|                 definition = definition.substr( 1, definition.length - 2 ); | 
|             } | 
|             if(definition) { | 
|                 args.push(definition); | 
|             } | 
|             if(method.substr( 0, 1 ) === '-' && | 
|                     (prefix_i = method.indexOf( '-', 1 ) + 1) > 0) { | 
|                 prefix = method.substr( 0, prefix_i); | 
|                 method = method.substr( prefix_i ); | 
|             } | 
|             results.push({ | 
|                 prefix: prefix, | 
|                 method: method.toLowerCase(), | 
|                 value: block, | 
|                 args: args | 
|             }); | 
|         } | 
|         args = []; //for some odd reason, setting .length = 0 didn't work in safari | 
|         method = | 
|             prefix = | 
|             definition = | 
|             block = ''; | 
|     }; | 
|   | 
|     appendResult(); | 
|     for(var i = 0, ii = value.length; i<ii; i++) { | 
|         c = value[i]; | 
|         if(mode === 0 && whitespace.indexOf( c ) > -1){ | 
|             continue; | 
|         } | 
|         switch(c) { | 
|             case '"': | 
|                 if(!quote) { | 
|                     quote = c; | 
|                 } | 
|                 else if(quote === c) { | 
|                     quote = null; | 
|                 } | 
|                 break; | 
|   | 
|             case '(': | 
|                 if(quote) { break; } | 
|                 else if(mode === 0) { | 
|                     mode = 1; | 
|                     block += c; | 
|                     continue; | 
|                 } else { | 
|                     numParen++; | 
|                 } | 
|                 break; | 
|   | 
|             case ')': | 
|                 if(quote) { break; } | 
|                 else if(mode === 1) { | 
|                     if(numParen === 0) { | 
|                         mode = 0; | 
|                         block += c; | 
|                         appendResult(); | 
|                         continue; | 
|                     } else { | 
|                         numParen--; | 
|                     } | 
|                 } | 
|                 break; | 
|   | 
|             case ',': | 
|                 if(quote) { break; } | 
|                 else if(mode === 0) { | 
|                     appendResult(); | 
|                     continue; | 
|                 } | 
|                 else if (mode === 1) { | 
|                     if(numParen === 0 && !method.match(/^url$/i)) { | 
|                         args.push(definition); | 
|                         definition = ''; | 
|                         block += c; | 
|                         continue; | 
|                     } | 
|                 } | 
|                 break; | 
|         } | 
|   | 
|         block += c; | 
|         if(mode === 0) { method += c; } | 
|         else { definition += c; } | 
|     } | 
|     appendResult(); | 
|   | 
|     return results; | 
| }; | 
|   | 
| _html2canvas.Util.Bounds = function (element) { | 
|   var clientRect, bounds = {}; | 
|   | 
|   if (element.getBoundingClientRect){ | 
|     clientRect = element.getBoundingClientRect(); | 
|   | 
|     // TODO add scroll position to bounds, so no scrolling of window necessary | 
|     bounds.top = clientRect.top; | 
|     bounds.bottom = clientRect.bottom || (clientRect.top + clientRect.height); | 
|     bounds.left = clientRect.left; | 
|   | 
|     bounds.width = element.offsetWidth; | 
|     bounds.height = element.offsetHeight; | 
|   } | 
|   | 
|   return bounds; | 
| }; | 
|   | 
| // TODO ideally, we'd want everything to go through this function instead of Util.Bounds, | 
| // but would require further work to calculate the correct positions for elements with offsetParents | 
| _html2canvas.Util.OffsetBounds = function (element) { | 
|   var parent = element.offsetParent ? _html2canvas.Util.OffsetBounds(element.offsetParent) : {top: 0, left: 0}; | 
|   | 
|   return { | 
|     top: element.offsetTop + parent.top, | 
|     bottom: element.offsetTop + element.offsetHeight + parent.top, | 
|     left: element.offsetLeft + parent.left, | 
|     width: element.offsetWidth, | 
|     height: element.offsetHeight | 
|   }; | 
| }; | 
|   | 
| function toPX(element, attribute, value ) { | 
|     var rsLeft = element.runtimeStyle && element.runtimeStyle[attribute], | 
|         left, | 
|         style = element.style; | 
|   | 
|     // Check if we are not dealing with pixels, (Opera has issues with this) | 
|     // Ported from jQuery css.js | 
|     // From the awesome hack by Dean Edwards | 
|     // http://erik.eae.net/archives/2007/07/27/18.54.15/#comment-102291 | 
|   | 
|     // If we're not dealing with a regular pixel number | 
|     // but a number that has a weird ending, we need to convert it to pixels | 
|   | 
|     if ( !/^-?[0-9]+\.?[0-9]*(?:px)?$/i.test( value ) && /^-?\d/.test(value) ) { | 
|         // Remember the original values | 
|         left = style.left; | 
|   | 
|         // Put in the new values to get a computed value out | 
|         if (rsLeft) { | 
|             element.runtimeStyle.left = element.currentStyle.left; | 
|         } | 
|         style.left = attribute === "fontSize" ? "1em" : (value || 0); | 
|         value = style.pixelLeft + "px"; | 
|   | 
|         // Revert the changed values | 
|         style.left = left; | 
|         if (rsLeft) { | 
|             element.runtimeStyle.left = rsLeft; | 
|         } | 
|     } | 
|   | 
|     if (!/^(thin|medium|thick)$/i.test(value)) { | 
|         return Math.round(parseFloat(value)) + "px"; | 
|     } | 
|   | 
|     return value; | 
| } | 
|   | 
| function asInt(val) { | 
|     return parseInt(val, 10); | 
| } | 
|   | 
| function parseBackgroundSizePosition(value, element, attribute, index) { | 
|     value = (value || '').split(','); | 
|     value = value[index || 0] || value[0] || 'auto'; | 
|     value = _html2canvas.Util.trimText(value).split(' '); | 
|   | 
|     if(attribute === 'backgroundSize' && (!value[0] || value[0].match(/cover|contain|auto/))) { | 
|         //these values will be handled in the parent function | 
|     } else { | 
|         value[0] = (value[0].indexOf( "%" ) === -1) ? toPX(element, attribute + "X", value[0]) : value[0]; | 
|         if(value[1] === undefined) { | 
|             if(attribute === 'backgroundSize') { | 
|                 value[1] = 'auto'; | 
|                 return value; | 
|             } else { | 
|                 // IE 9 doesn't return double digit always | 
|                 value[1] = value[0]; | 
|             } | 
|         } | 
|         value[1] = (value[1].indexOf("%") === -1) ? toPX(element, attribute + "Y", value[1]) : value[1]; | 
|     } | 
|     return value; | 
| } | 
|   | 
| _html2canvas.Util.getCSS = function (element, attribute, index) { | 
|     if (previousElement !== element) { | 
|       computedCSS = document.defaultView.getComputedStyle(element, null); | 
|     } | 
|   | 
|     var value = computedCSS[attribute]; | 
|   | 
|     if (/^background(Size|Position)$/.test(attribute)) { | 
|         return parseBackgroundSizePosition(value, element, attribute, index); | 
|     } else if (/border(Top|Bottom)(Left|Right)Radius/.test(attribute)) { | 
|       var arr = value.split(" "); | 
|       if (arr.length <= 1) { | 
|           arr[1] = arr[0]; | 
|       } | 
|       return arr.map(asInt); | 
|     } | 
|   | 
|   return value; | 
| }; | 
|   | 
| _html2canvas.Util.resizeBounds = function( current_width, current_height, target_width, target_height, stretch_mode ){ | 
|   var target_ratio = target_width / target_height, | 
|     current_ratio = current_width / current_height, | 
|     output_width, output_height; | 
|   | 
|   if(!stretch_mode || stretch_mode === 'auto') { | 
|     output_width = target_width; | 
|     output_height = target_height; | 
|   } else if(target_ratio < current_ratio ^ stretch_mode === 'contain') { | 
|     output_height = target_height; | 
|     output_width = target_height * current_ratio; | 
|   } else { | 
|     output_width = target_width; | 
|     output_height = target_width / current_ratio; | 
|   } | 
|   | 
|   return { | 
|     width: output_width, | 
|     height: output_height | 
|   }; | 
| }; | 
|   | 
| function backgroundBoundsFactory( prop, el, bounds, image, imageIndex, backgroundSize ) { | 
|     var bgposition =  _html2canvas.Util.getCSS( el, prop, imageIndex ) , | 
|     topPos, | 
|     left, | 
|     percentage, | 
|     val; | 
|   | 
|     if (bgposition.length === 1){ | 
|       val = bgposition[0]; | 
|   | 
|       bgposition = []; | 
|   | 
|       bgposition[0] = val; | 
|       bgposition[1] = val; | 
|     } | 
|   | 
|     if (bgposition[0].toString().indexOf("%") !== -1){ | 
|       percentage = (parseFloat(bgposition[0])/100); | 
|       left = bounds.width * percentage; | 
|       if(prop !== 'backgroundSize') { | 
|         left -= (backgroundSize || image).width*percentage; | 
|       } | 
|     } else { | 
|       if(prop === 'backgroundSize') { | 
|         if(bgposition[0] === 'auto') { | 
|           left = image.width; | 
|         } else { | 
|           if (/contain|cover/.test(bgposition[0])) { | 
|             var resized = _html2canvas.Util.resizeBounds(image.width, image.height, bounds.width, bounds.height, bgposition[0]); | 
|             left = resized.width; | 
|             topPos = resized.height; | 
|           } else { | 
|             left = parseInt(bgposition[0], 10); | 
|           } | 
|         } | 
|       } else { | 
|         left = parseInt( bgposition[0], 10); | 
|       } | 
|     } | 
|   | 
|   | 
|     if(bgposition[1] === 'auto') { | 
|       topPos = left / image.width * image.height; | 
|     } else if (bgposition[1].toString().indexOf("%") !== -1){ | 
|       percentage = (parseFloat(bgposition[1])/100); | 
|       topPos =  bounds.height * percentage; | 
|       if(prop !== 'backgroundSize') { | 
|         topPos -= (backgroundSize || image).height * percentage; | 
|       } | 
|   | 
|     } else { | 
|       topPos = parseInt(bgposition[1],10); | 
|     } | 
|   | 
|     return [left, topPos]; | 
| } | 
|   | 
| _html2canvas.Util.BackgroundPosition = function( el, bounds, image, imageIndex, backgroundSize ) { | 
|     var result = backgroundBoundsFactory( 'backgroundPosition', el, bounds, image, imageIndex, backgroundSize ); | 
|     return { left: result[0], top: result[1] }; | 
| }; | 
|   | 
| _html2canvas.Util.BackgroundSize = function( el, bounds, image, imageIndex ) { | 
|     var result = backgroundBoundsFactory( 'backgroundSize', el, bounds, image, imageIndex ); | 
|     return { width: result[0], height: result[1] }; | 
| }; | 
|   | 
| _html2canvas.Util.Extend = function (options, defaults) { | 
|   for (var key in options) { | 
|     if (options.hasOwnProperty(key)) { | 
|       defaults[key] = options[key]; | 
|     } | 
|   } | 
|   return defaults; | 
| }; | 
|   | 
|   | 
| /* | 
|  * Derived from jQuery.contents() | 
|  * Copyright 2010, John Resig | 
|  * Dual licensed under the MIT or GPL Version 2 licenses. | 
|  * http://jquery.org/license | 
|  */ | 
| _html2canvas.Util.Children = function( elem ) { | 
|   var children; | 
|   try { | 
|     children = (elem.nodeName && elem.nodeName.toUpperCase() === "IFRAME") ? elem.contentDocument || elem.contentWindow.document : (function(array) { | 
|       var ret = []; | 
|       if (array !== null) { | 
|         (function(first, second ) { | 
|           var i = first.length, | 
|           j = 0; | 
|   | 
|           if (typeof second.length === "number") { | 
|             for (var l = second.length; j < l; j++) { | 
|               first[i++] = second[j]; | 
|             } | 
|           } else { | 
|             while (second[j] !== undefined) { | 
|               first[i++] = second[j++]; | 
|             } | 
|           } | 
|   | 
|           first.length = i; | 
|   | 
|           return first; | 
|         })(ret, array); | 
|       } | 
|       return ret; | 
|     })(elem.childNodes); | 
|   | 
|   } catch (ex) { | 
|     _html2canvas.Util.log("html2canvas.Util.Children failed with exception: " + ex.message); | 
|     children = []; | 
|   } | 
|   return children; | 
| }; | 
|   | 
| _html2canvas.Util.isTransparent = function(backgroundColor) { | 
|   return (backgroundColor === "transparent" || backgroundColor === "rgba(0, 0, 0, 0)"); | 
| }; | 
| _html2canvas.Util.Font = (function () { | 
|   | 
|   var fontData = {}; | 
|   | 
|   return function(font, fontSize, doc) { | 
|     if (fontData[font + "-" + fontSize] !== undefined) { | 
|       return fontData[font + "-" + fontSize]; | 
|     } | 
|   | 
|     var container = doc.createElement('div'), | 
|     img = doc.createElement('img'), | 
|     span = doc.createElement('span'), | 
|     sampleText = 'Hidden Text', | 
|     baseline, | 
|     middle, | 
|     metricsObj; | 
|   | 
|     container.style.visibility = "hidden"; | 
|     container.style.fontFamily = font; | 
|     container.style.fontSize = fontSize; | 
|     container.style.margin = 0; | 
|     container.style.padding = 0; | 
|   | 
|     doc.body.appendChild(container); | 
|   | 
|     // http://probablyprogramming.com/2009/03/15/the-tiniest-gif-ever (handtinywhite.gif) | 
|     img.src = ""; | 
|     img.width = 1; | 
|     img.height = 1; | 
|   | 
|     img.style.margin = 0; | 
|     img.style.padding = 0; | 
|     img.style.verticalAlign = "baseline"; | 
|   | 
|     span.style.fontFamily = font; | 
|     span.style.fontSize = fontSize; | 
|     span.style.margin = 0; | 
|     span.style.padding = 0; | 
|   | 
|     span.appendChild(doc.createTextNode(sampleText)); | 
|     container.appendChild(span); | 
|     container.appendChild(img); | 
|     baseline = (img.offsetTop - span.offsetTop) + 1; | 
|   | 
|     container.removeChild(span); | 
|     container.appendChild(doc.createTextNode(sampleText)); | 
|   | 
|     container.style.lineHeight = "normal"; | 
|     img.style.verticalAlign = "super"; | 
|   | 
|     middle = (img.offsetTop-container.offsetTop) + 1; | 
|     metricsObj = { | 
|       baseline: baseline, | 
|       lineWidth: 1, | 
|       middle: middle | 
|     }; | 
|   | 
|     fontData[font + "-" + fontSize] = metricsObj; | 
|   | 
|     doc.body.removeChild(container); | 
|   | 
|     return metricsObj; | 
|   }; | 
| })(); | 
|   | 
| (function(){ | 
|   var Util = _html2canvas.Util, | 
|     Generate = {}; | 
|   | 
|   _html2canvas.Generate = Generate; | 
|   | 
|   var reGradients = [ | 
|   /^(-webkit-linear-gradient)\(([a-z\s]+)([\w\d\.\s,%\(\)]+)\)$/, | 
|   /^(-o-linear-gradient)\(([a-z\s]+)([\w\d\.\s,%\(\)]+)\)$/, | 
|   /^(-webkit-gradient)\((linear|radial),\s((?:\d{1,3}%?)\s(?:\d{1,3}%?),\s(?:\d{1,3}%?)\s(?:\d{1,3}%?))([\w\d\.\s,%\(\)\-]+)\)$/, | 
|   /^(-moz-linear-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?))([\w\d\.\s,%\(\)]+)\)$/, | 
|   /^(-webkit-radial-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?)),\s(\w+)\s([a-z\-]+)([\w\d\.\s,%\(\)]+)\)$/, | 
|   /^(-moz-radial-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?)),\s(\w+)\s?([a-z\-]*)([\w\d\.\s,%\(\)]+)\)$/, | 
|   /^(-o-radial-gradient)\(((?:\d{1,3}%?)\s(?:\d{1,3}%?)),\s(\w+)\s([a-z\-]+)([\w\d\.\s,%\(\)]+)\)$/ | 
|   ]; | 
|   | 
|   /* | 
|  * TODO: Add IE10 vendor prefix (-ms) support | 
|  * TODO: Add W3C gradient (linear-gradient) support | 
|  * TODO: Add old Webkit -webkit-gradient(radial, ...) support | 
|  * TODO: Maybe some RegExp optimizations are possible ;o) | 
|  */ | 
|   Generate.parseGradient = function(css, bounds) { | 
|     var gradient, i, len = reGradients.length, m1, stop, m2, m2Len, step, m3, tl,tr,br,bl; | 
|   | 
|     for(i = 0; i < len; i+=1){ | 
|       m1 = css.match(reGradients[i]); | 
|       if(m1) { | 
|         break; | 
|       } | 
|     } | 
|   | 
|     if(m1) { | 
|       switch(m1[1]) { | 
|         case '-webkit-linear-gradient': | 
|         case '-o-linear-gradient': | 
|   | 
|           gradient = { | 
|             type: 'linear', | 
|             x0: null, | 
|             y0: null, | 
|             x1: null, | 
|             y1: null, | 
|             colorStops: [] | 
|           }; | 
|   | 
|           // get coordinates | 
|           m2 = m1[2].match(/\w+/g); | 
|           if(m2){ | 
|             m2Len = m2.length; | 
|             for(i = 0; i < m2Len; i+=1){ | 
|               switch(m2[i]) { | 
|                 case 'top': | 
|                   gradient.y0 = 0; | 
|                   gradient.y1 = bounds.height; | 
|                   break; | 
|   | 
|                 case 'right': | 
|                   gradient.x0 = bounds.width; | 
|                   gradient.x1 = 0; | 
|                   break; | 
|   | 
|                 case 'bottom': | 
|                   gradient.y0 = bounds.height; | 
|                   gradient.y1 = 0; | 
|                   break; | 
|   | 
|                 case 'left': | 
|                   gradient.x0 = 0; | 
|                   gradient.x1 = bounds.width; | 
|                   break; | 
|               } | 
|             } | 
|           } | 
|           if(gradient.x0 === null && gradient.x1 === null){ // center | 
|             gradient.x0 = gradient.x1 = bounds.width / 2; | 
|           } | 
|           if(gradient.y0 === null && gradient.y1 === null){ // center | 
|             gradient.y0 = gradient.y1 = bounds.height / 2; | 
|           } | 
|   | 
|           // get colors and stops | 
|           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); | 
|           if(m2){ | 
|             m2Len = m2.length; | 
|             step = 1 / Math.max(m2Len - 1, 1); | 
|             for(i = 0; i < m2Len; i+=1){ | 
|               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)?/); | 
|               if(m3[2]){ | 
|                 stop = parseFloat(m3[2]); | 
|                 if(m3[3] === '%'){ | 
|                   stop /= 100; | 
|                 } else { // px - stupid opera | 
|                   stop /= bounds.width; | 
|                 } | 
|               } else { | 
|                 stop = i * step; | 
|               } | 
|               gradient.colorStops.push({ | 
|                 color: m3[1], | 
|                 stop: stop | 
|               }); | 
|             } | 
|           } | 
|           break; | 
|   | 
|         case '-webkit-gradient': | 
|   | 
|           gradient = { | 
|             type: m1[2] === 'radial' ? 'circle' : m1[2], // TODO: Add radial gradient support for older mozilla definitions | 
|             x0: 0, | 
|             y0: 0, | 
|             x1: 0, | 
|             y1: 0, | 
|             colorStops: [] | 
|           }; | 
|   | 
|           // get coordinates | 
|           m2 = m1[3].match(/(\d{1,3})%?\s(\d{1,3})%?,\s(\d{1,3})%?\s(\d{1,3})%?/); | 
|           if(m2){ | 
|             gradient.x0 = (m2[1] * bounds.width) / 100; | 
|             gradient.y0 = (m2[2] * bounds.height) / 100; | 
|             gradient.x1 = (m2[3] * bounds.width) / 100; | 
|             gradient.y1 = (m2[4] * bounds.height) / 100; | 
|           } | 
|   | 
|           // get colors and stops | 
|           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); | 
|           if(m2){ | 
|             m2Len = m2.length; | 
|             for(i = 0; i < m2Len; i+=1){ | 
|               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\.]+)?\))\)/); | 
|               stop = parseFloat(m3[2]); | 
|               if(m3[1] === 'from') { | 
|                 stop = 0.0; | 
|               } | 
|               if(m3[1] === 'to') { | 
|                 stop = 1.0; | 
|               } | 
|               gradient.colorStops.push({ | 
|                 color: m3[3], | 
|                 stop: stop | 
|               }); | 
|             } | 
|           } | 
|           break; | 
|   | 
|         case '-moz-linear-gradient': | 
|   | 
|           gradient = { | 
|             type: 'linear', | 
|             x0: 0, | 
|             y0: 0, | 
|             x1: 0, | 
|             y1: 0, | 
|             colorStops: [] | 
|           }; | 
|   | 
|           // get coordinates | 
|           m2 = m1[2].match(/(\d{1,3})%?\s(\d{1,3})%?/); | 
|   | 
|           // m2[1] == 0%   -> left | 
|           // m2[1] == 50%  -> center | 
|           // m2[1] == 100% -> right | 
|   | 
|           // m2[2] == 0%   -> top | 
|           // m2[2] == 50%  -> center | 
|           // m2[2] == 100% -> bottom | 
|   | 
|           if(m2){ | 
|             gradient.x0 = (m2[1] * bounds.width) / 100; | 
|             gradient.y0 = (m2[2] * bounds.height) / 100; | 
|             gradient.x1 = bounds.width - gradient.x0; | 
|             gradient.y1 = bounds.height - gradient.y0; | 
|           } | 
|   | 
|           // get colors and stops | 
|           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); | 
|           if(m2){ | 
|             m2Len = m2.length; | 
|             step = 1 / Math.max(m2Len - 1, 1); | 
|             for(i = 0; i < m2Len; i+=1){ | 
|               m3 = m2[i].match(/((?:rgb|rgba)\(\d{1,3},\s\d{1,3},\s\d{1,3}(?:,\s[0-9\.]+)?\))\s*(\d{1,3})?(%)?/); | 
|               if(m3[2]){ | 
|                 stop = parseFloat(m3[2]); | 
|                 if(m3[3]){ // percentage | 
|                   stop /= 100; | 
|                 } | 
|               } else { | 
|                 stop = i * step; | 
|               } | 
|               gradient.colorStops.push({ | 
|                 color: m3[1], | 
|                 stop: stop | 
|               }); | 
|             } | 
|           } | 
|           break; | 
|   | 
|         case '-webkit-radial-gradient': | 
|         case '-moz-radial-gradient': | 
|         case '-o-radial-gradient': | 
|   | 
|           gradient = { | 
|             type: 'circle', | 
|             x0: 0, | 
|             y0: 0, | 
|             x1: bounds.width, | 
|             y1: bounds.height, | 
|             cx: 0, | 
|             cy: 0, | 
|             rx: 0, | 
|             ry: 0, | 
|             colorStops: [] | 
|           }; | 
|   | 
|           // center | 
|           m2 = m1[2].match(/(\d{1,3})%?\s(\d{1,3})%?/); | 
|           if(m2){ | 
|             gradient.cx = (m2[1] * bounds.width) / 100; | 
|             gradient.cy = (m2[2] * bounds.height) / 100; | 
|           } | 
|   | 
|           // size | 
|           m2 = m1[3].match(/\w+/); | 
|           m3 = m1[4].match(/[a-z\-]*/); | 
|           if(m2 && m3){ | 
|             switch(m3[0]){ | 
|               case 'farthest-corner': | 
|               case 'cover': // is equivalent to farthest-corner | 
|               case '': // mozilla removes "cover" from definition :( | 
|                 tl = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.cy, 2)); | 
|                 tr = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2)); | 
|                 br = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2)); | 
|                 bl = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.cy, 2)); | 
|                 gradient.rx = gradient.ry = Math.max(tl, tr, br, bl); | 
|                 break; | 
|               case 'closest-corner': | 
|                 tl = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.cy, 2)); | 
|                 tr = Math.sqrt(Math.pow(gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2)); | 
|                 br = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.y1 - gradient.cy, 2)); | 
|                 bl = Math.sqrt(Math.pow(gradient.x1 - gradient.cx, 2) + Math.pow(gradient.cy, 2)); | 
|                 gradient.rx = gradient.ry = Math.min(tl, tr, br, bl); | 
|                 break; | 
|               case 'farthest-side': | 
|                 if(m2[0] === 'circle'){ | 
|                   gradient.rx = gradient.ry = Math.max( | 
|                     gradient.cx, | 
|                     gradient.cy, | 
|                     gradient.x1 - gradient.cx, | 
|                     gradient.y1 - gradient.cy | 
|                     ); | 
|                 } else { // ellipse | 
|   | 
|                   gradient.type = m2[0]; | 
|   | 
|                   gradient.rx = Math.max( | 
|                     gradient.cx, | 
|                     gradient.x1 - gradient.cx | 
|                     ); | 
|                   gradient.ry = Math.max( | 
|                     gradient.cy, | 
|                     gradient.y1 - gradient.cy | 
|                     ); | 
|                 } | 
|                 break; | 
|               case 'closest-side': | 
|               case 'contain': // is equivalent to closest-side | 
|                 if(m2[0] === 'circle'){ | 
|                   gradient.rx = gradient.ry = Math.min( | 
|                     gradient.cx, | 
|                     gradient.cy, | 
|                     gradient.x1 - gradient.cx, | 
|                     gradient.y1 - gradient.cy | 
|                     ); | 
|                 } else { // ellipse | 
|   | 
|                   gradient.type = m2[0]; | 
|   | 
|                   gradient.rx = Math.min( | 
|                     gradient.cx, | 
|                     gradient.x1 - gradient.cx | 
|                     ); | 
|                   gradient.ry = Math.min( | 
|                     gradient.cy, | 
|                     gradient.y1 - gradient.cy | 
|                     ); | 
|                 } | 
|                 break; | 
|   | 
|             // TODO: add support for "30px 40px" sizes (webkit only) | 
|             } | 
|           } | 
|   | 
|           // color stops | 
|           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); | 
|           if(m2){ | 
|             m2Len = m2.length; | 
|             step = 1 / Math.max(m2Len - 1, 1); | 
|             for(i = 0; i < m2Len; i+=1){ | 
|               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)?/); | 
|               if(m3[2]){ | 
|                 stop = parseFloat(m3[2]); | 
|                 if(m3[3] === '%'){ | 
|                   stop /= 100; | 
|                 } else { // px - stupid opera | 
|                   stop /= bounds.width; | 
|                 } | 
|               } else { | 
|                 stop = i * step; | 
|               } | 
|               gradient.colorStops.push({ | 
|                 color: m3[1], | 
|                 stop: stop | 
|               }); | 
|             } | 
|           } | 
|           break; | 
|       } | 
|     } | 
|   | 
|     return gradient; | 
|   }; | 
|   | 
|   function addScrollStops(grad) { | 
|     return function(colorStop) { | 
|       try { | 
|         grad.addColorStop(colorStop.stop, colorStop.color); | 
|       } | 
|       catch(e) { | 
|         Util.log(['failed to add color stop: ', e, '; tried to add: ', colorStop]); | 
|       } | 
|     }; | 
|   } | 
|   | 
|   Generate.Gradient = function(src, bounds) { | 
|     if(bounds.width === 0 || bounds.height === 0) { | 
|       return; | 
|     } | 
|   | 
|     var canvas = document.createElement('canvas'), | 
|     ctx = canvas.getContext('2d'), | 
|     gradient, grad; | 
|   | 
|     canvas.width = bounds.width; | 
|     canvas.height = bounds.height; | 
|   | 
|     // TODO: add support for multi defined background gradients | 
|     gradient = _html2canvas.Generate.parseGradient(src, bounds); | 
|   | 
|     if(gradient) { | 
|       switch(gradient.type) { | 
|         case 'linear': | 
|           grad = ctx.createLinearGradient(gradient.x0, gradient.y0, gradient.x1, gradient.y1); | 
|           gradient.colorStops.forEach(addScrollStops(grad)); | 
|           ctx.fillStyle = grad; | 
|           ctx.fillRect(0, 0, bounds.width, bounds.height); | 
|           break; | 
|   | 
|         case 'circle': | 
|           grad = ctx.createRadialGradient(gradient.cx, gradient.cy, 0, gradient.cx, gradient.cy, gradient.rx); | 
|           gradient.colorStops.forEach(addScrollStops(grad)); | 
|           ctx.fillStyle = grad; | 
|           ctx.fillRect(0, 0, bounds.width, bounds.height); | 
|           break; | 
|   | 
|         case 'ellipse': | 
|           var canvasRadial = document.createElement('canvas'), | 
|             ctxRadial = canvasRadial.getContext('2d'), | 
|             ri = Math.max(gradient.rx, gradient.ry), | 
|             di = ri * 2; | 
|   | 
|           canvasRadial.width = canvasRadial.height = di; | 
|   | 
|           grad = ctxRadial.createRadialGradient(gradient.rx, gradient.ry, 0, gradient.rx, gradient.ry, ri); | 
|           gradient.colorStops.forEach(addScrollStops(grad)); | 
|   | 
|           ctxRadial.fillStyle = grad; | 
|           ctxRadial.fillRect(0, 0, di, di); | 
|   | 
|           ctx.fillStyle = gradient.colorStops[gradient.colorStops.length - 1].color; | 
|           ctx.fillRect(0, 0, canvas.width, canvas.height); | 
|           ctx.drawImage(canvasRadial, gradient.cx - gradient.rx, gradient.cy - gradient.ry, 2 * gradient.rx, 2 * gradient.ry); | 
|           break; | 
|       } | 
|     } | 
|   | 
|     return canvas; | 
|   }; | 
|   | 
|   Generate.ListAlpha = function(number) { | 
|     var tmp = "", | 
|     modulus; | 
|   | 
|     do { | 
|       modulus = number % 26; | 
|       tmp = String.fromCharCode((modulus) + 64) + tmp; | 
|       number = number / 26; | 
|     }while((number*26) > 26); | 
|   | 
|     return tmp; | 
|   }; | 
|   | 
|   Generate.ListRoman = function(number) { | 
|     var romanArray = ["M", "CM", "D", "CD", "C", "XC", "L", "XL", "X", "IX", "V", "IV", "I"], | 
|     decimal = [1000, 900, 500, 400, 100, 90, 50, 40, 10, 9, 5, 4, 1], | 
|     roman = "", | 
|     v, | 
|     len = romanArray.length; | 
|   | 
|     if (number <= 0 || number >= 4000) { | 
|       return number; | 
|     } | 
|   | 
|     for (v=0; v < len; v+=1) { | 
|       while (number >= decimal[v]) { | 
|         number -= decimal[v]; | 
|         roman += romanArray[v]; | 
|       } | 
|     } | 
|   | 
|     return roman; | 
|   }; | 
| })(); | 
| function h2cRenderContext(width, height) { | 
|   var storage = []; | 
|   return { | 
|     storage: storage, | 
|     width: width, | 
|     height: height, | 
|     clip: function() { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "clip", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     translate: function() { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "translate", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     fill: function() { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "fill", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     save: function() { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "save", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     restore: function() { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "restore", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     fillRect: function () { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "fillRect", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     createPattern: function() { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "createPattern", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     drawShape: function() { | 
|   | 
|       var shape = []; | 
|   | 
|       storage.push({ | 
|         type: "function", | 
|         name: "drawShape", | 
|         'arguments': shape | 
|       }); | 
|   | 
|       return { | 
|         moveTo: function() { | 
|           shape.push({ | 
|             name: "moveTo", | 
|             'arguments': arguments | 
|           }); | 
|         }, | 
|         lineTo: function() { | 
|           shape.push({ | 
|             name: "lineTo", | 
|             'arguments': arguments | 
|           }); | 
|         }, | 
|         arcTo: function() { | 
|           shape.push({ | 
|             name: "arcTo", | 
|             'arguments': arguments | 
|           }); | 
|         }, | 
|         bezierCurveTo: function() { | 
|           shape.push({ | 
|             name: "bezierCurveTo", | 
|             'arguments': arguments | 
|           }); | 
|         }, | 
|         quadraticCurveTo: function() { | 
|           shape.push({ | 
|             name: "quadraticCurveTo", | 
|             'arguments': arguments | 
|           }); | 
|         } | 
|       }; | 
|   | 
|     }, | 
|     drawImage: function () { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "drawImage", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     fillText: function () { | 
|       storage.push({ | 
|         type: "function", | 
|         name: "fillText", | 
|         'arguments': arguments | 
|       }); | 
|     }, | 
|     setVariable: function (variable, value) { | 
|       storage.push({ | 
|         type: "variable", | 
|         name: variable, | 
|         'arguments': value | 
|       }); | 
|       return value; | 
|     } | 
|   }; | 
| } | 
| _html2canvas.Parse = function (images, options) { | 
|   //window.scroll(0,0); | 
|   | 
|   var element = (( options.elements === undefined ) ? document.body : options.elements[0]), // select body by default | 
|   numDraws = 0, | 
|   doc = element.ownerDocument, | 
|   Util = _html2canvas.Util, | 
|   support = Util.Support(options, doc), | 
|   ignoreElementsRegExp = new RegExp("(" + options.ignoreElements + ")"), | 
|   body = doc.body, | 
|   getCSS = Util.getCSS, | 
|   pseudoHide = "___html2canvas___pseudoelement", | 
|   hidePseudoElements = doc.createElement('style'); | 
|   | 
|   hidePseudoElements.innerHTML = '.' + pseudoHide + '-before:before { content: "" !important; display: none !important; }' + | 
|   '.' + pseudoHide + '-after:after { content: "" !important; display: none !important; }'; | 
|   | 
|   body.appendChild(hidePseudoElements); | 
|   | 
|   images = images || {}; | 
|   | 
|   function documentWidth () { | 
|     return Math.max( | 
|       Math.max(doc.body.scrollWidth, doc.documentElement.scrollWidth), | 
|       Math.max(doc.body.offsetWidth, doc.documentElement.offsetWidth), | 
|       Math.max(doc.body.clientWidth, doc.documentElement.clientWidth) | 
|       ); | 
|   } | 
|   | 
|   function documentHeight () { | 
|     return Math.max( | 
|       Math.max(doc.body.scrollHeight, doc.documentElement.scrollHeight), | 
|       Math.max(doc.body.offsetHeight, doc.documentElement.offsetHeight), | 
|       Math.max(doc.body.clientHeight, doc.documentElement.clientHeight) | 
|       ); | 
|   } | 
|   | 
|   function getCSSInt(element, attribute) { | 
|     var val = parseInt(getCSS(element, attribute), 10); | 
|     return (isNaN(val)) ? 0 : val; // borders in old IE are throwing 'medium' for demo.html | 
|   } | 
|   | 
|   function renderRect (ctx, x, y, w, h, bgcolor) { | 
|     if (bgcolor !== "transparent"){ | 
|       ctx.setVariable("fillStyle", bgcolor); | 
|       ctx.fillRect(x, y, w, h); | 
|       numDraws+=1; | 
|     } | 
|   } | 
|   | 
|   function capitalize(m, p1, p2) { | 
|     if (m.length > 0) { | 
|       return p1 + p2.toUpperCase(); | 
|     } | 
|   } | 
|   | 
|   function textTransform (text, transform) { | 
|     switch(transform){ | 
|       case "lowercase": | 
|         return text.toLowerCase(); | 
|       case "capitalize": | 
|         return text.replace( /(^|\s|:|-|\(|\))([a-z])/g, capitalize); | 
|       case "uppercase": | 
|         return text.toUpperCase(); | 
|       default: | 
|         return text; | 
|     } | 
|   } | 
|   | 
|   function noLetterSpacing(letter_spacing) { | 
|     return (/^(normal|none|0px)$/.test(letter_spacing)); | 
|   } | 
|   | 
|   function drawText(currentText, x, y, ctx){ | 
|     if (currentText !== null && Util.trimText(currentText).length > 0) { | 
|       ctx.fillText(currentText, x, y); | 
|       numDraws+=1; | 
|     } | 
|   } | 
|   | 
|   function setTextVariables(ctx, el, text_decoration, color) { | 
|     var align = false, | 
|     bold = getCSS(el, "fontWeight"), | 
|     family = getCSS(el, "fontFamily"), | 
|     size = getCSS(el, "fontSize"), | 
|     shadows = Util.parseTextShadows(getCSS(el, "textShadow")); | 
|   | 
|     switch(parseInt(bold, 10)){ | 
|       case 401: | 
|         bold = "bold"; | 
|         break; | 
|       case 400: | 
|         bold = "normal"; | 
|         break; | 
|     } | 
|   | 
|     ctx.setVariable("fillStyle", color); | 
|     ctx.setVariable("font", [getCSS(el, "fontStyle"), getCSS(el, "fontVariant"), bold, size, family].join(" ")); | 
|     ctx.setVariable("textAlign", (align) ? "right" : "left"); | 
|   | 
|     if (shadows.length) { | 
|       // TODO: support multiple text shadows | 
|       // apply the first text shadow | 
|       ctx.setVariable("shadowColor", shadows[0].color); | 
|       ctx.setVariable("shadowOffsetX", shadows[0].offsetX); | 
|       ctx.setVariable("shadowOffsetY", shadows[0].offsetY); | 
|       ctx.setVariable("shadowBlur", shadows[0].blur); | 
|     } | 
|   | 
|     if (text_decoration !== "none"){ | 
|       return Util.Font(family, size, doc); | 
|     } | 
|   } | 
|   | 
|   function renderTextDecoration(ctx, text_decoration, bounds, metrics, color) { | 
|     switch(text_decoration) { | 
|       case "underline": | 
|         // Draws a line at the baseline of the font | 
|         // 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 | 
|         renderRect(ctx, bounds.left, Math.round(bounds.top + metrics.baseline + metrics.lineWidth), bounds.width, 1, color); | 
|         break; | 
|       case "overline": | 
|         renderRect(ctx, bounds.left, Math.round(bounds.top), bounds.width, 1, color); | 
|         break; | 
|       case "line-through": | 
|         // TODO try and find exact position for line-through | 
|         renderRect(ctx, bounds.left, Math.ceil(bounds.top + metrics.middle + metrics.lineWidth), bounds.width, 1, color); | 
|         break; | 
|     } | 
|   } | 
|   | 
|   function getTextBounds(state, text, textDecoration, isLast, transform) { | 
|     var bounds; | 
|     if (support.rangeBounds && !transform) { | 
|       if (textDecoration !== "none" || Util.trimText(text).length !== 0) { | 
|         bounds = textRangeBounds(text, state.node, state.textOffset); | 
|       } | 
|       state.textOffset += text.length; | 
|     } else if (state.node && typeof state.node.nodeValue === "string" ){ | 
|       var newTextNode = (isLast) ? state.node.splitText(text.length) : null; | 
|       bounds = textWrapperBounds(state.node, transform); | 
|       state.node = newTextNode; | 
|     } | 
|     return bounds; | 
|   } | 
|   | 
|   function textRangeBounds(text, textNode, textOffset) { | 
|     var range = doc.createRange(); | 
|     range.setStart(textNode, textOffset); | 
|     range.setEnd(textNode, textOffset + text.length); | 
|     return range.getBoundingClientRect(); | 
|   } | 
|   | 
|   function textWrapperBounds(oldTextNode, transform) { | 
|     var parent = oldTextNode.parentNode, | 
|     wrapElement = doc.createElement('wrapper'), | 
|     backupText = oldTextNode.cloneNode(true); | 
|   | 
|     wrapElement.appendChild(oldTextNode.cloneNode(true)); | 
|     parent.replaceChild(wrapElement, oldTextNode); | 
|   | 
|     var bounds = transform ? Util.OffsetBounds(wrapElement) : Util.Bounds(wrapElement); | 
|     parent.replaceChild(backupText, wrapElement); | 
|     return bounds; | 
|   } | 
|   | 
|   function renderText(el, textNode, stack) { | 
|     var ctx = stack.ctx, | 
|     color = getCSS(el, "color"), | 
|     textDecoration = getCSS(el, "textDecoration"), | 
|     textAlign = getCSS(el, "textAlign"), | 
|     metrics, | 
|     textList, | 
|     state = { | 
|       node: textNode, | 
|       textOffset: 0 | 
|     }; | 
|   | 
|     if (Util.trimText(textNode.nodeValue).length > 0) { | 
|       textNode.nodeValue = textTransform(textNode.nodeValue, getCSS(el, "textTransform")); | 
|       textAlign = textAlign.replace(["-webkit-auto"],["auto"]); | 
|   | 
|       textList = (!options.letterRendering && /^(left|right|justify|auto)$/.test(textAlign) && noLetterSpacing(getCSS(el, "letterSpacing"))) ? | 
|       textNode.nodeValue.split(/(\b| )/) | 
|       : textNode.nodeValue.split(""); | 
|   | 
|       metrics = setTextVariables(ctx, el, textDecoration, color); | 
|   | 
|       if (options.chinese) { | 
|         textList.forEach(function(word, index) { | 
|           if (/.*[\u4E00-\u9FA5].*$/.test(word)) { | 
|             word = word.split(""); | 
|             word.unshift(index, 1); | 
|             textList.splice.apply(textList, word); | 
|           } | 
|         }); | 
|       } | 
|   | 
|       textList.forEach(function(text, index) { | 
|         var bounds = getTextBounds(state, text, textDecoration, (index < textList.length - 1), stack.transform.matrix); | 
|         if (bounds) { | 
|           drawText(text, bounds.left, bounds.bottom, ctx); | 
|           renderTextDecoration(ctx, textDecoration, bounds, metrics, color); | 
|         } | 
|       }); | 
|     } | 
|   } | 
|   | 
|   function listPosition (element, val) { | 
|     var boundElement = doc.createElement( "boundelement" ), | 
|     originalType, | 
|     bounds; | 
|   | 
|     boundElement.style.display = "inline"; | 
|   | 
|     originalType = element.style.listStyleType; | 
|     element.style.listStyleType = "none"; | 
|   | 
|     boundElement.appendChild(doc.createTextNode(val)); | 
|   | 
|     element.insertBefore(boundElement, element.firstChild); | 
|   | 
|     bounds = Util.Bounds(boundElement); | 
|     element.removeChild(boundElement); | 
|     element.style.listStyleType = originalType; | 
|     return bounds; | 
|   } | 
|   | 
|   function elementIndex(el) { | 
|     var i = -1, | 
|     count = 1, | 
|     childs = el.parentNode.childNodes; | 
|   | 
|     if (el.parentNode) { | 
|       while(childs[++i] !== el) { | 
|         if (childs[i].nodeType === 1) { | 
|           count++; | 
|         } | 
|       } | 
|       return count; | 
|     } else { | 
|       return -1; | 
|     } | 
|   } | 
|   | 
|   function listItemText(element, type) { | 
|     var currentIndex = elementIndex(element), text; | 
|     switch(type){ | 
|       case "decimal": | 
|         text = currentIndex; | 
|         break; | 
|       case "decimal-leading-zero": | 
|         text = (currentIndex.toString().length === 1) ? currentIndex = "0" + currentIndex.toString() : currentIndex.toString(); | 
|         break; | 
|       case "upper-roman": | 
|         text = _html2canvas.Generate.ListRoman( currentIndex ); | 
|         break; | 
|       case "lower-roman": | 
|         text = _html2canvas.Generate.ListRoman( currentIndex ).toLowerCase(); | 
|         break; | 
|       case "lower-alpha": | 
|         text = _html2canvas.Generate.ListAlpha( currentIndex ).toLowerCase(); | 
|         break; | 
|       case "upper-alpha": | 
|         text = _html2canvas.Generate.ListAlpha( currentIndex ); | 
|         break; | 
|     } | 
|   | 
|     return text + ". "; | 
|   } | 
|   | 
|   function renderListItem(element, stack, elBounds) { | 
|     var x, | 
|     text, | 
|     ctx = stack.ctx, | 
|     type = getCSS(element, "listStyleType"), | 
|     listBounds; | 
|   | 
|     if (/^(decimal|decimal-leading-zero|upper-alpha|upper-latin|upper-roman|lower-alpha|lower-greek|lower-latin|lower-roman)$/i.test(type)) { | 
|       text = listItemText(element, type); | 
|       listBounds = listPosition(element, text); | 
|       setTextVariables(ctx, element, "none", getCSS(element, "color")); | 
|   | 
|       if (getCSS(element, "listStylePosition") === "inside") { | 
|         ctx.setVariable("textAlign", "left"); | 
|         x = elBounds.left; | 
|       } else { | 
|         return; | 
|       } | 
|   | 
|       drawText(text, x, listBounds.bottom, ctx); | 
|     } | 
|   } | 
|   | 
|   function loadImage (src){ | 
|     var img = images[src]; | 
|     return (img && img.succeeded === true) ? img.img : false; | 
|   } | 
|   | 
|   function clipBounds(src, dst){ | 
|     var x = Math.max(src.left, dst.left), | 
|     y = Math.max(src.top, dst.top), | 
|     x2 = Math.min((src.left + src.width), (dst.left + dst.width)), | 
|     y2 = Math.min((src.top + src.height), (dst.top + dst.height)); | 
|   | 
|     return { | 
|       left:x, | 
|       top:y, | 
|       width:x2-x, | 
|       height:y2-y | 
|     }; | 
|   } | 
|   | 
|   function setZ(element, stack, parentStack){ | 
|     var newContext, | 
|     isPositioned = stack.cssPosition !== 'static', | 
|     zIndex = isPositioned ? getCSS(element, 'zIndex') : 'auto', | 
|     opacity = getCSS(element, 'opacity'), | 
|     isFloated = getCSS(element, 'cssFloat') !== 'none'; | 
|   | 
|     // https://developer.mozilla.org/en-US/docs/Web/Guide/CSS/Understanding_z_index/The_stacking_context | 
|     // When a new stacking context should be created: | 
|     // the root element (HTML), | 
|     // positioned (absolutely or relatively) with a z-index value other than "auto", | 
|     // elements with an opacity value less than 1. (See the specification for opacity), | 
|     // on mobile WebKit and Chrome 22+, position: fixed always creates a new stacking context, even when z-index is "auto" (See this post) | 
|   | 
|     stack.zIndex = newContext = h2czContext(zIndex); | 
|     newContext.isPositioned = isPositioned; | 
|     newContext.isFloated = isFloated; | 
|     newContext.opacity = opacity; | 
|     newContext.ownStacking = (zIndex !== 'auto' || opacity < 1); | 
|   | 
|     if (parentStack) { | 
|       parentStack.zIndex.children.push(stack); | 
|     } | 
|   } | 
|   | 
|   function renderImage(ctx, element, image, bounds, borders) { | 
|   | 
|     var paddingLeft = getCSSInt(element, 'paddingLeft'), | 
|     paddingTop = getCSSInt(element, 'paddingTop'), | 
|     paddingRight = getCSSInt(element, 'paddingRight'), | 
|     paddingBottom = getCSSInt(element, 'paddingBottom'); | 
|   | 
|     drawImage( | 
|       ctx, | 
|       image, | 
|       0, //sx | 
|       0, //sy | 
|       image.width, //sw | 
|       image.height, //sh | 
|       bounds.left + paddingLeft + borders[3].width, //dx | 
|       bounds.top + paddingTop + borders[0].width, // dy | 
|       bounds.width - (borders[1].width + borders[3].width + paddingLeft + paddingRight), //dw | 
|       bounds.height - (borders[0].width + borders[2].width + paddingTop + paddingBottom) //dh | 
|       ); | 
|   } | 
|   | 
|   function getBorderData(element) { | 
|     return ["Top", "Right", "Bottom", "Left"].map(function(side) { | 
|       return { | 
|         width: getCSSInt(element, 'border' + side + 'Width'), | 
|         color: getCSS(element, 'border' + side + 'Color') | 
|       }; | 
|     }); | 
|   } | 
|   | 
|   function getBorderRadiusData(element) { | 
|     return ["TopLeft", "TopRight", "BottomRight", "BottomLeft"].map(function(side) { | 
|       return getCSS(element, 'border' + side + 'Radius'); | 
|     }); | 
|   } | 
|   | 
|   var getCurvePoints = (function(kappa) { | 
|   | 
|     return function(x, y, r1, r2) { | 
|       var ox = (r1) * kappa, // control point offset horizontal | 
|       oy = (r2) * kappa, // control point offset vertical | 
|       xm = x + r1, // x-middle | 
|       ym = y + r2; // y-middle | 
|       return { | 
|         topLeft: bezierCurve({ | 
|           x:x, | 
|           y:ym | 
|         }, { | 
|           x:x, | 
|           y:ym - oy | 
|         }, { | 
|           x:xm - ox, | 
|           y:y | 
|         }, { | 
|           x:xm, | 
|           y:y | 
|         }), | 
|         topRight: bezierCurve({ | 
|           x:x, | 
|           y:y | 
|         }, { | 
|           x:x + ox, | 
|           y:y | 
|         }, { | 
|           x:xm, | 
|           y:ym - oy | 
|         }, { | 
|           x:xm, | 
|           y:ym | 
|         }), | 
|         bottomRight: bezierCurve({ | 
|           x:xm, | 
|           y:y | 
|         }, { | 
|           x:xm, | 
|           y:y + oy | 
|         }, { | 
|           x:x + ox, | 
|           y:ym | 
|         }, { | 
|           x:x, | 
|           y:ym | 
|         }), | 
|         bottomLeft: bezierCurve({ | 
|           x:xm, | 
|           y:ym | 
|         }, { | 
|           x:xm - ox, | 
|           y:ym | 
|         }, { | 
|           x:x, | 
|           y:y + oy | 
|         }, { | 
|           x:x, | 
|           y:y | 
|         }) | 
|       }; | 
|     }; | 
|   })(4 * ((Math.sqrt(2) - 1) / 3)); | 
|   | 
|   function bezierCurve(start, startControl, endControl, end) { | 
|   | 
|     var lerp = function (a, b, t) { | 
|       return { | 
|         x:a.x + (b.x - a.x) * t, | 
|         y:a.y + (b.y - a.y) * t | 
|       }; | 
|     }; | 
|   | 
|     return { | 
|       start: start, | 
|       startControl: startControl, | 
|       endControl: endControl, | 
|       end: end, | 
|       subdivide: function(t) { | 
|         var ab = lerp(start, startControl, t), | 
|         bc = lerp(startControl, endControl, t), | 
|         cd = lerp(endControl, end, t), | 
|         abbc = lerp(ab, bc, t), | 
|         bccd = lerp(bc, cd, t), | 
|         dest = lerp(abbc, bccd, t); | 
|         return [bezierCurve(start, ab, abbc, dest), bezierCurve(dest, bccd, cd, end)]; | 
|       }, | 
|       curveTo: function(borderArgs) { | 
|         borderArgs.push(["bezierCurve", startControl.x, startControl.y, endControl.x, endControl.y, end.x, end.y]); | 
|       }, | 
|       curveToReversed: function(borderArgs) { | 
|         borderArgs.push(["bezierCurve", endControl.x, endControl.y, startControl.x, startControl.y, start.x, start.y]); | 
|       } | 
|     }; | 
|   } | 
|   | 
|   function parseCorner(borderArgs, radius1, radius2, corner1, corner2, x, y) { | 
|     if (radius1[0] > 0 || radius1[1] > 0) { | 
|       borderArgs.push(["line", corner1[0].start.x, corner1[0].start.y]); | 
|       corner1[0].curveTo(borderArgs); | 
|       corner1[1].curveTo(borderArgs); | 
|     } else { | 
|       borderArgs.push(["line", x, y]); | 
|     } | 
|   | 
|     if (radius2[0] > 0 || radius2[1] > 0) { | 
|       borderArgs.push(["line", corner2[0].start.x, corner2[0].start.y]); | 
|     } | 
|   } | 
|   | 
|   function drawSide(borderData, radius1, radius2, outer1, inner1, outer2, inner2) { | 
|     var borderArgs = []; | 
|   | 
|     if (radius1[0] > 0 || radius1[1] > 0) { | 
|       borderArgs.push(["line", outer1[1].start.x, outer1[1].start.y]); | 
|       outer1[1].curveTo(borderArgs); | 
|     } else { | 
|       borderArgs.push([ "line", borderData.c1[0], borderData.c1[1]]); | 
|     } | 
|   | 
|     if (radius2[0] > 0 || radius2[1] > 0) { | 
|       borderArgs.push(["line", outer2[0].start.x, outer2[0].start.y]); | 
|       outer2[0].curveTo(borderArgs); | 
|       borderArgs.push(["line", inner2[0].end.x, inner2[0].end.y]); | 
|       inner2[0].curveToReversed(borderArgs); | 
|     } else { | 
|       borderArgs.push([ "line", borderData.c2[0], borderData.c2[1]]); | 
|       borderArgs.push([ "line", borderData.c3[0], borderData.c3[1]]); | 
|     } | 
|   | 
|     if (radius1[0] > 0 || radius1[1] > 0) { | 
|       borderArgs.push(["line", inner1[1].end.x, inner1[1].end.y]); | 
|       inner1[1].curveToReversed(borderArgs); | 
|     } else { | 
|       borderArgs.push([ "line", borderData.c4[0], borderData.c4[1]]); | 
|     } | 
|   | 
|     return borderArgs; | 
|   } | 
|   | 
|   function calculateCurvePoints(bounds, borderRadius, borders) { | 
|   | 
|     var x = bounds.left, | 
|     y = bounds.top, | 
|     width = bounds.width, | 
|     height = bounds.height, | 
|   | 
|     tlh = borderRadius[0][0], | 
|     tlv = borderRadius[0][1], | 
|     trh = borderRadius[1][0], | 
|     trv = borderRadius[1][1], | 
|     brh = borderRadius[2][0], | 
|     brv = borderRadius[2][1], | 
|     blh = borderRadius[3][0], | 
|     blv = borderRadius[3][1], | 
|   | 
|     topWidth = width - trh, | 
|     rightHeight = height - brv, | 
|     bottomWidth = width - brh, | 
|     leftHeight = height - blv; | 
|   | 
|     return { | 
|       topLeftOuter: getCurvePoints( | 
|         x, | 
|         y, | 
|         tlh, | 
|         tlv | 
|         ).topLeft.subdivide(0.5), | 
|   | 
|       topLeftInner: getCurvePoints( | 
|         x + borders[3].width, | 
|         y + borders[0].width, | 
|         Math.max(0, tlh - borders[3].width), | 
|         Math.max(0, tlv - borders[0].width) | 
|         ).topLeft.subdivide(0.5), | 
|   | 
|       topRightOuter: getCurvePoints( | 
|         x + topWidth, | 
|         y, | 
|         trh, | 
|         trv | 
|         ).topRight.subdivide(0.5), | 
|   | 
|       topRightInner: getCurvePoints( | 
|         x + Math.min(topWidth, width + borders[3].width), | 
|         y + borders[0].width, | 
|         (topWidth > width + borders[3].width) ? 0 :trh - borders[3].width, | 
|         trv - borders[0].width | 
|         ).topRight.subdivide(0.5), | 
|   | 
|       bottomRightOuter: getCurvePoints( | 
|         x + bottomWidth, | 
|         y + rightHeight, | 
|         brh, | 
|         brv | 
|         ).bottomRight.subdivide(0.5), | 
|   | 
|       bottomRightInner: getCurvePoints( | 
|         x + Math.min(bottomWidth, width + borders[3].width), | 
|         y + Math.min(rightHeight, height + borders[0].width), | 
|         Math.max(0, brh - borders[1].width), | 
|         Math.max(0, brv - borders[2].width) | 
|         ).bottomRight.subdivide(0.5), | 
|   | 
|       bottomLeftOuter: getCurvePoints( | 
|         x, | 
|         y + leftHeight, | 
|         blh, | 
|         blv | 
|         ).bottomLeft.subdivide(0.5), | 
|   | 
|       bottomLeftInner: getCurvePoints( | 
|         x + borders[3].width, | 
|         y + leftHeight, | 
|         Math.max(0, blh - borders[3].width), | 
|         Math.max(0, blv - borders[2].width) | 
|         ).bottomLeft.subdivide(0.5) | 
|     }; | 
|   } | 
|   | 
|   function getBorderClip(element, borderPoints, borders, radius, bounds) { | 
|     var backgroundClip = getCSS(element, 'backgroundClip'), | 
|     borderArgs = []; | 
|   | 
|     switch(backgroundClip) { | 
|       case "content-box": | 
|       case "padding-box": | 
|         parseCorner(borderArgs, radius[0], radius[1], borderPoints.topLeftInner, borderPoints.topRightInner, bounds.left + borders[3].width, bounds.top + borders[0].width); | 
|         parseCorner(borderArgs, radius[1], radius[2], borderPoints.topRightInner, borderPoints.bottomRightInner, bounds.left + bounds.width - borders[1].width, bounds.top + borders[0].width); | 
|         parseCorner(borderArgs, radius[2], radius[3], borderPoints.bottomRightInner, borderPoints.bottomLeftInner, bounds.left + bounds.width - borders[1].width, bounds.top + bounds.height - borders[2].width); | 
|         parseCorner(borderArgs, radius[3], radius[0], borderPoints.bottomLeftInner, borderPoints.topLeftInner, bounds.left + borders[3].width, bounds.top + bounds.height - borders[2].width); | 
|         break; | 
|   | 
|       default: | 
|         parseCorner(borderArgs, radius[0], radius[1], borderPoints.topLeftOuter, borderPoints.topRightOuter, bounds.left, bounds.top); | 
|         parseCorner(borderArgs, radius[1], radius[2], borderPoints.topRightOuter, borderPoints.bottomRightOuter, bounds.left + bounds.width, bounds.top); | 
|         parseCorner(borderArgs, radius[2], radius[3], borderPoints.bottomRightOuter, borderPoints.bottomLeftOuter, bounds.left + bounds.width, bounds.top + bounds.height); | 
|         parseCorner(borderArgs, radius[3], radius[0], borderPoints.bottomLeftOuter, borderPoints.topLeftOuter, bounds.left, bounds.top + bounds.height); | 
|         break; | 
|     } | 
|   | 
|     return borderArgs; | 
|   } | 
|   | 
|   function parseBorders(element, bounds, borders){ | 
|     var x = bounds.left, | 
|     y = bounds.top, | 
|     width = bounds.width, | 
|     height = bounds.height, | 
|     borderSide, | 
|     bx, | 
|     by, | 
|     bw, | 
|     bh, | 
|     borderArgs, | 
|     // http://www.w3.org/TR/css3-background/#the-border-radius | 
|     borderRadius = getBorderRadiusData(element), | 
|     borderPoints = calculateCurvePoints(bounds, borderRadius, borders), | 
|     borderData = { | 
|       clip: getBorderClip(element, borderPoints, borders, borderRadius, bounds), | 
|       borders: [] | 
|     }; | 
|   | 
|     for (borderSide = 0; borderSide < 4; borderSide++) { | 
|   | 
|       if (borders[borderSide].width > 0) { | 
|         bx = x; | 
|         by = y; | 
|         bw = width; | 
|         bh = height - (borders[2].width); | 
|   | 
|         switch(borderSide) { | 
|           case 0: | 
|             // top border | 
|             bh = borders[0].width; | 
|   | 
|             borderArgs = drawSide({ | 
|               c1: [bx, by], | 
|               c2: [bx + bw, by], | 
|               c3: [bx + bw - borders[1].width, by + bh], | 
|               c4: [bx + borders[3].width, by + bh] | 
|             }, borderRadius[0], borderRadius[1], | 
|             borderPoints.topLeftOuter, borderPoints.topLeftInner, borderPoints.topRightOuter, borderPoints.topRightInner); | 
|             break; | 
|           case 1: | 
|             // right border | 
|             bx = x + width - (borders[1].width); | 
|             bw = borders[1].width; | 
|   | 
|             borderArgs = drawSide({ | 
|               c1: [bx + bw, by], | 
|               c2: [bx + bw, by + bh + borders[2].width], | 
|               c3: [bx, by + bh], | 
|               c4: [bx, by + borders[0].width] | 
|             }, borderRadius[1], borderRadius[2], | 
|             borderPoints.topRightOuter, borderPoints.topRightInner, borderPoints.bottomRightOuter, borderPoints.bottomRightInner); | 
|             break; | 
|           case 2: | 
|             // bottom border | 
|             by = (by + height) - (borders[2].width); | 
|             bh = borders[2].width; | 
|   | 
|             borderArgs = drawSide({ | 
|               c1: [bx + bw, by + bh], | 
|               c2: [bx, by + bh], | 
|               c3: [bx + borders[3].width, by], | 
|               c4: [bx + bw - borders[3].width, by] | 
|             }, borderRadius[2], borderRadius[3], | 
|             borderPoints.bottomRightOuter, borderPoints.bottomRightInner, borderPoints.bottomLeftOuter, borderPoints.bottomLeftInner); | 
|             break; | 
|           case 3: | 
|             // left border | 
|             bw = borders[3].width; | 
|   | 
|             borderArgs = drawSide({ | 
|               c1: [bx, by + bh + borders[2].width], | 
|               c2: [bx, by], | 
|               c3: [bx + bw, by + borders[0].width], | 
|               c4: [bx + bw, by + bh] | 
|             }, borderRadius[3], borderRadius[0], | 
|             borderPoints.bottomLeftOuter, borderPoints.bottomLeftInner, borderPoints.topLeftOuter, borderPoints.topLeftInner); | 
|             break; | 
|         } | 
|   | 
|         borderData.borders.push({ | 
|           args: borderArgs, | 
|           color: borders[borderSide].color | 
|         }); | 
|   | 
|       } | 
|     } | 
|   | 
|     return borderData; | 
|   } | 
|   | 
|   function createShape(ctx, args) { | 
|     var shape = ctx.drawShape(); | 
|     args.forEach(function(border, index) { | 
|       shape[(index === 0) ? "moveTo" : border[0] + "To" ].apply(null, border.slice(1)); | 
|     }); | 
|     return shape; | 
|   } | 
|   | 
|   function renderBorders(ctx, borderArgs, color) { | 
|     if (color !== "transparent") { | 
|       ctx.setVariable( "fillStyle", color); | 
|       createShape(ctx, borderArgs); | 
|       ctx.fill(); | 
|       numDraws+=1; | 
|     } | 
|   } | 
|   | 
|   function renderFormValue (el, bounds, stack){ | 
|   | 
|     var valueWrap = doc.createElement('valuewrap'), | 
|     cssPropertyArray = ['lineHeight','textAlign','fontFamily','color','fontSize','paddingLeft','paddingTop','width','height','border','borderLeftWidth','borderTopWidth'], | 
|     textValue, | 
|     textNode; | 
|   | 
|     cssPropertyArray.forEach(function(property) { | 
|       try { | 
|         valueWrap.style[property] = getCSS(el, property); | 
|       } catch(e) { | 
|         // Older IE has issues with "border" | 
|         Util.log("html2canvas: Parse: Exception caught in renderFormValue: " + e.message); | 
|       } | 
|     }); | 
|   | 
|     valueWrap.style.borderColor = "black"; | 
|     valueWrap.style.borderStyle = "solid"; | 
|     valueWrap.style.display = "block"; | 
|     valueWrap.style.position = "absolute"; | 
|   | 
|     if (/^(submit|reset|button|text|password)$/.test(el.type) || el.nodeName === "SELECT"){ | 
|       valueWrap.style.lineHeight = getCSS(el, "height"); | 
|     } | 
|   | 
|     valueWrap.style.top = bounds.top + "px"; | 
|     valueWrap.style.left = bounds.left + "px"; | 
|   | 
|     textValue = (el.nodeName === "SELECT") ? (el.options[el.selectedIndex] || 0).text : el.value; | 
|     if(!textValue) { | 
|       textValue = el.placeholder; | 
|     } | 
|   | 
|     textNode = doc.createTextNode(textValue); | 
|   | 
|     valueWrap.appendChild(textNode); | 
|     body.appendChild(valueWrap); | 
|   | 
|     renderText(el, textNode, stack); | 
|     body.removeChild(valueWrap); | 
|   } | 
|   | 
|   function drawImage (ctx) { | 
|     ctx.drawImage.apply(ctx, Array.prototype.slice.call(arguments, 1)); | 
|     numDraws+=1; | 
|   } | 
|   | 
|   function getPseudoElement(el, which) { | 
|     var elStyle = window.getComputedStyle(el, which); | 
|     if(!elStyle || !elStyle.content || elStyle.content === "none" || elStyle.content === "-moz-alt-content" || elStyle.display === "none") { | 
|       return; | 
|     } | 
|     var content = elStyle.content + '', | 
|     first = content.substr( 0, 1 ); | 
|     //strips quotes | 
|     if(first === content.substr( content.length - 1 ) && first.match(/'|"/)) { | 
|       content = content.substr( 1, content.length - 2 ); | 
|     } | 
|   | 
|     var isImage = content.substr( 0, 3 ) === 'url', | 
|     elps = document.createElement( isImage ? 'img' : 'span' ); | 
|   | 
|     elps.className = pseudoHide + "-before " + pseudoHide + "-after"; | 
|   | 
|     Object.keys(elStyle).filter(indexedProperty).forEach(function(prop) { | 
|       // Prevent assigning of read only CSS Rules, ex. length, parentRule | 
|       try { | 
|         elps.style[prop] = elStyle[prop]; | 
|       } catch (e) { | 
|         Util.log(['Tried to assign readonly property ', prop, 'Error:', e]); | 
|       } | 
|     }); | 
|   | 
|     if(isImage) { | 
|       elps.src = Util.parseBackgroundImage(content)[0].args[0]; | 
|     } else { | 
|       elps.innerHTML = content; | 
|     } | 
|     return elps; | 
|   } | 
|   | 
|   function indexedProperty(property) { | 
|     return (isNaN(window.parseInt(property, 10))); | 
|   } | 
|   | 
|   function injectPseudoElements(el, stack) { | 
|     var before = getPseudoElement(el, ':before'), | 
|     after = getPseudoElement(el, ':after'); | 
|     if(!before && !after) { | 
|       return; | 
|     } | 
|   | 
|     if(before) { | 
|       el.className += " " + pseudoHide + "-before"; | 
|       el.parentNode.insertBefore(before, el); | 
|       parseElement(before, stack, true); | 
|       el.parentNode.removeChild(before); | 
|       el.className = el.className.replace(pseudoHide + "-before", "").trim(); | 
|     } | 
|   | 
|     if (after) { | 
|       el.className += " " + pseudoHide + "-after"; | 
|       el.appendChild(after); | 
|       parseElement(after, stack, true); | 
|       el.removeChild(after); | 
|       el.className = el.className.replace(pseudoHide + "-after", "").trim(); | 
|     } | 
|   | 
|   } | 
|   | 
|   function renderBackgroundRepeat(ctx, image, backgroundPosition, bounds) { | 
|     var offsetX = Math.round(bounds.left + backgroundPosition.left), | 
|     offsetY = Math.round(bounds.top + backgroundPosition.top); | 
|   | 
|     ctx.createPattern(image); | 
|     ctx.translate(offsetX, offsetY); | 
|     ctx.fill(); | 
|     ctx.translate(-offsetX, -offsetY); | 
|   } | 
|   | 
|   function backgroundRepeatShape(ctx, image, backgroundPosition, bounds, left, top, width, height) { | 
|     var args = []; | 
|     args.push(["line", Math.round(left), Math.round(top)]); | 
|     args.push(["line", Math.round(left + width), Math.round(top)]); | 
|     args.push(["line", Math.round(left + width), Math.round(height + top)]); | 
|     args.push(["line", Math.round(left), Math.round(height + top)]); | 
|     createShape(ctx, args); | 
|     ctx.save(); | 
|     ctx.clip(); | 
|     renderBackgroundRepeat(ctx, image, backgroundPosition, bounds); | 
|     ctx.restore(); | 
|   } | 
|   | 
|   function renderBackgroundColor(ctx, backgroundBounds, bgcolor) { | 
|     renderRect( | 
|       ctx, | 
|       backgroundBounds.left, | 
|       backgroundBounds.top, | 
|       backgroundBounds.width, | 
|       backgroundBounds.height, | 
|       bgcolor | 
|       ); | 
|   } | 
|   | 
|   function renderBackgroundRepeating(el, bounds, ctx, image, imageIndex) { | 
|     var backgroundSize = Util.BackgroundSize(el, bounds, image, imageIndex), | 
|     backgroundPosition = Util.BackgroundPosition(el, bounds, image, imageIndex, backgroundSize), | 
|     backgroundRepeat = getCSS(el, "backgroundRepeat").split(",").map(Util.trimText); | 
|   | 
|     image = resizeImage(image, backgroundSize); | 
|   | 
|     backgroundRepeat = backgroundRepeat[imageIndex] || backgroundRepeat[0]; | 
|   | 
|     switch (backgroundRepeat) { | 
|       case "repeat-x": | 
|         backgroundRepeatShape(ctx, image, backgroundPosition, bounds, | 
|           bounds.left, bounds.top + backgroundPosition.top, 99999, image.height); | 
|         break; | 
|   | 
|       case "repeat-y": | 
|         backgroundRepeatShape(ctx, image, backgroundPosition, bounds, | 
|           bounds.left + backgroundPosition.left, bounds.top, image.width, 99999); | 
|         break; | 
|   | 
|       case "no-repeat": | 
|         backgroundRepeatShape(ctx, image, backgroundPosition, bounds, | 
|           bounds.left + backgroundPosition.left, bounds.top + backgroundPosition.top, image.width, image.height); | 
|         break; | 
|   | 
|       default: | 
|         renderBackgroundRepeat(ctx, image, backgroundPosition, { | 
|           top: bounds.top, | 
|           left: bounds.left, | 
|           width: image.width, | 
|           height: image.height | 
|         }); | 
|         break; | 
|     } | 
|   } | 
|   | 
|   function renderBackgroundImage(element, bounds, ctx) { | 
|     var backgroundImage = getCSS(element, "backgroundImage"), | 
|     backgroundImages = Util.parseBackgroundImage(backgroundImage), | 
|     image, | 
|     imageIndex = backgroundImages.length; | 
|   | 
|     while(imageIndex--) { | 
|       backgroundImage = backgroundImages[imageIndex]; | 
|   | 
|       if (!backgroundImage.args || backgroundImage.args.length === 0) { | 
|         continue; | 
|       } | 
|   | 
|       var key = backgroundImage.method === 'url' ? | 
|       backgroundImage.args[0] : | 
|       backgroundImage.value; | 
|   | 
|       image = loadImage(key); | 
|   | 
|       // TODO add support for background-origin | 
|       if (image) { | 
|         renderBackgroundRepeating(element, bounds, ctx, image, imageIndex); | 
|       } else { | 
|         Util.log("html2canvas: Error loading background:", backgroundImage); | 
|       } | 
|     } | 
|   } | 
|   | 
|   function resizeImage(image, bounds) { | 
|     if(image.width === bounds.width && image.height === bounds.height) { | 
|       return image; | 
|     } | 
|   | 
|     var ctx, canvas = doc.createElement('canvas'); | 
|     canvas.width = bounds.width; | 
|     canvas.height = bounds.height; | 
|     ctx = canvas.getContext("2d"); | 
|     drawImage(ctx, image, 0, 0, image.width, image.height, 0, 0, bounds.width, bounds.height ); | 
|     return canvas; | 
|   } | 
|   | 
|   function setOpacity(ctx, element, parentStack) { | 
|     return ctx.setVariable("globalAlpha", getCSS(element, "opacity") * ((parentStack) ? parentStack.opacity : 1)); | 
|   } | 
|   | 
|   function removePx(str) { | 
|     return str.replace("px", ""); | 
|   } | 
|   | 
|   var transformRegExp = /(matrix)\((.+)\)/; | 
|   | 
|   function getTransform(element, parentStack) { | 
|     var transform = getCSS(element, "transform") || getCSS(element, "-webkit-transform") || getCSS(element, "-moz-transform") || getCSS(element, "-ms-transform") || getCSS(element, "-o-transform"); | 
|     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"; | 
|   | 
|     transformOrigin = transformOrigin.split(" ").map(removePx).map(Util.asFloat); | 
|   | 
|     var matrix; | 
|     if (transform && transform !== "none") { | 
|       var match = transform.match(transformRegExp); | 
|       if (match) { | 
|         switch(match[1]) { | 
|           case "matrix": | 
|             matrix = match[2].split(",").map(Util.trimText).map(Util.asFloat); | 
|             break; | 
|         } | 
|       } | 
|     } | 
|   | 
|     return { | 
|       origin: transformOrigin, | 
|       matrix: matrix | 
|     }; | 
|   } | 
|   | 
|   function createStack(element, parentStack, bounds, transform) { | 
|     var ctx = h2cRenderContext((!parentStack) ? documentWidth() : bounds.width , (!parentStack) ? documentHeight() : bounds.height), | 
|     stack = { | 
|       ctx: ctx, | 
|       opacity: setOpacity(ctx, element, parentStack), | 
|       cssPosition: getCSS(element, "position"), | 
|       borders: getBorderData(element), | 
|       transform: transform, | 
|       clip: (parentStack && parentStack.clip) ? Util.Extend( {}, parentStack.clip ) : null | 
|     }; | 
|   | 
|     setZ(element, stack, parentStack); | 
|   | 
|     // TODO correct overflow for absolute content residing under a static position | 
|     if (options.useOverflow === true && /(hidden|scroll|auto)/.test(getCSS(element, "overflow")) === true && /(BODY)/i.test(element.nodeName) === false){ | 
|       stack.clip = (stack.clip) ? clipBounds(stack.clip, bounds) : bounds; | 
|     } | 
|   | 
|     return stack; | 
|   } | 
|   | 
|   function getBackgroundBounds(borders, bounds, clip) { | 
|     var backgroundBounds = { | 
|       left: bounds.left + borders[3].width, | 
|       top: bounds.top + borders[0].width, | 
|       width: bounds.width - (borders[1].width + borders[3].width), | 
|       height: bounds.height - (borders[0].width + borders[2].width) | 
|     }; | 
|   | 
|     if (clip) { | 
|       backgroundBounds = clipBounds(backgroundBounds, clip); | 
|     } | 
|   | 
|     return backgroundBounds; | 
|   } | 
|   | 
|   function getBounds(element, transform) { | 
|     var bounds = (transform.matrix) ? Util.OffsetBounds(element) : Util.Bounds(element); | 
|     transform.origin[0] += bounds.left; | 
|     transform.origin[1] += bounds.top; | 
|     return bounds; | 
|   } | 
|   | 
|   function renderElement(element, parentStack, pseudoElement, ignoreBackground) { | 
|     var transform = getTransform(element, parentStack), | 
|     bounds = getBounds(element, transform), | 
|     image, | 
|     stack = createStack(element, parentStack, bounds, transform), | 
|     borders = stack.borders, | 
|     ctx = stack.ctx, | 
|     backgroundBounds = getBackgroundBounds(borders, bounds, stack.clip), | 
|     borderData = parseBorders(element, bounds, borders), | 
|     backgroundColor = (ignoreElementsRegExp.test(element.nodeName)) ? "#efefef" : getCSS(element, "backgroundColor"); | 
|   | 
|   | 
|     createShape(ctx, borderData.clip); | 
|   | 
|     ctx.save(); | 
|     ctx.clip(); | 
|   | 
|     if (backgroundBounds.height > 0 && backgroundBounds.width > 0 && !ignoreBackground) { | 
|       renderBackgroundColor(ctx, bounds, backgroundColor); | 
|       renderBackgroundImage(element, backgroundBounds, ctx); | 
|     } else if (ignoreBackground) { | 
|       stack.backgroundColor =  backgroundColor; | 
|     } | 
|   | 
|     ctx.restore(); | 
|   | 
|     borderData.borders.forEach(function(border) { | 
|       renderBorders(ctx, border.args, border.color); | 
|     }); | 
|   | 
|     if (!pseudoElement) { | 
|       injectPseudoElements(element, stack); | 
|     } | 
|   | 
|     switch(element.nodeName){ | 
|       case "IMG": | 
|         if ((image = loadImage(element.getAttribute('src')))) { | 
|           renderImage(ctx, element, image, bounds, borders); | 
|         } else { | 
|           Util.log("html2canvas: Error loading <img>:" + element.getAttribute('src')); | 
|         } | 
|         break; | 
|       case "INPUT": | 
|         // TODO add all relevant type's, i.e. HTML5 new stuff | 
|         // todo add support for placeholder attribute for browsers which support it | 
|         if (/^(text|url|email|submit|button|reset)$/.test(element.type) && (element.value || element.placeholder || "").length > 0){ | 
|           renderFormValue(element, bounds, stack); | 
|         } | 
|         break; | 
|       case "TEXTAREA": | 
|         if ((element.value || element.placeholder || "").length > 0){ | 
|           renderFormValue(element, bounds, stack); | 
|         } | 
|         break; | 
|       case "SELECT": | 
|         if ((element.options||element.placeholder || "").length > 0){ | 
|           renderFormValue(element, bounds, stack); | 
|         } | 
|         break; | 
|       case "LI": | 
|         renderListItem(element, stack, backgroundBounds); | 
|         break; | 
|       case "CANVAS": | 
|         renderImage(ctx, element, element, bounds, borders); | 
|         break; | 
|     } | 
|   | 
|     return stack; | 
|   } | 
|   | 
|   function isElementVisible(element) { | 
|     return (getCSS(element, 'display') !== "none" && getCSS(element, 'visibility') !== "hidden" && !element.hasAttribute("data-html2canvas-ignore")); | 
|   } | 
|   | 
|   function parseElement (element, stack, pseudoElement) { | 
|     if (isElementVisible(element)) { | 
|       stack = renderElement(element, stack, pseudoElement, false) || stack; | 
|       if (!ignoreElementsRegExp.test(element.nodeName)) { | 
|         parseChildren(element, stack, pseudoElement); | 
|       } | 
|     } | 
|   } | 
|   | 
|   function parseChildren(element, stack, pseudoElement) { | 
|     Util.Children(element).forEach(function(node) { | 
|       if (node.nodeType === node.ELEMENT_NODE) { | 
|         parseElement(node, stack, pseudoElement); | 
|       } else if (node.nodeType === node.TEXT_NODE) { | 
|         renderText(element, node, stack); | 
|       } | 
|     }); | 
|   } | 
|   | 
|   function init() { | 
|     var background = getCSS(document.documentElement, "backgroundColor"), | 
|       transparentBackground = (Util.isTransparent(background) && element === document.body), | 
|       stack = renderElement(element, null, false, transparentBackground); | 
|     parseChildren(element, stack); | 
|   | 
|     if (transparentBackground) { | 
|       background = stack.backgroundColor; | 
|     } | 
|   | 
|     body.removeChild(hidePseudoElements); | 
|     return { | 
|       backgroundColor: background, | 
|       stack: stack | 
|     }; | 
|   } | 
|   | 
|   return init(); | 
| }; | 
|   | 
| function h2czContext(zindex) { | 
|   return { | 
|     zindex: zindex, | 
|     children: [] | 
|   }; | 
| } | 
|   | 
| _html2canvas.Preload = function( options ) { | 
|   | 
|   var images = { | 
|     numLoaded: 0,   // also failed are counted here | 
|     numFailed: 0, | 
|     numTotal: 0, | 
|     cleanupDone: false | 
|   }, | 
|   pageOrigin, | 
|   Util = _html2canvas.Util, | 
|   methods, | 
|   i, | 
|   count = 0, | 
|   element = options.elements[0] || document.body, | 
|   doc = element.ownerDocument, | 
|   domImages = element.getElementsByTagName('img'), // Fetch images of the present element only | 
|   imgLen = domImages.length, | 
|   link = doc.createElement("a"), | 
|   supportCORS = (function( img ){ | 
|     return (img.crossOrigin !== undefined); | 
|   })(new Image()), | 
|   timeoutTimer; | 
|   | 
|   link.href = window.location.href; | 
|   pageOrigin  = link.protocol + link.host; | 
|   | 
|   function isSameOrigin(url){ | 
|     link.href = url; | 
|     link.href = link.href; // YES, BELIEVE IT OR NOT, that is required for IE9 - http://jsfiddle.net/niklasvh/2e48b/ | 
|     var origin = link.protocol + link.host; | 
|     return (origin === pageOrigin); | 
|   } | 
|   | 
|   function start(){ | 
|     Util.log("html2canvas: start: images: " + images.numLoaded + " / " + images.numTotal + " (failed: " + images.numFailed + ")"); | 
|     if (!images.firstRun && images.numLoaded >= images.numTotal){ | 
|       Util.log("Finished loading images: # " + images.numTotal + " (failed: " + images.numFailed + ")"); | 
|   | 
|       if (typeof options.complete === "function"){ | 
|         options.complete(images); | 
|       } | 
|   | 
|     } | 
|   } | 
|   | 
|   // TODO modify proxy to serve images with CORS enabled, where available | 
|   function proxyGetImage(url, img, imageObj){ | 
|     var callback_name, | 
|     scriptUrl = options.proxy, | 
|     script; | 
|   | 
|     link.href = url; | 
|     url = link.href; // work around for pages with base href="" set - WARNING: this may change the url | 
|   | 
|     callback_name = 'html2canvas_' + (count++); | 
|     imageObj.callbackname = callback_name; | 
|   | 
|     if (scriptUrl.indexOf("?") > -1) { | 
|       scriptUrl += "&"; | 
|     } else { | 
|       scriptUrl += "?"; | 
|     } | 
|     scriptUrl += 'url=' + encodeURIComponent(url) + '&callback=' + callback_name; | 
|     script = doc.createElement("script"); | 
|   | 
|     window[callback_name] = function(a){ | 
|       if (a.substring(0,6) === "error:"){ | 
|         imageObj.succeeded = false; | 
|         images.numLoaded++; | 
|         images.numFailed++; | 
|         start(); | 
|       } else { | 
|         setImageLoadHandlers(img, imageObj); | 
|         img.src = a; | 
|       } | 
|       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) | 
|       try { | 
|         delete window[callback_name];  // for all browser that support this | 
|       } catch(ex) {} | 
|       script.parentNode.removeChild(script); | 
|       script = null; | 
|       delete imageObj.script; | 
|       delete imageObj.callbackname; | 
|     }; | 
|   | 
|     script.setAttribute("type", "text/javascript"); | 
|     script.setAttribute("src", scriptUrl); | 
|     imageObj.script = script; | 
|     window.document.body.appendChild(script); | 
|   | 
|   } | 
|   | 
|   function loadPseudoElement(element, type) { | 
|     var style = window.getComputedStyle(element, type), | 
|     content = style.content; | 
|     if (content.substr(0, 3) === 'url') { | 
|       methods.loadImage(_html2canvas.Util.parseBackgroundImage(content)[0].args[0]); | 
|     } | 
|     loadBackgroundImages(style.backgroundImage, element); | 
|   } | 
|   | 
|   function loadPseudoElementImages(element) { | 
|     loadPseudoElement(element, ":before"); | 
|     loadPseudoElement(element, ":after"); | 
|   } | 
|   | 
|   function loadGradientImage(backgroundImage, bounds) { | 
|     var img = _html2canvas.Generate.Gradient(backgroundImage, bounds); | 
|   | 
|     if (img !== undefined){ | 
|       images[backgroundImage] = { | 
|         img: img, | 
|         succeeded: true | 
|       }; | 
|       images.numTotal++; | 
|       images.numLoaded++; | 
|       start(); | 
|     } | 
|   } | 
|   | 
|   function invalidBackgrounds(background_image) { | 
|     return (background_image && background_image.method && background_image.args && background_image.args.length > 0 ); | 
|   } | 
|   | 
|   function loadBackgroundImages(background_image, el) { | 
|     var bounds; | 
|   | 
|     _html2canvas.Util.parseBackgroundImage(background_image).filter(invalidBackgrounds).forEach(function(background_image) { | 
|       if (background_image.method === 'url') { | 
|         methods.loadImage(background_image.args[0]); | 
|       } else if(background_image.method.match(/\-?gradient$/)) { | 
|         if(bounds === undefined) { | 
|           bounds = _html2canvas.Util.Bounds(el); | 
|         } | 
|         loadGradientImage(background_image.value, bounds); | 
|       } | 
|     }); | 
|   } | 
|   | 
|   function getImages (el) { | 
|     var elNodeType = false; | 
|   | 
|     // Firefox fails with permission denied on pages with iframes | 
|     try { | 
|       Util.Children(el).forEach(getImages); | 
|     } | 
|     catch( e ) {} | 
|   | 
|     try { | 
|       elNodeType = el.nodeType; | 
|     } catch (ex) { | 
|       elNodeType = false; | 
|       Util.log("html2canvas: failed to access some element's nodeType - Exception: " + ex.message); | 
|     } | 
|   | 
|     if (elNodeType === 1 || elNodeType === undefined) { | 
|       loadPseudoElementImages(el); | 
|       try { | 
|         loadBackgroundImages(Util.getCSS(el, 'backgroundImage'), el); | 
|       } catch(e) { | 
|         Util.log("html2canvas: failed to get background-image - Exception: " + e.message); | 
|       } | 
|       loadBackgroundImages(el); | 
|     } | 
|   } | 
|   | 
|   function setImageLoadHandlers(img, imageObj) { | 
|     img.onload = function() { | 
|       if ( imageObj.timer !== undefined ) { | 
|         // CORS succeeded | 
|         window.clearTimeout( imageObj.timer ); | 
|       } | 
|   | 
|       images.numLoaded++; | 
|       imageObj.succeeded = true; | 
|       img.onerror = img.onload = null; | 
|       start(); | 
|     }; | 
|     img.onerror = function() { | 
|       if (img.crossOrigin === "anonymous") { | 
|         // CORS failed | 
|         window.clearTimeout( imageObj.timer ); | 
|   | 
|         // let's try with proxy instead | 
|         if ( options.proxy ) { | 
|           var src = img.src; | 
|           img = new Image(); | 
|           imageObj.img = img; | 
|           img.src = src; | 
|   | 
|           proxyGetImage( img.src, img, imageObj ); | 
|           return; | 
|         } | 
|       } | 
|   | 
|       images.numLoaded++; | 
|       images.numFailed++; | 
|       imageObj.succeeded = false; | 
|       img.onerror = img.onload = null; | 
|       start(); | 
|     }; | 
|   } | 
|   | 
|   methods = { | 
|     loadImage: function( src ) { | 
|       var img, imageObj; | 
|       if ( src && images[src] === undefined ) { | 
|         img = new Image(); | 
|         if ( src.match(/data:image\/.*;base64,/i) ) { | 
|           img.src = src.replace(/url\(['"]{0,}|['"]{0,}\)$/ig, ''); | 
|           imageObj = images[src] = { | 
|             img: img | 
|           }; | 
|           images.numTotal++; | 
|           setImageLoadHandlers(img, imageObj); | 
|         } else if ( isSameOrigin( src ) || options.allowTaint ===  true ) { | 
|           imageObj = images[src] = { | 
|             img: img | 
|           }; | 
|           images.numTotal++; | 
|           setImageLoadHandlers(img, imageObj); | 
|           img.src = src; | 
|         } else if ( supportCORS && !options.allowTaint && options.useCORS ) { | 
|           // attempt to load with CORS | 
|   | 
|           img.crossOrigin = "anonymous"; | 
|           imageObj = images[src] = { | 
|             img: img | 
|           }; | 
|           images.numTotal++; | 
|           setImageLoadHandlers(img, imageObj); | 
|           img.src = src; | 
|         } else if ( options.proxy ) { | 
|           imageObj = images[src] = { | 
|             img: img | 
|           }; | 
|           images.numTotal++; | 
|           proxyGetImage( src, img, imageObj ); | 
|         } | 
|       } | 
|   | 
|     }, | 
|     cleanupDOM: function(cause) { | 
|       var img, src; | 
|       if (!images.cleanupDone) { | 
|         if (cause && typeof cause === "string") { | 
|           Util.log("html2canvas: Cleanup because: " + cause); | 
|         } else { | 
|           Util.log("html2canvas: Cleanup after timeout: " + options.timeout + " ms."); | 
|         } | 
|   | 
|         for (src in images) { | 
|           if (images.hasOwnProperty(src)) { | 
|             img = images[src]; | 
|             if (typeof img === "object" && img.callbackname && img.succeeded === undefined) { | 
|               // cancel proxy image request | 
|               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) | 
|               try { | 
|                 delete window[img.callbackname];  // for all browser that support this | 
|               } catch(ex) {} | 
|               if (img.script && img.script.parentNode) { | 
|                 img.script.setAttribute("src", "about:blank");  // try to cancel running request | 
|                 img.script.parentNode.removeChild(img.script); | 
|               } | 
|               images.numLoaded++; | 
|               images.numFailed++; | 
|               Util.log("html2canvas: Cleaned up failed img: '" + src + "' Steps: " + images.numLoaded + " / " + images.numTotal); | 
|             } | 
|           } | 
|         } | 
|   | 
|         // cancel any pending requests | 
|         if(window.stop !== undefined) { | 
|           window.stop(); | 
|         } else if(document.execCommand !== undefined) { | 
|           document.execCommand("Stop", false); | 
|         } | 
|         if (document.close !== undefined) { | 
|           document.close(); | 
|         } | 
|         images.cleanupDone = true; | 
|         if (!(cause && typeof cause === "string")) { | 
|           start(); | 
|         } | 
|       } | 
|     }, | 
|   | 
|     renderingDone: function() { | 
|       if (timeoutTimer) { | 
|         window.clearTimeout(timeoutTimer); | 
|       } | 
|     } | 
|   }; | 
|   | 
|   if (options.timeout > 0) { | 
|     timeoutTimer = window.setTimeout(methods.cleanupDOM, options.timeout); | 
|   } | 
|   | 
|   Util.log('html2canvas: Preload starts: finding background-images'); | 
|   images.firstRun = true; | 
|   | 
|   getImages(element); | 
|   | 
|   Util.log('html2canvas: Preload: Finding images'); | 
|   // load <img> images | 
|   for (i = 0; i < imgLen; i+=1){ | 
|     methods.loadImage( domImages[i].getAttribute( "src" ) ); | 
|   } | 
|   | 
|   images.firstRun = false; | 
|   Util.log('html2canvas: Preload: Done.'); | 
|   if (images.numTotal === images.numLoaded) { | 
|     start(); | 
|   } | 
|   | 
|   return methods; | 
| }; | 
|   | 
| _html2canvas.Renderer = function(parseQueue, options){ | 
|   | 
|   // http://www.w3.org/TR/CSS21/zindex.html | 
|   function createRenderQueue(parseQueue) { | 
|     var queue = [], | 
|     rootContext; | 
|   | 
|     rootContext = (function buildStackingContext(rootNode) { | 
|       var rootContext = {}; | 
|       function insert(context, node, specialParent) { | 
|         var zi = (node.zIndex.zindex === 'auto') ? 0 : Number(node.zIndex.zindex), | 
|         contextForChildren = context, // the stacking context for children | 
|         isPositioned = node.zIndex.isPositioned, | 
|         isFloated = node.zIndex.isFloated, | 
|         stub = {node: node}, | 
|         childrenDest = specialParent; // where children without z-index should be pushed into | 
|   | 
|         if (node.zIndex.ownStacking) { | 
|           // '!' comes before numbers in sorted array | 
|           contextForChildren = stub.context = { '!': [{node:node, children: []}]}; | 
|           childrenDest = undefined; | 
|         } else if (isPositioned || isFloated) { | 
|           childrenDest = stub.children = []; | 
|         } | 
|   | 
|         if (zi === 0 && specialParent) { | 
|           specialParent.push(stub); | 
|         } else { | 
|           if (!context[zi]) { context[zi] = []; } | 
|           context[zi].push(stub); | 
|         } | 
|   | 
|         node.zIndex.children.forEach(function(childNode) { | 
|           insert(contextForChildren, childNode, childrenDest); | 
|         }); | 
|       } | 
|       insert(rootContext, rootNode); | 
|       return rootContext; | 
|     })(parseQueue); | 
|   | 
|     function sortZ(context) { | 
|       Object.keys(context).sort().forEach(function(zi) { | 
|         var nonPositioned = [], | 
|         floated = [], | 
|         positioned = [], | 
|         list = []; | 
|   | 
|         // positioned after static | 
|         context[zi].forEach(function(v) { | 
|           if (v.node.zIndex.isPositioned || v.node.zIndex.opacity < 1) { | 
|             // http://www.w3.org/TR/css3-color/#transparency | 
|             // non-positioned element with opactiy < 1 should be stacked as if it were a positioned element with ‘z-index: 0’ and ‘opacity: 1’. | 
|             positioned.push(v); | 
|           } else if (v.node.zIndex.isFloated) { | 
|             floated.push(v); | 
|           } else { | 
|             nonPositioned.push(v); | 
|           } | 
|         }); | 
|   | 
|         (function walk(arr) { | 
|           arr.forEach(function(v) { | 
|             list.push(v); | 
|             if (v.children) { walk(v.children); } | 
|           }); | 
|         })(nonPositioned.concat(floated, positioned)); | 
|   | 
|         list.forEach(function(v) { | 
|           if (v.context) { | 
|             sortZ(v.context); | 
|           } else { | 
|             queue.push(v.node); | 
|           } | 
|         }); | 
|       }); | 
|     } | 
|   | 
|     sortZ(rootContext); | 
|   | 
|     return queue; | 
|   } | 
|   | 
|   function getRenderer(rendererName) { | 
|     var renderer; | 
|   | 
|     if (typeof options.renderer === "string" && _html2canvas.Renderer[rendererName] !== undefined) { | 
|       renderer = _html2canvas.Renderer[rendererName](options); | 
|     } else if (typeof rendererName === "function") { | 
|       renderer = rendererName(options); | 
|     } else { | 
|       throw new Error("Unknown renderer"); | 
|     } | 
|   | 
|     if ( typeof renderer !== "function" ) { | 
|       throw new Error("Invalid renderer defined"); | 
|     } | 
|     return renderer; | 
|   } | 
|   | 
|   return getRenderer(options.renderer)(parseQueue, options, document, createRenderQueue(parseQueue.stack), _html2canvas); | 
| }; | 
|   | 
| _html2canvas.Util.Support = function (options, doc) { | 
|   | 
|   function supportSVGRendering() { | 
|     var img = new Image(), | 
|     canvas = doc.createElement("canvas"), | 
|     ctx = (canvas.getContext === undefined) ? false : canvas.getContext("2d"); | 
|     if (ctx === false) { | 
|       return false; | 
|     } | 
|     canvas.width = canvas.height = 10; | 
|     img.src = [ | 
|     "data:image/svg+xml,", | 
|     "<svg xmlns='http://www.w3.org/2000/svg' width='10' height='10'>", | 
|     "<foreignObject width='10' height='10'>", | 
|     "<div xmlns='http://www.w3.org/1999/xhtml' style='width:10;height:10;'>", | 
|     "sup", | 
|     "</div>", | 
|     "</foreignObject>", | 
|     "</svg>" | 
|     ].join(""); | 
|     try { | 
|       ctx.drawImage(img, 0, 0); | 
|       canvas.toDataURL(); | 
|     } catch(e) { | 
|       return false; | 
|     } | 
|     _html2canvas.Util.log('html2canvas: Parse: SVG powered rendering available'); | 
|     return true; | 
|   } | 
|   | 
|   // Test whether we can use ranges to measure bounding boxes | 
|   // Opera doesn't provide valid bounds.height/bottom even though it supports the method. | 
|   | 
|   function supportRangeBounds() { | 
|     var r, testElement, rangeBounds, rangeHeight, support = false; | 
|   | 
|     if (doc.createRange) { | 
|       r = doc.createRange(); | 
|       if (r.getBoundingClientRect) { | 
|         testElement = doc.createElement('boundtest'); | 
|         testElement.style.height = "123px"; | 
|         testElement.style.display = "block"; | 
|         doc.body.appendChild(testElement); | 
|   | 
|         r.selectNode(testElement); | 
|         rangeBounds = r.getBoundingClientRect(); | 
|         rangeHeight = rangeBounds.height; | 
|   | 
|         if (rangeHeight === 123) { | 
|           support = true; | 
|         } | 
|         doc.body.removeChild(testElement); | 
|       } | 
|     } | 
|   | 
|     return support; | 
|   } | 
|   | 
|   return { | 
|     rangeBounds: supportRangeBounds(), | 
|     svgRendering: options.svgRendering && supportSVGRendering() | 
|   }; | 
| }; | 
| window.html2canvas = function(elements, opts) { | 
|   elements = (elements.length) ? elements : [elements]; | 
|   var queue, | 
|   canvas, | 
|   options = { | 
|     // general | 
|     logging: false, | 
|     elements: elements, | 
|     background: "#fff", | 
|   | 
|     // preload options | 
|     proxy: null, | 
|     timeout: 0,    // no timeout | 
|     useCORS: false, // try to load images as CORS (where available), before falling back to proxy | 
|     allowTaint: false, // whether to allow images to taint the canvas, won't need proxy if set to true | 
|   | 
|     // parse options | 
|     svgRendering: false, // use svg powered rendering where available (FF11+) | 
|     ignoreElements: "IFRAME|OBJECT|PARAM", | 
|     useOverflow: true, | 
|     letterRendering: false, | 
|     chinese: false, | 
|   | 
|     // render options | 
|   | 
|     width: null, | 
|     height: null, | 
|     taintTest: true, // do a taint test with all images before applying to canvas | 
|     renderer: "Canvas" | 
|   }; | 
|   | 
|   options = _html2canvas.Util.Extend(opts, options); | 
|   | 
|   _html2canvas.logging = options.logging; | 
|   options.complete = function( images ) { | 
|   | 
|     if (typeof options.onpreloaded === "function") { | 
|       if ( options.onpreloaded( images ) === false ) { | 
|         return; | 
|       } | 
|     } | 
|     queue = _html2canvas.Parse( images, options ); | 
|   | 
|     if (typeof options.onparsed === "function") { | 
|       if ( options.onparsed( queue ) === false ) { | 
|         return; | 
|       } | 
|     } | 
|   | 
|     canvas = _html2canvas.Renderer( queue, options ); | 
|   | 
|     if (typeof options.onrendered === "function") { | 
|       options.onrendered( canvas ); | 
|     } | 
|   | 
|   | 
|   }; | 
|   | 
|   // for pages without images, we still want this to be async, i.e. return methods before executing | 
|   window.setTimeout( function(){ | 
|     _html2canvas.Preload( options ); | 
|   }, 0 ); | 
|   | 
|   return { | 
|     render: function( queue, opts ) { | 
|       return _html2canvas.Renderer( queue, _html2canvas.Util.Extend(opts, options) ); | 
|     }, | 
|     parse: function( images, opts ) { | 
|       return _html2canvas.Parse( images, _html2canvas.Util.Extend(opts, options) ); | 
|     }, | 
|     preload: function( opts ) { | 
|       return _html2canvas.Preload( _html2canvas.Util.Extend(opts, options) ); | 
|     }, | 
|     log: _html2canvas.Util.log | 
|   }; | 
| }; | 
|   | 
| window.html2canvas.log = _html2canvas.Util.log; // for renderers | 
| window.html2canvas.Renderer = { | 
|   Canvas: undefined // We are assuming this will be used | 
| }; | 
| _html2canvas.Renderer.Canvas = function(options) { | 
|   options = options || {}; | 
|   | 
|   var doc = document, | 
|   safeImages = [], | 
|   testCanvas = document.createElement("canvas"), | 
|   testctx = testCanvas.getContext("2d"), | 
|   Util = _html2canvas.Util, | 
|   canvas = options.canvas || doc.createElement('canvas'); | 
|   | 
|   function createShape(ctx, args) { | 
|     ctx.beginPath(); | 
|     args.forEach(function(arg) { | 
|       ctx[arg.name].apply(ctx, arg['arguments']); | 
|     }); | 
|     ctx.closePath(); | 
|   } | 
|   | 
|   function safeImage(item) { | 
|     if (safeImages.indexOf(item['arguments'][0].src ) === -1) { | 
|       testctx.drawImage(item['arguments'][0], 0, 0); | 
|       try { | 
|         testctx.getImageData(0, 0, 1, 1); | 
|       } catch(e) { | 
|         testCanvas = doc.createElement("canvas"); | 
|         testctx = testCanvas.getContext("2d"); | 
|         return false; | 
|       } | 
|       safeImages.push(item['arguments'][0].src); | 
|     } | 
|     return true; | 
|   } | 
|   | 
|   function renderItem(ctx, item) { | 
|     switch(item.type){ | 
|       case "variable": | 
|         ctx[item.name] = item['arguments']; | 
|         break; | 
|       case "function": | 
|         switch(item.name) { | 
|           case "createPattern": | 
|             if (item['arguments'][0].width > 0 && item['arguments'][0].height > 0) { | 
|               try { | 
|                 ctx.fillStyle = ctx.createPattern(item['arguments'][0], "repeat"); | 
|               } | 
|               catch(e) { | 
|                 Util.log("html2canvas: Renderer: Error creating pattern", e.message); | 
|               } | 
|             } | 
|             break; | 
|           case "drawShape": | 
|             createShape(ctx, item['arguments']); | 
|             break; | 
|           case "drawImage": | 
|             if (item['arguments'][8] > 0 && item['arguments'][7] > 0) { | 
|               if (!options.taintTest || (options.taintTest && safeImage(item))) { | 
|                 ctx.drawImage.apply( ctx, item['arguments'] ); | 
|               } | 
|             } | 
|             break; | 
|           default: | 
|             ctx[item.name].apply(ctx, item['arguments']); | 
|         } | 
|         break; | 
|     } | 
|   } | 
|   | 
|   return function(parsedData, options, document, queue, _html2canvas) { | 
|     var ctx = canvas.getContext("2d"), | 
|     newCanvas, | 
|     bounds, | 
|     fstyle, | 
|     zStack = parsedData.stack; | 
|   | 
|     canvas.width = canvas.style.width =  options.width || zStack.ctx.width; | 
|     canvas.height = canvas.style.height = options.height || zStack.ctx.height; | 
|   | 
|     fstyle = ctx.fillStyle; | 
|     ctx.fillStyle = (Util.isTransparent(zStack.backgroundColor) && options.background !== undefined) ? options.background : parsedData.backgroundColor; | 
|     ctx.fillRect(0, 0, canvas.width, canvas.height); | 
|     ctx.fillStyle = fstyle; | 
|   | 
|     queue.forEach(function(storageContext) { | 
|       // set common settings for canvas | 
|       ctx.textBaseline = "bottom"; | 
|       ctx.save(); | 
|   | 
|       if (storageContext.transform.matrix) { | 
|         ctx.translate(storageContext.transform.origin[0], storageContext.transform.origin[1]); | 
|         ctx.transform.apply(ctx, storageContext.transform.matrix); | 
|         ctx.translate(-storageContext.transform.origin[0], -storageContext.transform.origin[1]); | 
|       } | 
|   | 
|       if (storageContext.clip){ | 
|         ctx.beginPath(); | 
|         ctx.rect(storageContext.clip.left, storageContext.clip.top, storageContext.clip.width, storageContext.clip.height); | 
|         ctx.clip(); | 
|       } | 
|   | 
|       if (storageContext.ctx.storage) { | 
|         storageContext.ctx.storage.forEach(function(item) { | 
|           renderItem(ctx, item); | 
|         }); | 
|       } | 
|   | 
|       ctx.restore(); | 
|     }); | 
|   | 
|     Util.log("html2canvas: Renderer: Canvas renderer done - returning canvas obj"); | 
|   | 
|     if (options.elements.length === 1) { | 
|       if (typeof options.elements[0] === "object" && options.elements[0].nodeName !== "BODY") { | 
|         // crop image to the bounds of selected (single) element | 
|         bounds = _html2canvas.Util.Bounds(options.elements[0]); | 
|         newCanvas = document.createElement('canvas'); | 
|         newCanvas.width = Math.ceil(bounds.width); | 
|         newCanvas.height = Math.ceil(bounds.height); | 
|         ctx = newCanvas.getContext("2d"); | 
|   | 
|         ctx.drawImage(canvas, bounds.left, bounds.top, bounds.width, bounds.height, 0, 0, bounds.width, bounds.height); | 
|         canvas = null; | 
|         return newCanvas; | 
|       } | 
|     } | 
|   | 
|     return canvas; | 
|   }; | 
| }; | 
| })(window,document); |