html2canvas

需求:H5页面,进入页面后,将H5页面转为图片,方便用户保存,转发,以及识别页面中 的二维码

使用html2canvas遇到的问题

由于html2canvas插件的局限性,只支持截取当前浏览器可视区域的内容,所以导致整个H5页面截取不完整

网上有很多解决办法,包括
1:修改canvas画布大小,按一比二扩大什么的
2:修改插件源码,让其乘以二什么等等
有很多说法,但我亲测了几个,对我都没有什么用,并没有解决我的问题

方案:html2canvas是通过选取dom节点,来进行canvas绘制,当元素设置为不可见的时候是无法绘制的,所以我采用了一种取巧的方法,在html2canvas方法出发前,将准备截图的dom节点复制到一个隐藏在页面最下层的dom节点内,将该节点的层级设置为最底层,即:z-index最小。
position: absolute;
z-index: 0;
截图时,选取dom节点为该节点,就可以完整的将所有区域都截取到!
然后我们就可以在onrendered回调中对截取的图片进行处理。

  var height = $('#content').height();//克隆节点,默认为false,不复制方法属性,为true是全部复制。
 var cloneDom = $('#content').clone(true);//设置克隆节点的css属性,因为之前的层级为0,我们只需要比被克隆的节点层级低即可。
    cloneDom.css({
        "background-color": "white",
        "position": "absolute",
        "top": "0px",
        "z-index": "-1",
        "height": height
    });
    //将克隆节点动态追加到body后面。
    $("body").append(cloneDom);
    //插件生成base64img图片。
    html2canvas(cloneDom, {
        onrendered: function(canvas) {
            var image = new Image();
            image.src = canvas.toDataURL("./static/images");
            //打印出来之后:....
            //可以通过chrome来查看
            document.body.appendChild(image);
            $('#content').addClass('hide');
            $('.mask').removeClass('hide');
            // console.log(img);
        }
    });
     ```



html2canvas代码

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

“`

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值