SCDNG PELER
Server IP : 144.76.124.212  /  Your IP : 216.73.216.138
Web Server : LiteSpeed
System : Linux l4cp.vnetindia.com 4.18.0-553.40.1.lve.el8.x86_64 #1 SMP Wed Feb 12 18:54:57 UTC 2025 x86_64
User : rakcha ( 1356)
PHP Version : 7.4.33
Disable Function : NONE
MySQL : OFF  |  cURL : ON  |  WGET : ON  |  Perl : ON  |  Python : ON  |  Sudo : OFF  |  Pkexec : OFF
Directory :  /home5/rakcha/www/app/js/

Upload File :
current_dir [ Writeable ] document_root [ Writeable ]

 

Command :


[ HOME ]     

Current File : /home5/rakcha/www/app/js/html2canvas.js
/*

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




Anon7 - 2022
SCDN GOK