diff --git a/.gitignore b/.gitignore
new file mode 100644
index 00000000..62c89355
--- /dev/null
+++ b/.gitignore
@@ -0,0 +1 @@
+.idea/
\ No newline at end of file
diff --git a/bower.json b/bower.json
new file mode 100644
index 00000000..1d1e8f40
--- /dev/null
+++ b/bower.json
@@ -0,0 +1,18 @@
+{
+ "name": "tableExport.jquery.plugin",
+ "version": "0.0.0",
+ "homepage": "https://github.com/caguthrie/tableExport.jquery.plugin",
+ "authors": [
+ "Christopher Guthrie "
+ ],
+ "description": "tableExport plugin",
+ "license": "MIT",
+ "main": ["jquery.base64.js", "jspdf/jspdf.js"],
+ "ignore": [
+ "**/.*",
+ "node_modules",
+ "bower_components",
+ "test",
+ "tests"
+ ]
+}
diff --git a/html2canvas.js b/html2canvas.js
deleted file mode 100644
index df3ee7ec..00000000
--- a/html2canvas.js
+++ /dev/null
@@ -1,3010 +0,0 @@
-/*
- html2canvas 0.4.1
- Copyright (c) 2013 Niklas von Hertzen
-
- Released under MIT License
-*/
-
-(function(window, document, undefined){
-
-//"use strict";
-
-var _html2canvas = {},
-previousElement,
-computedCSS,
-html2canvas;
-
-_html2canvas.Util = {};
-
-_html2canvas.Util.log = function(a) {
- if (_html2canvas.logging && window.console && window.console.log) {
- window.console.log(a);
- }
-};
-
-_html2canvas.Util.trimText = (function(isNative){
- return function(input) {
- return isNative ? isNative.apply(input) : ((input || '') + '').replace( /^\s+|\s+$/g , '' );
- };
-})(String.prototype.trim);
-
-_html2canvas.Util.asFloat = function(v) {
- return parseFloat(v);
-};
-
-(function() {
- // TODO: support all possible length values
- var TEXT_SHADOW_PROPERTY = /((rgba|rgb)\([^\)]+\)(\s-?\d+px){0,})/g;
- var TEXT_SHADOW_VALUES = /(-?\d+px)|(#.+)|(rgb\(.+\))|(rgba\(.+\))/g;
- _html2canvas.Util.parseTextShadows = function (value) {
- if (!value || value === 'none') {
- return [];
- }
-
- // find multiple shadow declarations
- var shadows = value.match(TEXT_SHADOW_PROPERTY),
- results = [];
- for (var i = 0; shadows && (i < shadows.length); i++) {
- var s = shadows[i].match(TEXT_SHADOW_VALUES);
- results.push({
- color: s[0],
- offsetX: s[1] ? s[1].replace('px', '') : 0,
- offsetY: s[2] ? s[2].replace('px', '') : 0,
- blur: s[3] ? s[3].replace('px', '') : 0
- });
- }
- return results;
- };
-})();
-
-_html2canvas.Util.parseBackgroundImage = function (value) {
- var whitespace = ' \r\n\t',
- method, definition, prefix, prefix_i, block, results = [],
- c, mode = 0, numParen = 0, quote, args;
-
- var appendResult = function(){
- if(method) {
- if(definition.substr( 0, 1 ) === '"') {
- definition = definition.substr( 1, definition.length - 2 );
- }
- if(definition) {
- args.push(definition);
- }
- if(method.substr( 0, 1 ) === '-' &&
- (prefix_i = method.indexOf( '-', 1 ) + 1) > 0) {
- prefix = method.substr( 0, prefix_i);
- method = method.substr( prefix_i );
- }
- results.push({
- prefix: prefix,
- method: method.toLowerCase(),
- value: block,
- args: args
- });
- }
- args = []; //for some odd reason, setting .length = 0 didn't work in safari
- method =
- prefix =
- definition =
- block = '';
- };
-
- appendResult();
- for(var i = 0, ii = value.length; i -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 isPercentage(value) {
- return value.toString().indexOf("%") !== -1;
-}
-
-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)$/))) {
- return value;
- } 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
- };
-};
-
-_html2canvas.Util.BackgroundPosition = function(element, bounds, image, imageIndex, backgroundSize ) {
- var backgroundPosition = _html2canvas.Util.getCSS(element, 'backgroundPosition', imageIndex),
- leftPosition,
- topPosition;
- if (backgroundPosition.length === 1){
- backgroundPosition = [backgroundPosition[0], backgroundPosition[0]];
- }
-
- if (isPercentage(backgroundPosition[0])){
- leftPosition = (bounds.width - (backgroundSize || image).width) * (parseFloat(backgroundPosition[0]) / 100);
- } else {
- leftPosition = parseInt(backgroundPosition[0], 10);
- }
-
- if (backgroundPosition[1] === 'auto') {
- topPosition = leftPosition / image.width * image.height;
- } else if (isPercentage(backgroundPosition[1])){
- topPosition = (bounds.height - (backgroundSize || image).height) * parseFloat(backgroundPosition[1]) / 100;
- } else {
- topPosition = parseInt(backgroundPosition[1], 10);
- }
-
- if (backgroundPosition[0] === 'auto') {
- leftPosition = topPosition / image.height * image.width;
- }
-
- return {left: leftPosition, top: topPosition};
-};
-
-_html2canvas.Util.BackgroundSize = function(element, bounds, image, imageIndex) {
- var backgroundSize = _html2canvas.Util.getCSS(element, 'backgroundSize', imageIndex), width, height;
-
- if (backgroundSize.length === 1) {
- backgroundSize = [backgroundSize[0], backgroundSize[0]];
- }
-
- if (isPercentage(backgroundSize[0])) {
- width = bounds.width * parseFloat(backgroundSize[0]) / 100;
- } else if (/contain|cover/.test(backgroundSize[0])) {
- return _html2canvas.Util.resizeBounds(image.width, image.height, bounds.width, bounds.height, backgroundSize[0]);
- } else {
- width = parseInt(backgroundSize[0], 10);
- }
-
- if (backgroundSize[0] === 'auto' && backgroundSize[1] === 'auto') {
- height = image.height;
- } else if (backgroundSize[1] === 'auto') {
- height = width / image.width * image.height;
- } else if (isPercentage(backgroundSize[1])) {
- height = bounds.height * parseFloat(backgroundSize[1]) / 100;
- } else {
- height = parseInt(backgroundSize[1], 10);
- }
-
- if (backgroundSize[0] === 'auto') {
- width = height / image.height * image.width;
- }
-
- return {width: width, height: height};
-};
-
-_html2canvas.Util.BackgroundRepeat = function(element, imageIndex) {
- var backgroundRepeat = _html2canvas.Util.getCSS(element, "backgroundRepeat").split(",").map(_html2canvas.Util.trimText);
- return backgroundRepeat[imageIndex] || backgroundRepeat[0];
-};
-
-_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 || 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, cb) {
- 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",
- hidePseudoElementsStyles = doc.createElement('style');
-
- hidePseudoElementsStyles.innerHTML = '.' + pseudoHide +
- '-parent:before { content: "" !important; display: none !important; }' +
- '.' + pseudoHide + '-parent:after { content: "" !important; display: none !important; }';
-
- body.appendChild(hidePseudoElementsStyles);
-
- images = images || {};
-
- init();
-
- function init() {
- var background = getCSS(document.documentElement, "backgroundColor"),
- transparentBackground = (Util.isTransparent(background) && element === document.body),
- stack = renderElement(element, null, false, transparentBackground);
-
- // create pseudo elements in a single pass to prevent synchronous layouts
- addPseudoElements(element);
-
- parseChildren(element, stack, function() {
- if (transparentBackground) {
- background = stack.backgroundColor;
- }
-
- removePseudoElements();
-
- Util.log('Done parsing, moving to Render.');
-
- cb({
- backgroundColor: background,
- stack: stack
- });
- });
- }
-
- // Given a root element, find all pseudo elements below, create elements mocking pseudo element styles
- // so we can process them as normal elements, and hide the original pseudo elements so they don't interfere
- // with layout.
- function addPseudoElements(el) {
- // These are done in discrete steps to prevent a relayout loop caused by addClass() invalidating
- // layouts & getPseudoElement calling getComputedStyle.
- var jobs = [], classes = [];
- getPseudoElementClasses();
- findPseudoElements(el);
- runJobs();
-
- function getPseudoElementClasses(){
- var findPsuedoEls = /:before|:after/;
- var sheets = document.styleSheets;
- for (var i = 0, j = sheets.length; i < j; i++) {
- try {
- var rules = sheets[i].cssRules;
- for (var k = 0, l = rules.length; k < l; k++) {
- if(findPsuedoEls.test(rules[k].selectorText)) {
- classes.push(rules[k].selectorText);
- }
- }
- }
- catch(e) { // will throw security exception for style sheets loaded from external domains
- }
- }
-
- // Trim off the :after and :before (or ::after and ::before)
- for (i = 0, j = classes.length; i < j; i++) {
- classes[i] = classes[i].match(/(^[^:]*)/)[1];
- }
- }
-
- // Using the list of elements we know how pseudo el styles, create fake pseudo elements.
- function findPseudoElements(el) {
- var els = document.querySelectorAll(classes.join(','));
- for(var i = 0, j = els.length; i < j; i++) {
- createPseudoElements(els[i]);
- }
- }
-
- // Create pseudo elements & add them to a job queue.
- function createPseudoElements(el) {
- var before = getPseudoElement(el, ':before'),
- after = getPseudoElement(el, ':after');
-
- if(before) {
- jobs.push({type: 'before', pseudo: before, el: el});
- }
-
- if (after) {
- jobs.push({type: 'after', pseudo: after, el: el});
- }
- }
-
- // Adds a class to the pseudo's parent to prevent the original before/after from messing
- // with layouts.
- // Execute the inserts & addClass() calls in a batch to prevent relayouts.
- function runJobs() {
- // Add Class
- jobs.forEach(function(job){
- addClass(job.el, pseudoHide + "-parent");
- });
-
- // Insert el
- jobs.forEach(function(job){
- if(job.type === 'before'){
- job.el.insertBefore(job.pseudo, job.el.firstChild);
- } else {
- job.el.appendChild(job.pseudo);
- }
- });
- }
- }
-
-
-
- // Delete our fake pseudo elements from the DOM. This will remove those actual elements
- // and the classes on their parents that hide the actual pseudo elements.
- // Note that NodeLists are 'live' collections so you can't use a for loop here. They are
- // actually deleted from the NodeList after each iteration.
- function removePseudoElements(){
- // delete pseudo elements
- body.removeChild(hidePseudoElementsStyles);
- var pseudos = document.getElementsByClassName(pseudoHide + "-element");
- while (pseudos.length) {
- pseudos[0].parentNode.removeChild(pseudos[0]);
- }
-
- // Remove pseudo hiding classes
- var parents = document.getElementsByClassName(pseudoHide + "-parent");
- while(parents.length) {
- removeClass(parents[0], pseudoHide + "-parent");
- }
- }
-
- function addClass (el, className) {
- if (el.classList) {
- el.classList.add(className);
- } else {
- el.className = el.className + " " + className;
- }
- }
-
- function removeClass (el, className) {
- if (el.classList) {
- el.classList.remove(className);
- } else {
- el.className = el.className.replace(className, "").trim();
- }
- }
-
- function hasClass (el, className) {
- return el.className.indexOf(className) > -1;
- }
-
- // Note that this doesn't work in < IE8, but we don't support that anyhow
- function nodeListToArray (nodeList) {
- return Array.prototype.slice.call(nodeList);
- }
-
- 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);
- newContext.depth = parentStack ? (parentStack.zIndex.depth + 1) : 0;
-
- if (parentStack) {
- parentStack.zIndex.children.push(stack);
- }
- }
-
- function h2czContext(zindex) {
- return {
- depth: 0,
- zindex: zindex,
- children: []
- };
- }
-
- 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');
- });
- }
-
- function getCurvePoints(x, y, r1, r2) {
- var kappa = 4 * ((Math.sqrt(2) - 1) / 3);
- 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
- })
- };
- }
-
- 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);
- var parentStyle = window.getComputedStyle(el);
- // If no content attribute is present, the pseudo element is hidden,
- // or the parent has a content property equal to the content on the pseudo element,
- // move along.
- if(!elStyle || !elStyle.content || elStyle.content === "none" || elStyle.content === "-moz-alt-content" ||
- elStyle.display === "none" || parentStyle.content === elStyle.content) {
- return;
- }
- var content = elStyle.content + '';
-
- // Strip inner quotes
- if(content[0] === "'" || content[0] === "\"") {
- content = content.replace(/(^['"])|(['"]$)/g, '');
- }
-
- var isImage = content.substr( 0, 3 ) === 'url',
- elps = document.createElement( isImage ? 'img' : 'span' );
-
- elps.className = pseudoHide + "-element ";
-
- 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 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 = Util.BackgroundRepeat(el, imageIndex);
-
- image = resizeImage(image, backgroundSize);
-
- switch (backgroundRepeat) {
- case "repeat-x":
- case "repeat no-repeat":
- backgroundRepeatShape(ctx, image, backgroundPosition, bounds,
- bounds.left, bounds.top + backgroundPosition.top, 99999, image.height);
- break;
- case "repeat-y":
- case "no-repeat repeat":
- 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", "");
- }
-
- function getTransform(element, parentStack) {
- var transformRegExp = /(matrix)\((.+)\)/;
- 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, 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);
- });
-
- switch(element.nodeName){
- case "IMG":
- if ((image = loadImage(element.getAttribute('src')))) {
- renderImage(ctx, element, image, bounds, borders);
- } else {
- Util.log("html2canvas: Error loading
:" + 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, cb) {
- if (!cb) {
- cb = function(){};
- }
- if (isElementVisible(element)) {
- stack = renderElement(element, stack, false) || stack;
- if (!ignoreElementsRegExp.test(element.nodeName)) {
- return parseChildren(element, stack, cb);
- }
- }
- cb();
- }
-
- function parseChildren(element, stack, cb) {
- var children = Util.Children(element);
- // After all nodes have processed, finished() will call the cb.
- // We add one and kick it off so this will still work when children.length === 0.
- // Note that unless async is true, this will happen synchronously, just will callbacks.
- var jobs = children.length + 1;
- finished();
-
- if (options.async) {
- children.forEach(function(node) {
- // Don't block the page from rendering
- setTimeout(function(){ parseNode(node); }, 0);
- });
- } else {
- children.forEach(parseNode);
- }
-
- function parseNode(node) {
- if (node.nodeType === node.ELEMENT_NODE) {
- parseElement(node, stack, finished);
- } else if (node.nodeType === node.TEXT_NODE) {
- renderText(element, node, stack);
- finished();
- } else {
- finished();
- }
- }
- function finished(el) {
- if (--jobs <= 0){
- Util.log("finished rendering " + children.length + " children.");
- cb();
- }
- }
- }
-};
-_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
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){
- function sortZindex(a, b) {
- if (a === 'children') {
- return -1;
- } else if (b === 'children') {
- return 1;
- } else {
- return a - b;
- }
- }
-
- // 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) {
- contextForChildren = stub.context = {
- children: [{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(sortZindex).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,",
- ""
- ].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,
- async: false, // If true, parsing will not block, but if the user scrolls during parse the image can get weird
-
- // 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;
- }
- }
- _html2canvas.Parse( images, options, function(queue) {
- 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(parsedData.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);
diff --git a/jquery.base64.js b/jquery.base64.js
index 6c98f156..322a0f7b 100644
--- a/jquery.base64.js
+++ b/jquery.base64.js
@@ -1,190 +1,59 @@
-/*jslint adsafe: false, bitwise: true, browser: true, cap: false, css: false,
- debug: false, devel: true, eqeqeq: true, es5: false, evil: false,
- forin: false, fragment: false, immed: true, laxbreak: false, newcap: true,
- nomen: false, on: false, onevar: true, passfail: false, plusplus: true,
- regexp: false, rhino: true, safe: false, strict: false, sub: false,
- undef: true, white: false, widget: false, windows: false */
-/*global jQuery: false, window: false */
-//"use strict";
-
-/*
- * Original code (c) 2010 Nick Galbreath
- * http://code.google.com/p/stringencoders/source/browse/#svn/trunk/javascript
- *
- * jQuery port (c) 2010 Carlo Zottmann
- * http://github.com/carlo/jquery-base64
- *
- * Permission is hereby granted, free of charge, to any person
- * obtaining a copy of this software and associated documentation
- * files (the "Software"), to deal in the Software without
- * restriction, including without limitation the rights to use,
- * copy, modify, merge, publish, distribute, sublicense, and/or sell
- * copies of the Software, and to permit persons to whom the
- * Software is furnished to do so, subject to the following
- * conditions:
- *
- * The above copyright notice and this permission notice shall be
- * included in all copies or substantial portions of the Software.
- *
- * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
- * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
- * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
- * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
- * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
- * OTHER DEALINGS IN THE SOFTWARE.
-*/
-
-/* base64 encode/decode compatible with window.btoa/atob
- *
- * window.atob/btoa is a Firefox extension to convert binary data (the "b")
- * to base64 (ascii, the "a").
- *
- * It is also found in Safari and Chrome. It is not available in IE.
- *
- * if (!window.btoa) window.btoa = $.base64.encode
- * if (!window.atob) window.atob = $.base64.decode
- *
- * The original spec's for atob/btoa are a bit lacking
- * https://developer.mozilla.org/en/DOM/window.atob
- * https://developer.mozilla.org/en/DOM/window.btoa
- *
- * window.btoa and $.base64.encode takes a string where charCodeAt is [0,255]
- * If any character is not [0,255], then an exception is thrown.
- *
- * window.atob and $.base64.decode take a base64-encoded string
- * If the input length is not a multiple of 4, or contains invalid characters
- * then an exception is thrown.
- */
-
-jQuery.base64 = ( function( $ ) {
-
- var _PADCHAR = "=",
- _ALPHA = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",
- _VERSION = "1.0";
-
-
- function _getbyte64( s, i ) {
- // This is oddly fast, except on Chrome/V8.
- // Minimal or no improvement in performance by using a
- // object with properties mapping chars to value (eg. 'A': 0)
-
- var idx = _ALPHA.indexOf( s.charAt( i ) );
-
- if ( idx === -1 ) {
- throw "Cannot decode base64";
- }
-
- return idx;
- }
-
-
- function _decode( s ) {
- var pads = 0,
- i,
- b10,
- imax = s.length,
- x = [];
-
- s = String( s );
-
- if ( imax === 0 ) {
- return s;
- }
-
- if ( imax % 4 !== 0 ) {
- throw "Cannot decode base64";
- }
-
- if ( s.charAt( imax - 1 ) === _PADCHAR ) {
- pads = 1;
-
- if ( s.charAt( imax - 2 ) === _PADCHAR ) {
- pads = 2;
- }
-
- // either way, we want to ignore this last block
- imax -= 4;
- }
-
- for ( i = 0; i < imax; i += 4 ) {
- b10 = ( _getbyte64( s, i ) << 18 ) | ( _getbyte64( s, i + 1 ) << 12 ) | ( _getbyte64( s, i + 2 ) << 6 ) | _getbyte64( s, i + 3 );
- x.push( String.fromCharCode( b10 >> 16, ( b10 >> 8 ) & 0xff, b10 & 0xff ) );
- }
-
- switch ( pads ) {
- case 1:
- b10 = ( _getbyte64( s, i ) << 18 ) | ( _getbyte64( s, i + 1 ) << 12 ) | ( _getbyte64( s, i + 2 ) << 6 );
- x.push( String.fromCharCode( b10 >> 16, ( b10 >> 8 ) & 0xff ) );
- break;
-
- case 2:
- b10 = ( _getbyte64( s, i ) << 18) | ( _getbyte64( s, i + 1 ) << 12 );
- x.push( String.fromCharCode( b10 >> 16 ) );
- break;
- }
-
- return x.join( "" );
- }
-
-
- function _getbyte( s, i ) {
- var x = s.charCodeAt( i );
-
- if ( x > 255 ) {
- throw "INVALID_CHARACTER_ERR: DOM Exception 5";
- }
-
- return x;
- }
-
-
- function _encode( s ) {
- if ( arguments.length !== 1 ) {
- throw "SyntaxError: exactly one argument required";
- }
-
- s = String( s );
-
- var i,
- b10,
- x = [],
- imax = s.length - s.length % 3;
-
- if ( s.length === 0 ) {
- return s;
- }
-
- for ( i = 0; i < imax; i += 3 ) {
- b10 = ( _getbyte( s, i ) << 16 ) | ( _getbyte( s, i + 1 ) << 8 ) | _getbyte( s, i + 2 );
- x.push( _ALPHA.charAt( b10 >> 18 ) );
- x.push( _ALPHA.charAt( ( b10 >> 12 ) & 0x3F ) );
- x.push( _ALPHA.charAt( ( b10 >> 6 ) & 0x3f ) );
- x.push( _ALPHA.charAt( b10 & 0x3f ) );
- }
-
- switch ( s.length - imax ) {
- case 1:
- b10 = _getbyte( s, i ) << 16;
- x.push( _ALPHA.charAt( b10 >> 18 ) + _ALPHA.charAt( ( b10 >> 12 ) & 0x3F ) + _PADCHAR + _PADCHAR );
- break;
-
- case 2:
- b10 = ( _getbyte( s, i ) << 16 ) | ( _getbyte( s, i + 1 ) << 8 );
- x.push( _ALPHA.charAt( b10 >> 18 ) + _ALPHA.charAt( ( b10 >> 12 ) & 0x3F ) + _ALPHA.charAt( ( b10 >> 6 ) & 0x3f ) + _PADCHAR );
- break;
- }
-
- return x.join( "" );
- }
-
-
- return {
- decode: _decode,
- encode: _encode,
- VERSION: _VERSION
- };
-
-}( jQuery ) );
-
+jQuery.base64 = (function($) {
+
+ // private property
+ var keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
+
+ // private method for UTF-8 encoding
+ function utf8Encode(string) {
+ string = string.replace(/\r\n/g,"\n");
+ var utftext = "";
+ for (var n = 0; n < string.length; n++) {
+ var c = string.charCodeAt(n);
+ if (c < 128) {
+ utftext += String.fromCharCode(c);
+ }
+ else if((c > 127) && (c < 2048)) {
+ utftext += String.fromCharCode((c >> 6) | 192);
+ utftext += String.fromCharCode((c & 63) | 128);
+ }
+ else {
+ utftext += String.fromCharCode((c >> 12) | 224);
+ utftext += String.fromCharCode(((c >> 6) & 63) | 128);
+ utftext += String.fromCharCode((c & 63) | 128);
+ }
+ }
+ return utftext;
+ }
+
+ function encode(input) {
+ var output = "";
+ var chr1, chr2, chr3, enc1, enc2, enc3, enc4;
+ var i = 0;
+ input = utf8Encode(input);
+ while (i < input.length) {
+ chr1 = input.charCodeAt(i++);
+ chr2 = input.charCodeAt(i++);
+ chr3 = input.charCodeAt(i++);
+ enc1 = chr1 >> 2;
+ enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
+ enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
+ enc4 = chr3 & 63;
+ if (isNaN(chr2)) {
+ enc3 = enc4 = 64;
+ } else if (isNaN(chr3)) {
+ enc4 = 64;
+ }
+ output = output +
+ keyStr.charAt(enc1) + keyStr.charAt(enc2) +
+ keyStr.charAt(enc3) + keyStr.charAt(enc4);
+ }
+ return output;
+ }
+
+ return {
+ encode: function (str) {
+ return encode(str);
+ }
+ };
+
+}(jQuery));
\ No newline at end of file
diff --git a/jspdf/jspdf.js b/jspdf/jspdf.js
index 2e703c93..ab551d70 100644
--- a/jspdf/jspdf.js
+++ b/jspdf/jspdf.js
@@ -1,303 +1,4 @@
-/**
- * jsPDF
- * (c) 2009 James Hall
- *
- * Some parts based on FPDF.
- */
-
-var jsPDF = function(){
-
- // Private properties
- var version = '20090504';
- var buffer = '';
-
- var pdfVersion = '1.3'; // PDF Version
- var defaultPageFormat = 'a4';
- var pageFormats = { // Size in mm of various paper formats
- 'a3': [841.89, 1190.55],
- 'a4': [595.28, 841.89],
- 'a5': [420.94, 595.28],
- 'letter': [612, 792],
- 'legal': [612, 1008]
- };
- var textColor = '0 g';
- var page = 0;
- var objectNumber = 2; // 'n' Current object number
- var state = 0; // Current document state
- var pages = new Array();
- var offsets = new Array(); // List of offsets
- var lineWidth = 0.200025; // 2mm
- var pageHeight;
- var k; // Scale factor
- var unit = 'mm'; // Default to mm for units
- var fontNumber; // TODO: This is temp, replace with real font handling
- var documentProperties = {};
- var fontSize = 16; // Default font size
- var pageFontSize = 16;
-
- // Initilisation
- if (unit == 'pt') {
- k = 1;
- } else if(unit == 'mm') {
- k = 72/25.4;
- } else if(unit == 'cm') {
- k = 72/2.54;
- } else if(unit == 'in') {
- k = 72;
- }
-
- // Private functions
- var newObject = function() {
- //Begin a new object
- objectNumber ++;
- offsets[objectNumber] = buffer.length;
- out(objectNumber + ' 0 obj');
- }
-
-
- var putHeader = function() {
- out('%PDF-' + pdfVersion);
- }
-
- var putPages = function() {
-
- // TODO: Fix, hardcoded to a4 portrait
- var wPt = pageWidth * k;
- var hPt = pageHeight * k;
-
- for(n=1; n <= page; n++) {
- newObject();
- out('<>');
- out('endobj');
-
- //Page content
- p = pages[n];
- newObject();
- out('<>');
- putStream(p);
- out('endobj');
- }
- offsets[1] = buffer.length;
- out('1 0 obj');
- out('<< page; i++) {
- kids += (3 + 2 * i) + ' 0 R ';
- }
- out(kids + ']');
- out('/Count ' + page);
- out(sprintf('/MediaBox [0 0 %.2f %.2f]', wPt, hPt));
- out('>>');
- out('endobj');
- }
-
- var putStream = function(str) {
- out('stream');
- out(str);
- out('endstream');
- }
-
- var putResources = function() {
- putFonts();
- putImages();
-
- //Resource dictionary
- offsets[2] = buffer.length;
- out('2 0 obj');
- out('<<');
- putResourceDictionary();
- out('>>');
- out('endobj');
- }
-
- var putFonts = function() {
- // TODO: Only supports core font hardcoded to Helvetica
- newObject();
- fontNumber = objectNumber;
- name = 'Helvetica';
- out('<>');
- out('endobj');
- }
-
- var putImages = function() {
- // TODO
- }
-
- var putResourceDictionary = function() {
- out('/ProcSet [/PDF /Text /ImageB /ImageC /ImageI]');
- out('/Font <<');
- // Do this for each font, the '1' bit is the index of the font
- // fontNumber is currently the object number related to 'putFonts'
- out('/F1 ' + fontNumber + ' 0 R');
- out('>>');
- out('/XObject <<');
- putXobjectDict();
- out('>>');
- }
-
- var putXobjectDict = function() {
- // TODO
- // Loop through images
- }
-
-
- var putInfo = function() {
- out('/Producer (jsPDF ' + version + ')');
- if(documentProperties.title != undefined) {
- out('/Title (' + pdfEscape(documentProperties.title) + ')');
- }
- if(documentProperties.subject != undefined) {
- out('/Subject (' + pdfEscape(documentProperties.subject) + ')');
- }
- if(documentProperties.author != undefined) {
- out('/Author (' + pdfEscape(documentProperties.author) + ')');
- }
- if(documentProperties.keywords != undefined) {
- out('/Keywords (' + pdfEscape(documentProperties.keywords) + ')');
- }
- if(documentProperties.creator != undefined) {
- out('/Creator (' + pdfEscape(documentProperties.creator) + ')');
- }
- var created = new Date();
- var year = created.getFullYear();
- var month = (created.getMonth() + 1);
- var day = created.getDate();
- var hour = created.getHours();
- var minute = created.getMinutes();
- var second = created.getSeconds();
- out('/CreationDate (D:' + sprintf('%02d%02d%02d%02d%02d%02d', year, month, day, hour, minute, second) + ')');
- }
-
- var putCatalog = function () {
- out('/Type /Catalog');
- out('/Pages 1 0 R');
- // TODO: Add zoom and layout modes
- out('/OpenAction [3 0 R /FitH null]');
- out('/PageLayout /OneColumn');
- }
-
- function putTrailer() {
- out('/Size ' + (objectNumber + 1));
- out('/Root ' + objectNumber + ' 0 R');
- out('/Info ' + (objectNumber - 1) + ' 0 R');
- }
-
- var endDocument = function() {
- state = 1;
- putHeader();
- putPages();
-
- putResources();
- //Info
- newObject();
- out('<<');
- putInfo();
- out('>>');
- out('endobj');
-
- //Catalog
- newObject();
- out('<<');
- putCatalog();
- out('>>');
- out('endobj');
-
- //Cross-ref
- var o = buffer.length;
- out('xref');
- out('0 ' + (objectNumber + 1));
- out('0000000000 65535 f ');
- for (var i=1; i <= objectNumber; i++) {
- out(sprintf('%010d 00000 n ', offsets[i]));
- }
- //Trailer
- out('trailer');
- out('<<');
- putTrailer();
- out('>>');
- out('startxref');
- out(o);
- out('%%EOF');
- state = 3;
- }
-
- var beginPage = function() {
- page ++;
- // Do dimension stuff
- state = 2;
- pages[page] = '';
-
- // TODO: Hardcoded at A4 and portrait
- pageHeight = pageFormats['a4'][1] / k;
- pageWidth = pageFormats['a4'][0] / k;
- }
-
- var out = function(string) {
- if(state == 2) {
- pages[page] += string + '\n';
- } else {
- buffer += string + '\n';
- }
- }
-
- var _addPage = function() {
- beginPage();
- // Set line width
- out(sprintf('%.2f w', (lineWidth * k)));
-
- // Set font - TODO
- // 16 is the font size
- pageFontSize = fontSize;
- out('BT /F1 ' + parseInt(fontSize) + '.00 Tf ET');
- }
-
- // Add the first page automatically
- _addPage();
-
- // Escape text
- var pdfEscape = function(text) {
- return text.replace(/\\/g, '\\\\').replace(/\(/g, '\\(').replace(/\)/g, '\\)');
- }
-
- return {
- addPage: function() {
- _addPage();
- },
- text: function(x, y, text) {
- // need page height
- if(pageFontSize != fontSize) {
- out('BT /F1 ' + parseInt(fontSize) + '.00 Tf ET');
- pageFontSize = fontSize;
- }
- var str = sprintf('BT %.2f %.2f Td (%s) Tj ET', x * k, (pageHeight - y) * k, pdfEscape(text));
- out(str);
- },
- setProperties: function(properties) {
- documentProperties = properties;
- },
- addImage: function(imageData, format, x, y, w, h) {
-
- },
- output: function(type, options) {
- endDocument();
- if(type == undefined) {
- return buffer;
- }
- if(type == 'datauri') {
- document.location.href = 'data:application/pdf;base64,' + Base64.encode(buffer);
- }
- // @TODO: Add different output options
- },
- setFontSize: function(size) {
- fontSize = size;
- }
- }
-
-};
+!function(t,e){e["true"]=t;var n=function(t){"use strict";function e(e){var n={};this.subscribe=function(t,e,r){if("function"!=typeof e)return!1;n.hasOwnProperty(t)||(n[t]={});var s=Math.random().toString(35);return n[t][s]=[e,!!r],s},this.unsubscribe=function(t){for(var e in n)if(n[e][t])return delete n[e][t],!0;return!1},this.publish=function(r){if(n.hasOwnProperty(r)){var s=Array.prototype.slice.call(arguments,1),i=[];for(var o in n[r]){var a=n[r][o];try{a[0].apply(e,s)}catch(u){t.console&&console.error("jsPDF PubSub Error",u.message,u)}a[1]&&i.push(o)}i.length&&i.forEach(this.unsubscribe)}}}function n(a,u,c,l){var f={};"object"==typeof a&&(f=a,a=f.orientation,u=f.unit||u,c=f.format||c,l=f.compress||f.compressPdf||l),u=u||"mm",c=c||"a4",a=(""+(a||"P")).toLowerCase();var d,h,p,m,w,g,y,v,b,q=((""+c).toLowerCase(),!!l&&"function"==typeof Uint8Array),x=f.textColor||"0 g",k=f.drawColor||"0 G",_=f.fontSize||16,A=f.lineHeight||1.15,C=f.lineWidth||.200025,S=2,E=!1,z=[],T={},I={},B=0,O=[],P={},R=[],F=0,D=0,U=0,N={title:"",subject:"",author:"",keywords:"",creator:""},L={},j=new e(L),M=function(t){return t.toFixed(2)},H=function(t){return t.toFixed(3)},G=function(t){return("0"+parseInt(t)).slice(-2)},W=function(t){E?O[m].push(t):(U+=t.length+1,R.push(t))},V=function(){return S++,z[S]=U,W(S+" 0 obj"),S},J=function(t){W("stream"),W(t),W("endstream")},X=function(){var e,r,i,o,a,u,c,l,f;for(c=t.adler32cs||n.adler32cs,q&&"undefined"==typeof c&&(q=!1),e=1;B>=e;e++){if(V(),l=(w=P[e].width)*h,f=(g=P[e].height)*h,W("<>"),W("endobj"),r=O[e].join("\n"),V(),q){for(i=[],o=r.length;o--;)i[o]=r.charCodeAt(o);u=c.from(r),a=new s(6),a.append(new Uint8Array(i)),r=a.flush(),i=new Uint8Array(r.length+6),i.set(new Uint8Array([120,156])),i.set(r,2),i.set(new Uint8Array([255&u,u>>8&255,u>>16&255,u>>24&255]),r.length+2),r=String.fromCharCode.apply(null,i),W("<>")}else W("<>");J(r),W("endobj")}z[1]=U,W("1 0 obj"),W("<o;o++)d+=3+2*o+" 0 R ";W(d+"]"),W("/Count "+B),W(">>"),W("endobj")},Y=function(t){t.objectNumber=V(),W("<>"),W("endobj")},K=function(){for(var t in T)T.hasOwnProperty(t)&&Y(T[t])},Q=function(){j.publish("putXobjectDict")},$=function(){W("/ProcSet [/PDF /Text /ImageB /ImageC /ImageI]"),W("/Font <<");for(var t in T)T.hasOwnProperty(t)&&W("/"+t+" "+T[t].objectNumber+" 0 R");W(">>"),W("/XObject <<"),Q(),W(">>")},Z=function(){K(),j.publish("putResources"),z[2]=U,W("2 0 obj"),W("<<"),$(),W(">>"),W("endobj"),j.publish("postPutResources")},te=function(t,e,n){I.hasOwnProperty(e)||(I[e]={}),I[e][n]=t},ee=function(t,e,n,r){var s="F"+(Object.keys(T).length+1).toString(10),i=T[s]={id:s,PostScriptName:t,fontName:e,fontStyle:n,encoding:r,metadata:{}};return te(s,e,n),j.publish("addFont",i),s},ne=function(){for(var t="helvetica",e="times",n="courier",r="normal",s="bold",i="italic",o="bolditalic",a="StandardEncoding",u=[["Helvetica",t,r],["Helvetica-Bold",t,s],["Helvetica-Oblique",t,i],["Helvetica-BoldOblique",t,o],["Courier",n,r],["Courier-Bold",n,s],["Courier-Oblique",n,i],["Courier-BoldOblique",n,o],["Times-Roman",e,r],["Times-Bold",e,s],["Times-Italic",e,i],["Times-BoldItalic",e,o]],c=0,l=u.length;l>c;c++){var f=ee(u[c][0],u[c][1],u[c][2],a),d=u[c][0].split("-");te(f,d[0],d[1]||"")}j.publish("addFonts",{fonts:T,dictionary:I})},re=function(e){return e.foo=function(){try{return e.apply(this,arguments)}catch(n){var r=n.stack||"";~r.indexOf(" at ")&&(r=r.split(" at ")[1]);var s="Error in function "+r.split("\n")[0].split("<")[0]+": "+n.message;if(!t.console)throw new Error(s);t.console.error(s,n),t.alert&&alert(s)}},e.foo.bar=e,e.foo},se=function(t,e){var n,r,s,i,o,a,u,c,l;if(e=e||{},s=e.sourceEncoding||"Unicode",o=e.outputEncoding,(e.autoencode||o)&&T[d].metadata&&T[d].metadata[s]&&T[d].metadata[s].encoding&&(i=T[d].metadata[s].encoding,!o&&T[d].encoding&&(o=T[d].encoding),!o&&i.codePages&&(o=i.codePages[0]),"string"==typeof o&&(o=i[o]),o)){for(u=!1,a=[],n=0,r=t.length;r>n;n++)c=o[t.charCodeAt(n)],a.push(c?String.fromCharCode(c):t[n]),a[n].charCodeAt(0)>>8&&(u=!0);t=a.join("")}for(n=t.length;void 0===u&&0!==n;)t.charCodeAt(n-1)>>8&&(u=!0),n--;if(!u)return t;for(a=e.noBOM?[]:[254,255],n=0,r=t.length;r>n;n++){if(c=t.charCodeAt(n),l=c>>8,l>>8)throw new Error("Character at position "+n+" of string '"+t+"' exceeds 16bits. Cannot be encoded into UCS-2 BE");a.push(l),a.push(c-(l<<8))}return String.fromCharCode.apply(void 0,a)},ie=function(t,e){return se(t,e).replace(/\\/g,"\\\\").replace(/\(/g,"\\(").replace(/\)/g,"\\)")},oe=function(){W("/Producer (jsPDF "+n.version+")");for(var t in N)N.hasOwnProperty(t)&&N[t]&&W("/"+t.substr(0,1).toUpperCase()+t.substr(1)+" ("+ie(N[t])+")");var e=new Date,r=e.getTimezoneOffset(),s=0>r?"+":"-",i=Math.floor(Math.abs(r/60)),o=Math.abs(r%60),a=[s,G(i),"'",G(o),"'"].join("");W(["/CreationDate (D:",e.getFullYear(),G(e.getMonth()+1),G(e.getDate()),G(e.getHours()),G(e.getMinutes()),G(e.getSeconds()),a,")"].join(""))},ae=function(){switch(W("/Type /Catalog"),W("/Pages 1 0 R"),v||(v="fullwidth"),v){case"fullwidth":W("/OpenAction [3 0 R /FitH null]");break;case"fullheight":W("/OpenAction [3 0 R /FitV null]");break;case"fullpage":W("/OpenAction [3 0 R /Fit]");break;case"original":W("/OpenAction [3 0 R /XYZ null null 1]");break;default:var t=""+v;"%"===t.substr(t.length-1)&&(v=parseInt(v)/100),"number"==typeof v&&W("/OpenAction [3 0 R /XYZ null null "+M(v)+"]")}switch(b||(b="continuous"),b){case"continuous":W("/PageLayout /OneColumn");break;case"single":W("/PageLayout /SinglePage");break;case"two":case"twoleft":W("/PageLayout /TwoColumnLeft");break;case"tworight":W("/PageLayout /TwoColumnRight")}y&&W("/PageMode /"+y),j.publish("putCatalog")},ue=function(){W("/Size "+(S+1)),W("/Root "+S+" 0 R"),W("/Info "+(S-1)+" 0 R")},ce=function(t,e){var n="string"==typeof e&&e.toLowerCase();if("string"==typeof t){var r=t.toLowerCase();o.hasOwnProperty(r)&&(t=o[r][0]/h,e=o[r][1]/h)}if(Array.isArray(t)&&(e=t[1],t=t[0]),n){switch(n.substr(0,1)){case"l":e>t&&(n="s");break;case"p":t>e&&(n="s")}"s"===n&&(p=t,t=e,e=p)}E=!0,O[++B]=[],P[B]={width:Number(t)||w,height:Number(e)||g},fe(B)},le=function(){ce.apply(this,arguments),W(M(C*h)+" w"),W(k),0!==F&&W(F+" J"),0!==D&&W(D+" j"),j.publish("addPage",{pageNumber:B})},fe=function(t){t>0&&B>=t&&(m=t,w=P[t].width,g=P[t].height)},de=function(t,e){var n;t=void 0!==t?t:T[d].fontName,e=void 0!==e?e:T[d].fontStyle;try{n=I[t][e]}catch(r){}if(!n)throw new Error("Unable to look up font label for font '"+t+"', '"+e+"'. Refer to getFontList() for available fonts.");return n},he=function(){E=!1,S=2,R=[],z=[],W("%PDF-"+i),X(),Z(),V(),W("<<"),oe(),W(">>"),W("endobj"),V(),W("<<"),ae(),W(">>"),W("endobj");var t,e=U,n="0000000000";for(W("xref"),W("0 "+(S+1)),W(n+" 65535 f "),t=1;S>=t;t++)W((n+z[t]).slice(-10)+" 00000 n ");return W("trailer"),W("<<"),ue(),W(">>"),W("startxref"),W(e),W("%%EOF"),E=!0,R.join("\n")},pe=function(t){var e="S";return"F"===t?e="f":"FD"===t||"DF"===t?e="B":("f"===t||"f*"===t||"B"===t||"B*"===t)&&(e=t),e},me=function(){for(var t=he(),e=t.length,n=new ArrayBuffer(e),r=new Uint8Array(n);e--;)r[e]=t.charCodeAt(e);return n},we=function(){return new Blob([me()],{type:"application/pdf"})},ge=re(function(e,n){var s="dataur"===(""+e).substr(0,6)?"data:application/pdf;base64,"+btoa(he()):0;switch(e){case void 0:return he();case"save":if(navigator.getUserMedia&&(void 0===t.URL||void 0===t.URL.createObjectURL))return L.output("dataurlnewwindow");r(we(),n),"function"==typeof r.unload&&t.setTimeout&&setTimeout(r.unload,911);break;case"arraybuffer":return me();case"blob":return we();case"bloburi":case"bloburl":return t.URL&&t.URL.createObjectURL(we())||void 0;case"datauristring":case"dataurlstring":return s;case"dataurlnewwindow":var i=t.open(s);if(i||"undefined"==typeof safari)return i;case"datauri":case"dataurl":return t.document.location.href=s;default:throw new Error('Output type "'+e+'" is not supported.')}});switch(u){case"pt":h=1;break;case"mm":h=72/25.4;break;case"cm":h=72/2.54;break;case"in":h=72;break;case"px":h=96/72;break;case"pc":h=12;break;case"em":h=12;break;case"ex":h=6;break;default:throw"Invalid unit: "+u}L.internal={pdfEscape:ie,getStyle:pe,getFont:function(){return T[de.apply(L,arguments)]},getFontSize:function(){return _},getLineHeight:function(){return _*A},write:function(t){W(1===arguments.length?t:Array.prototype.join.call(arguments," "))},getCoordinateString:function(t){return M(t*h)},getVerticalCoordinateString:function(t){return M((g-t)*h)},collections:{},newObject:V,putStream:J,events:j,scaleFactor:h,pageSize:{get width(){return w},get height(){return g}},output:function(t,e){return ge(t,e)},getNumberOfPages:function(){return O.length-1},pages:O},L.addPage=function(){return le.apply(this,arguments),this},L.setPage=function(){return fe.apply(this,arguments),this},L.setDisplayMode=function(t,e,n){return v=t,b=e,y=n,this},L.text=function(t,e,n,r,s){function i(t){return t=t.split(" ").join(Array(f.TabLen||9).join(" ")),ie(t,r)}"number"==typeof t&&(p=n,n=e,e=t,t=p),"string"==typeof t&&t.match(/[\n\r]/)&&(t=t.split(/\r\n|\r|\n/g)),"number"==typeof r&&(s=r,r=null);var o,a="",u="Td";if(s){s*=Math.PI/180;var c=Math.cos(s),l=Math.sin(s);a=[M(c),M(l),M(-1*l),M(c),""].join(" "),u="Tm"}if(r=r||{},"noBOM"in r||(r.noBOM=!0),"autoencode"in r||(r.autoencode=!0),"string"==typeof t)t=i(t);else{if(!(t instanceof Array))throw new Error('Type of text must be string or Array. "'+t+'" is not recognized.');for(var m=t.concat(),w=[],y=m.length;y--;)w.push(i(m.shift()));var v=Math.ceil((g-n)*h/(_*A));v>=0&&vs;s++,e+=r)this.text(t[s],e,n)},L.line=function(t,e,n,r){return this.lines([[n-t,r-e]],t,e)},L.clip=function(){W("W"),W("S")},L.lines=function(t,e,n,r,s,i){var o,a,u,c,l,f,d,m,w,y,v;for("number"==typeof t&&(p=n,n=e,e=t,t=p),r=r||[1,1],W(H(e*h)+" "+H((g-n)*h)+" m "),o=r[0],a=r[1],c=t.length,y=e,v=n,u=0;c>u;u++)l=t[u],2===l.length?(y=l[0]*o+y,v=l[1]*a+v,W(H(y*h)+" "+H((g-v)*h)+" l")):(f=l[0]*o+y,d=l[1]*a+v,m=l[2]*o+y,w=l[3]*a+v,y=l[4]*o+y,v=l[5]*a+v,W(H(f*h)+" "+H((g-d)*h)+" "+H(m*h)+" "+H((g-w)*h)+" "+H(y*h)+" "+H((g-v)*h)+" c"));return i&&W(" h"),null!==s&&W(pe(s)),this},L.rect=function(t,e,n,r,s){pe(s);return W([M(t*h),M((g-e)*h),M(n*h),M(-r*h),"re"].join(" ")),null!==s&&W(pe(s)),this},L.triangle=function(t,e,n,r,s,i,o){return this.lines([[n-t,r-e],[s-n,i-r],[t-s,e-i]],t,e,[1,1],o,!0),this},L.roundedRect=function(t,e,n,r,s,i,o){var a=4/3*(Math.SQRT2-1);return this.lines([[n-2*s,0],[s*a,0,s,i-i*a,s,i],[0,r-2*i],[0,i*a,-(s*a),i,-s,i],[-n+2*s,0],[-(s*a),0,-s,-(i*a),-s,-i],[0,-r+2*i],[0,-(i*a),s*a,-i,s,-i]],t+s,e,[1,1],o),this},L.ellipse=function(t,e,n,r,s){var i=4/3*(Math.SQRT2-1)*n,o=4/3*(Math.SQRT2-1)*r;return W([M((t+n)*h),M((g-e)*h),"m",M((t+n)*h),M((g-(e-o))*h),M((t+i)*h),M((g-(e-r))*h),M(t*h),M((g-(e-r))*h),"c"].join(" ")),W([M((t-i)*h),M((g-(e-r))*h),M((t-n)*h),M((g-(e-o))*h),M((t-n)*h),M((g-e)*h),"c"].join(" ")),W([M((t-n)*h),M((g-(e+o))*h),M((t-i)*h),M((g-(e+r))*h),M(t*h),M((g-(e+r))*h),"c"].join(" ")),W([M((t+i)*h),M((g-(e+r))*h),M((t+n)*h),M((g-(e+o))*h),M((t+n)*h),M((g-e)*h),"c"].join(" ")),null!==s&&W(pe(s)),this},L.circle=function(t,e,n,r){return this.ellipse(t,e,n,n,r)},L.setProperties=function(t){for(var e in N)N.hasOwnProperty(e)&&t[e]&&(N[e]=t[e]);return this},L.setFontSize=function(t){return _=t,this},L.setFont=function(t,e){return d=de(t,e),this},L.setFontStyle=L.setFontType=function(t){return d=de(void 0,t),this},L.getFontList=function(){var t,e,n,r={};for(t in I)if(I.hasOwnProperty(t)){r[t]=n=[];for(e in I[t])I[t].hasOwnProperty(e)&&n.push(e)}return r},L.setLineWidth=function(t){return W((t*h).toFixed(2)+" w"),this},L.setDrawColor=function(t,e,n,r){var s;return s=void 0===e||void 0===r&&t===e===n?"string"==typeof t?t+" G":M(t/255)+" G":void 0===r?"string"==typeof t?[t,e,n,"RG"].join(" "):[M(t/255),M(e/255),M(n/255),"RG"].join(" "):"string"==typeof t?[t,e,n,r,"K"].join(" "):[M(t),M(e),M(n),M(r),"K"].join(" "),W(s),this},L.setFillColor=function(t,e,n,r){var s;return s=void 0===e||void 0===r&&t===e===n?"string"==typeof t?t+" g":M(t/255)+" g":void 0===r?"string"==typeof t?[t,e,n,"rg"].join(" "):[M(t/255),M(e/255),M(n/255),"rg"].join(" "):"string"==typeof t?[t,e,n,r,"k"].join(" "):[M(t),M(e),M(n),M(r),"k"].join(" "),W(s),this},L.setTextColor=function(t,e,n){if("string"==typeof t&&/^#[0-9A-Fa-f]{6}$/.test(t)){var r=parseInt(t.substr(1),16);t=r>>16&255,e=r>>8&255,n=255&r}return x=0===t&&0===e&&0===n||"undefined"==typeof e?H(t/255)+" g":[H(t/255),H(e/255),H(n/255),"rg"].join(" "),this},L.CapJoinStyles={0:0,butt:0,but:0,miter:0,1:1,round:1,rounded:1,circle:1,2:2,projecting:2,project:2,square:2,bevel:2},L.setLineCap=function(t){var e=this.CapJoinStyles[t];if(void 0===e)throw new Error("Line cap style of '"+t+"' is not recognized. See or extend .CapJoinStyles property for valid styles");return F=e,W(e+" J"),this},L.setLineJoin=function(t){var e=this.CapJoinStyles[t];if(void 0===e)throw new Error("Line join style of '"+t+"' is not recognized. See or extend .CapJoinStyles property for valid styles");return D=e,W(e+" j"),this},L.output=ge,L.save=function(t){L.output("save",t)};for(var ye in n.API)n.API.hasOwnProperty(ye)&&("events"===ye&&n.API.events.length?!function(t,e){var n,r,s;for(s=e.length-1;-1!==s;s--)n=e[s][0],r=e[s][1],t.subscribe.apply(t,[n].concat("function"==typeof r?[r]:r))}(j,n.API.events):L[ye]=n.API[ye]);return ne(),d="F1",le(c,a),j.publish("initialized"),L}var i="1.3",o={a0:[2383.94,3370.39],a1:[1683.78,2383.94],a2:[1190.55,1683.78],a3:[841.89,1190.55],a4:[595.28,841.89],a5:[419.53,595.28],a6:[297.64,419.53],a7:[209.76,297.64],a8:[147.4,209.76],a9:[104.88,147.4],a10:[73.7,104.88],b0:[2834.65,4008.19],b1:[2004.09,2834.65],b2:[1417.32,2004.09],b3:[1000.63,1417.32],b4:[708.66,1000.63],b5:[498.9,708.66],b6:[354.33,498.9],b7:[249.45,354.33],b8:[175.75,249.45],b9:[124.72,175.75],b10:[87.87,124.72],c0:[2599.37,3676.54],c1:[1836.85,2599.37],c2:[1298.27,1836.85],c3:[918.43,1298.27],c4:[649.13,918.43],c5:[459.21,649.13],c6:[323.15,459.21],c7:[229.61,323.15],c8:[161.57,229.61],c9:[113.39,161.57],c10:[79.37,113.39],dl:[311.81,623.62],letter:[612,792],"government-letter":[576,756],legal:[612,1008],"junior-legal":[576,360],ledger:[1224,792],tabloid:[792,1224],"credit-card":[153,243]};return n.API={events:[]},n.version="1.0.272-git 2014-09-29T15:09:diegocr","function"==typeof define&&define.amd?define("jsPDF",function(){return n}):t.jsPDF=n,n}("undefined"!=typeof self&&self||"undefined"!=typeof window&&window||this);!function(t){"use strict";t.addHTML=function(t,e,n,r,s){if("undefined"==typeof html2canvas&&"undefined"==typeof rasterizeHTML)throw new Error("You need either https://github.com/niklasvh/html2canvas or https://github.com/cburgmer/rasterizeHTML.js");"number"!=typeof e&&(r=e,s=n),"function"==typeof r&&(s=r,r=null);var i=this.internal,o=i.scaleFactor,a=i.pageSize.width,u=i.pageSize.height;if(r=r||{},r.onrendered=function(t){e=parseInt(e)||0,n=parseInt(n)||0;var i=r.dim||{},c=i.h||0,l=i.w||Math.min(a,t.width/o)-e,f="JPEG";if(r.format&&(f=r.format),t.height>u&&r.pagesplit){var d=function(){for(var r=0;;){var i=document.createElement("canvas");i.width=Math.min(a*o,t.width),i.height=Math.min(u*o,t.height-r);var c=i.getContext("2d");c.drawImage(t,0,r,t.width,i.height,0,0,i.width,i.height);var d=[i,e,r?0:n,i.width/o,i.height/o,f,null,"SLOW"];if(this.addImage.apply(this,d),r+=i.height,r>=t.height)break;this.addPage()}s(l,r,null,d)}.bind(this);if("CANVAS"===t.nodeName){var h=new Image;h.onload=d,h.src=t.toDataURL("image/png"),t=h}else d()}else{var p=Math.random().toString(35),m=[t,e,n,l,c,f,p,"SLOW"];this.addImage.apply(this,m),s(l,c,p,m)}}.bind(this),"undefined"!=typeof html2canvas&&!r.rstz)return html2canvas(t,r);if("undefined"!=typeof rasterizeHTML){var c="drawDocument";return"string"==typeof t&&(c=/^http/.test(t)?"drawURL":"drawHTML"),r.width=r.width||a*o,rasterizeHTML[c](t,void 0,r).then(function(t){r.onrendered(t.image)},function(t){s(null,t)})}return null}}(n.API),function(t){"use strict";var e="addImage_",n=["jpeg","jpg","png"],r=function(t){var e=this.internal.newObject(),n=this.internal.write,s=this.internal.putStream;if(t.n=e,n("<<<"+t.dp+">>"),"trns"in t&&t.trns.constructor==Array){for(var i="",o=0,a=t.trns.length;a>o;o++)i+=t.trns[o]+" "+t.trns[o]+" ";n("/Mask ["+i+"]")}if("smask"in t&&n("/SMask "+(e+1)+" 0 R"),n("/Length "+t.data.length+">>"),s(t.data),n("endobj"),"smask"in t){var u="/Predictor 15 /Colors 1 /BitsPerComponent "+t.bpc+" /Columns "+t.w,c={w:t.w,h:t.h,cs:"DeviceGray",bpc:t.bpc,dp:u,data:t.smask};"f"in t&&(c.f=t.f),r.call(this,c)}t.cs===this.color_spaces.INDEXED&&(this.internal.newObject(),n("<< /Length "+t.pal.length+">>"),s(this.arrayBufferToBinaryString(new Uint8Array(t.pal))),n("endobj"))},s=function(){var t=this.internal.collections[e+"images"];for(var n in t)r.call(this,t[n])},i=function(){var t,n=this.internal.collections[e+"images"],r=this.internal.write;for(var s in n)t=n[s],r("/I"+t.i,t.n,"0","R")},o=function(e){return e&&"string"==typeof e&&(e=e.toUpperCase()),e in t.image_compression?e:t.image_compression.NONE},a=function(){var t=this.internal.collections[e+"images"];return t||(this.internal.collections[e+"images"]=t={},this.internal.events.subscribe("putResources",s),this.internal.events.subscribe("putXobjectDict",i)),t},u=function(t){var e=0;return t&&(e=Object.keys?Object.keys(t).length:function(t){var e=0;for(var n in t)t.hasOwnProperty(n)&&e++;return e}(t)),e},c=function(t){return"undefined"==typeof t||null===t},l=function(e){return"string"==typeof e&&t.sHashCode(e)},f=function(t){return-1===n.indexOf(t)},d=function(e){return"function"!=typeof t["process"+e.toUpperCase()]},h=function(t){return"object"==typeof t&&1===t.nodeType},p=function(t,e,n){if("IMG"===t.nodeName&&t.hasAttribute("src")){var r=""+t.getAttribute("src");if(!n&&0===r.indexOf("data:image/"))return r;!e&&/\.png(?:[?#].*)?$/i.test(r)&&(e="png")}if("CANVAS"===t.nodeName)var s=t;else{var s=document.createElement("canvas");s.width=t.clientWidth||t.width,s.height=t.clientHeight||t.height;var i=s.getContext("2d");if(!i)throw"addImage requires canvas to be supported by browser.";if(n){var o,a,u,c,l,f,d,h,p=Math.PI/180;"object"==typeof n&&(o=n.x,a=n.y,u=n.bg,n=n.angle),h=n*p,c=Math.abs(Math.cos(h)),l=Math.abs(Math.sin(h)),f=s.width,d=s.height,s.width=d*l+f*c,s.height=d*c+f*l,isNaN(o)&&(o=s.width/2),isNaN(a)&&(a=s.height/2),i.clearRect(0,0,s.width,s.height),i.fillStyle=u||"white",i.fillRect(0,0,s.width,s.height),i.save(),i.translate(o,a),i.rotate(h),i.drawImage(t,-(f/2),-(d/2)),i.rotate(-h),i.translate(-o,-a),i.restore()}else i.drawImage(t,0,0,s.width,s.height)}return s.toDataURL("png"==(""+e).toLowerCase()?"image/png":"image/jpeg")},m=function(t,e){var n;if(e)for(var r in e)if(t===e[r].alias){n=e[r];break}return n},w=function(t,e,n){return t||e||(t=-96,e=-96),0>t&&(t=-1*n.w*72/t/this.internal.scaleFactor),0>e&&(e=-1*n.h*72/e/this.internal.scaleFactor),0===t&&(t=e*n.w/n.h),0===e&&(e=t*n.h/n.w),[t,e]},g=function(t,e,n,r,s,i,o){var a=w.call(this,n,r,s),u=this.internal.getCoordinateString,c=this.internal.getVerticalCoordinateString;n=a[0],r=a[1],o[i]=s,this.internal.write("q",u(n),"0 0",u(r),u(t),c(e+r),"cm /I"+s.i,"Do Q")};t.color_spaces={DEVICE_RGB:"DeviceRGB",DEVICE_GRAY:"DeviceGray",DEVICE_CMYK:"DeviceCMYK",CAL_GREY:"CalGray",CAL_RGB:"CalRGB",LAB:"Lab",ICC_BASED:"ICCBased",INDEXED:"Indexed",PATTERN:"Pattern",SEPERATION:"Seperation",DEVICE_N:"DeviceN"},t.decode={DCT_DECODE:"DCTDecode",FLATE_DECODE:"FlateDecode",LZW_DECODE:"LZWDecode",JPX_DECODE:"JPXDecode",JBIG2_DECODE:"JBIG2Decode",ASCII85_DECODE:"ASCII85Decode",ASCII_HEX_DECODE:"ASCIIHexDecode",RUN_LENGTH_DECODE:"RunLengthDecode",CCITT_FAX_DECODE:"CCITTFaxDecode"},t.image_compression={NONE:"NONE",FAST:"FAST",MEDIUM:"MEDIUM",SLOW:"SLOW"},t.sHashCode=function(t){return Array.prototype.reduce&&t.split("").reduce(function(t,e){return t=(t<<5)-t+e.charCodeAt(0),t&t},0)},t.isString=function(t){return"string"==typeof t},t.extractInfoFromBase64DataURI=function(t){return/^data:([\w]+?\/([\w]+?));base64,(.+?)$/g.exec(t)},t.supportsArrayBuffer=function(){return"undefined"!=typeof ArrayBuffer&&"undefined"!=typeof Uint8Array},t.isArrayBuffer=function(t){return this.supportsArrayBuffer()?t instanceof ArrayBuffer:!1},t.isArrayBufferView=function(t){return this.supportsArrayBuffer()?"undefined"==typeof Uint32Array?!1:t instanceof Int8Array||t instanceof Uint8Array||"undefined"!=typeof Uint8ClampedArray&&t instanceof Uint8ClampedArray||t instanceof Int16Array||t instanceof Uint16Array||t instanceof Int32Array||t instanceof Uint32Array||t instanceof Float32Array||t instanceof Float64Array:!1},t.binaryStringToUint8Array=function(t){for(var e=t.length,n=new Uint8Array(e),r=0;e>r;r++)n[r]=t.charCodeAt(r);return n},t.arrayBufferToBinaryString=function(t){this.isArrayBuffer(t)&&(t=new Uint8Array(t));for(var e="",n=t.byteLength,r=0;n>r;r++)e+=String.fromCharCode(t[r]);return e},t.arrayBufferToBase64=function(t){for(var e,n,r,s,i,o="",a="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",u=new Uint8Array(t),c=u.byteLength,l=c%3,f=c-l,d=0;f>d;d+=3)i=u[d]<<16|u[d+1]<<8|u[d+2],e=(16515072&i)>>18,n=(258048&i)>>12,r=(4032&i)>>6,s=63&i,o+=a[e]+a[n]+a[r]+a[s];return 1==l?(i=u[f],e=(252&i)>>2,n=(3&i)<<4,o+=a[e]+a[n]+"=="):2==l&&(i=u[f]<<8|u[f+1],e=(64512&i)>>10,n=(1008&i)>>4,r=(15&i)<<2,o+=a[e]+a[n]+a[r]+"="),o},t.createImageInfo=function(t,e,n,r,s,i,o,a,u,c,l,f){var d={alias:a,w:e,h:n,cs:r,bpc:s,i:o,data:t};return i&&(d.f=i),u&&(d.dp=u),c&&(d.trns=c),l&&(d.pal=l),f&&(d.smask=f),d},t.addImage=function(t,e,r,s,i,w,y,v,b){if("string"!=typeof e){var q=w;w=i,i=s,s=r,r=e,e=q}if("object"==typeof t&&!h(t)&&"imageData"in t){var x=t;t=x.imageData,e=x.format||e,r=x.x||r||0,s=x.y||s||0,i=x.w||i,w=x.h||w,y=x.alias||y,v=x.compression||v,b=x.rotation||x.angle||b}if(isNaN(r)||isNaN(s))throw console.error("jsPDF.addImage: Invalid coordinates",arguments),new Error("Invalid coordinates passed to jsPDF.addImage");var k,_=a.call(this);if(!(k=m(t,_))){var A;if(h(t)&&(t=p(t,e,b)),c(y)&&(y=l(t)),!(k=m(y,_))){if(this.isString(t)){var C=this.extractInfoFromBase64DataURI(t);C?(e=C[2],t=atob(C[3])):137===t.charCodeAt(0)&&80===t.charCodeAt(1)&&78===t.charCodeAt(2)&&71===t.charCodeAt(3)&&(e="png")}if(e=(e||"JPEG").toLowerCase(),f(e))throw new Error("addImage currently only supports formats "+n+", not '"+e+"'");if(d(e))throw new Error("please ensure that the plugin for '"+e+"' support is added");if(this.supportsArrayBuffer()&&(A=t,t=this.binaryStringToUint8Array(t)),k=this["process"+e.toUpperCase()](t,u(_),y,o(v),A),!k)throw new Error("An unkwown error occurred whilst processing the image")}}return g.call(this,r,s,i,w,k,k.i,_),this};var y=function(t){var e,n,r;if(255===!t.charCodeAt(0)||216===!t.charCodeAt(1)||255===!t.charCodeAt(2)||224===!t.charCodeAt(3)||!t.charCodeAt(6)==="J".charCodeAt(0)||!t.charCodeAt(7)==="F".charCodeAt(0)||!t.charCodeAt(8)==="I".charCodeAt(0)||!t.charCodeAt(9)==="F".charCodeAt(0)||0===!t.charCodeAt(10))throw new Error("getJpegSize requires a binary string jpeg file");for(var s=256*t.charCodeAt(4)+t.charCodeAt(5),i=4,o=t.length;o>i;){if(i+=s,255!==t.charCodeAt(i))throw new Error("getJpegSize could not find the size of the image");if(192===t.charCodeAt(i+1)||193===t.charCodeAt(i+1)||194===t.charCodeAt(i+1)||195===t.charCodeAt(i+1)||196===t.charCodeAt(i+1)||197===t.charCodeAt(i+1)||198===t.charCodeAt(i+1)||199===t.charCodeAt(i+1))return n=256*t.charCodeAt(i+5)+t.charCodeAt(i+6),e=256*t.charCodeAt(i+7)+t.charCodeAt(i+8),r=t.charCodeAt(i+9),[e,n,r];i+=2,s=256*t.charCodeAt(i)+t.charCodeAt(i+1)}},v=function(t){var e=t[0]<<8|t[1];if(65496!==e)throw new Error("Supplied data is not a JPEG");for(var n,r,s,i,o=t.length,a=(t[4]<<8)+t[5],u=4;o>u;){if(u+=a,n=b(t,u),a=(n[2]<<8)+n[3],(192===n[1]||194===n[1])&&255===n[0]&&a>7)return n=b(t,u+5),r=(n[2]<<8)+n[3],s=(n[0]<<8)+n[1],i=n[4],{width:r,height:s,numcomponents:i};u+=2}throw new Error("getJpegSizeFromBytes could not find the size of the image")},b=function(t,e){return t.subarray(e,e+5)};t.processJPEG=function(t,e,n,r,s){var i,o=this.color_spaces.DEVICE_RGB,a=this.decode.DCT_DECODE,u=8;return this.isString(t)?(i=y(t),this.createImageInfo(t,i[0],i[1],1==i[3]?this.color_spaces.DEVICE_GRAY:o,u,a,e,n)):(this.isArrayBuffer(t)&&(t=new Uint8Array(t)),this.isArrayBufferView(t)?(i=v(t),t=s||this.arrayBufferToBinaryString(t),this.createImageInfo(t,i.width,i.height,1==i.numcomponents?this.color_spaces.DEVICE_GRAY:o,u,a,e,n)):null)},t.processJPG=function(){return this.processJPEG.apply(this,arguments)}}(n.API),function(t){"use strict";t.autoPrint=function(){var t;return this.internal.events.subscribe("postPutResources",function(){t=this.internal.newObject(),this.internal.write("<< /S/Named /Type/Action /N/Print >>","endobj")}),this.internal.events.subscribe("putCatalog",function(){this.internal.write("/OpenAction "+t+" 0 R")}),this}}(n.API),function(t){"use strict";var e,n,r,s,i=3,o=13,a={x:void 0,y:void 0,w:void 0,h:void 0,ln:void 0},u=1,c=function(t,e,n,r,s){a={x:t,y:e,w:n,h:r,ln:s}},l=function(){return a},f={left:0,top:0,bottom:0};t.setHeaderFunction=function(t){s=t},t.getTextDimensions=function(t){e=this.internal.getFont().fontName,n=this.table_font_size||this.internal.getFontSize(),r=this.internal.getFont().fontStyle;var s,i,o=19.049976/25.4;return i=document.createElement("font"),i.id="jsPDFCell",i.style.fontStyle=r,i.style.fontName=e,i.style.fontSize=n+"pt",i.textContent=t,document.body.appendChild(i),s={w:(i.offsetWidth+1)*o,h:(i.offsetHeight+1)*o},document.body.removeChild(i),s},t.cellAddPage=function(){var t=this.margins||f;this.addPage(),c(t.left,t.top,void 0,void 0),u+=1},t.cellInitialize=function(){a={x:void 0,y:void 0,w:void 0,h:void 0,ln:void 0},u=1},t.cell=function(t,e,n,r,s,a,u){var d=l();if(void 0!==d.ln)if(d.ln===a)t=d.x+d.w,e=d.y;else{var h=this.margins||f;d.y+d.h+r+o>=this.internal.pageSize.height-h.bottom&&(this.cellAddPage(),this.printHeaders&&this.tableHeaderRow&&this.printHeaderRow(a,!0)),e=l().y+l().h}if(void 0!==s[0])if(this.printingHeaderRow?this.rect(t,e,n,r,"FD"):this.rect(t,e,n,r),"right"===u){if(s instanceof Array)for(var p=0;pn;n+=1)s=t[n],e?-1===e(i,s)&&(i=s):s>i&&(i=s);return i},t.table=function(e,n,r,s,i){if(!r)throw"No data for PDF table";var o,c,l,d,h,p,m,w,g,y,v=[],b=[],q={},x={},k=[],_=[],A=!1,C=!0,S=12,E=f;if(E.width=this.internal.pageSize.width,i&&(i.autoSize===!0&&(A=!0),i.printHeaders===!1&&(C=!1),i.fontSize&&(S=i.fontSize),i.margins&&(E=i.margins)),this.lnMod=0,a={x:void 0,y:void 0,w:void 0,h:void 0,ln:void 0},u=1,this.printHeaders=C,this.margins=E,this.setFontSize(S),this.table_font_size=S,void 0===s||null===s)v=Object.keys(r[0]);else if(s[0]&&"string"!=typeof s[0]){var z=19.049976/25.4;for(c=0,l=s.length;l>c;c+=1)o=s[c],v.push(o.name),b.push(o.prompt),x[o.name]=o.width*z}else v=s;if(A)for(y=function(t){return t[o]},c=0,l=v.length;l>c;c+=1){for(o=v[c],q[o]=r.map(y),k.push(this.getTextDimensions(b[c]||o).w),p=q[o],m=0,d=p.length;d>m;m+=1)h=p[m],k.push(this.getTextDimensions(h).w);x[o]=t.arrayMax(k)}if(C){var T=this.calculateLineHeight(v,x,b.length?b:v);for(c=0,l=v.length;l>c;c+=1)o=v[c],_.push([e,n,x[o],T,String(b.length?b[c]:o)]);this.setTableHeaderRow(_),this.printHeaderRow(1,!1)}for(c=0,l=r.length;l>c;c+=1){var T;for(w=r[c],T=this.calculateLineHeight(v,x,w),m=0,g=v.length;g>m;m+=1)o=v[m],this.cell(e,n,x[o],T,w[o],c+2,o.align)}return this.lastCellPos=a,this.table_x=e,this.table_y=n,this},t.calculateLineHeight=function(t,e,n){for(var r,s=0,o=0;os&&(s=a)}return s},t.setTableHeaderRow=function(t){this.tableHeaderRow=t},t.printHeaderRow=function(t,e){if(!this.tableHeaderRow)throw"Property tableHeaderRow does not exist.";var n,r,i,o;if(this.printingHeaderRow=!0,void 0!==s){var a=s(this,u);c(a[0],a[1],a[2],a[3],-1)}this.setFontStyle("bold");var l=[];for(i=0,o=this.tableHeaderRow.length;o>i;i+=1)this.setFillColor(200,200,200),n=this.tableHeaderRow[i],e&&(n[1]=this.margins&&this.margins.top||0,l.push(n)),r=[].concat(n),this.cell.apply(this,r.concat(t));l.length>0&&this.setTableHeaderRow(l),this.setFontStyle("normal"),this.printingHeaderRow=!1}}(n.API),function(t){var e,n,r,s,i,o,a,u,c,l,f,d,h,p,m,w,g,y,v;e=function(){function t(){}return function(e){return t.prototype=e,new t}}(),c=function(t){var e,n,r,s,i,o,a;for(n=0,r=t.length,e=void 0,s=!1,o=!1;!s&&n!==r;)e=t[n]=t[n].trimLeft(),e&&(s=!0),n++;for(n=r-1;r&&!o&&-1!==n;)e=t[n]=t[n].trimRight(),e&&(o=!0),n--;for(i=/\s+$/g,a=!0,n=0;n!==r;)e=t[n].replace(/\s+/g," "),a&&(e=e.trimLeft()),e&&(a=i.test(e)),t[n]=e,n++;return t},l=function(t,e,n,r){return this.pdf=t,this.x=e,this.y=n,this.settings=r,this.watchFunctions=[],this.init(),this},f=function(t){var e,n,s;for(e=void 0,s=t.split(","),n=s.shift();!e&&n;)e=r[n.trim().toLowerCase()],n=s.shift();return e},d=function(t){t="auto"===t?"0px":t,t.indexOf("em")>-1&&!isNaN(Number(t.replace("em","")))&&(t=18.719*Number(t.replace("em",""))+"px"),t.indexOf("pt")>-1&&!isNaN(Number(t.replace("pt","")))&&(t=1.333*Number(t.replace("pt",""))+"px");var e,n,r;return n=void 0,e=16,(r=h[t])?r:(r={"xx-small":9,"x-small":11,small:13,medium:16,large:19,"x-large":23,"xx-large":28,auto:0}[{css_line_height_string:t}],r!==n?h[t]=r/e:(r=parseFloat(t))?h[t]=r/e:(r=t.match(/([\d\.]+)(px)/),h[t]=3===r.length?parseFloat(r[1])/e:1))},u=function(t){var e,n,r;return r=function(t){var e;return e=function(t){return document.defaultView&&document.defaultView.getComputedStyle?document.defaultView.getComputedStyle(t,null):t.currentStyle?t.currentStyle:t.style}(t),function(t){return t=t.replace(/-\D/g,function(t){return t.charAt(1).toUpperCase()}),e[t]}}(t),e={},n=void 0,e["font-family"]=f(r("font-family"))||"times",e["font-style"]=s[r("font-style")]||"normal",e["text-align"]=TextAlignMap[r("text-align")]||"left",n=i[r("font-weight")]||"normal","bold"===n&&(e["font-style"]="normal"===e["font-style"]?n:n+e["font-style"]),e["font-size"]=d(r("font-size"))||1,e["line-height"]=d(r("line-height"))||1,e.display="inline"===r("display")?"inline":"block",n="block"===e.display,e["margin-top"]=n&&d(r("margin-top"))||0,e["margin-bottom"]=n&&d(r("margin-bottom"))||0,e["padding-top"]=n&&d(r("padding-top"))||0,e["padding-bottom"]=n&&d(r("padding-bottom"))||0,e["margin-left"]=n&&d(r("margin-left"))||0,e["margin-right"]=n&&d(r("margin-right"))||0,e["padding-left"]=n&&d(r("padding-left"))||0,e["padding-right"]=n&&d(r("padding-right"))||0,e["float"]=o[r("cssFloat")]||"none",e.clear=a[r("clear")]||"none",e},p=function(t,e,n){var r,s,i,o,a;if(i=!1,s=void 0,o=void 0,a=void 0,r=n["#"+t.id])if("function"==typeof r)i=r(t,e);else for(s=0,o=r.length;!i&&s!==o;)i=r[s](t,e),s++;if(r=n[t.nodeName],!i&&r)if("function"==typeof r)i=r(t,e);else for(s=0,o=r.length;!i&&s!==o;)i=r[s](t,e),s++;return i},v=function(t,e){var n,r,s,i,o,a,u,c,l,f;
+ for(n=[],r=[],s=0,f=t.rows[0].cells.length,c=t.clientWidth;f>s;)l=t.rows[0].cells[s],r[s]={name:l.textContent.toLowerCase().replace(/\s+/g,""),prompt:l.textContent.replace(/\r?\n/g,""),width:l.clientWidth/c*e.pdf.internal.pageSize.width},s++;for(s=1;sa;){if(s=i[a],"object"==typeof s){if(e.executeWatchFunctions(s),1===s.nodeType&&"HEADER"===s.nodeName){var w=s,g=e.pdf.margins_doc.top;e.pdf.internal.events.subscribe("addPage",function(){e.y=g,n(w,e,r),e.pdf.margins_doc.top=e.y+10,e.y+=10},!1)}if(8===s.nodeType&&"#comment"===s.nodeName)~s.textContent.indexOf("ADD_PAGE")&&(e.pdf.addPage(),e.y=e.pdf.margins_doc.top);else if(1!==s.nodeType||b[s.nodeName])if(3===s.nodeType){var y=s.nodeValue;if(s.nodeValue&&"LI"===s.parentNode.nodeName)if("OL"===s.parentNode.parentNode.nodeName)y=q++ +". "+y;else{var x=16*o["font-size"],k=2;x>20&&(k=3),h=function(t,e){this.pdf.circle(t,e,k,"FD")}}e.addText(y,o)}else"string"==typeof s&&e.addText(s,o);else{var _;if("IMG"===s.nodeName){var A=s.getAttribute("src");_=m[e.pdf.sHashCode(A)||A]}if(_){e.pdf.internal.pageSize.height-e.pdf.margins_doc.bottome.pdf.margins_doc.top&&(e.pdf.addPage(),e.y=e.pdf.margins_doc.top,e.executeWatchFunctions(s));var C=u(s),S=e.x,E=12/e.pdf.internal.scaleFactor,z=(C["margin-left"]+C["padding-left"])*E,T=(C["margin-right"]+C["padding-right"])*E,I=(C["margin-top"]+C["padding-top"])*E,B=(C["margin-bottom"]+C["padding-bottom"])*E;S+=void 0!==C["float"]&&"right"===C["float"]?e.settings.width-s.width-T:z,e.pdf.addImage(_,S,e.y+I,s.width,s.height),_=void 0,"right"===C["float"]||"left"===C["float"]?(e.watchFunctions.push(function(t,n,r,s){return e.y>=n?(e.x+=t,e.settings.width+=r,!0):s&&1===s.nodeType&&!b[s.nodeName]&&e.x+s.width>e.pdf.margins_doc.left+e.pdf.margins_doc.width?(e.x+=t,e.y=n,e.settings.width+=r,!0):!1}.bind(this,"left"===C["float"]?-s.width-z-T:0,e.y+s.height+I+B,s.width)),e.watchFunctions.push(function(t,n,r){return e.y0){s=s[0];var i=e.pdf.internal.write,o=e.y;e.pdf.internal.write=function(){},n(s,e,r);var a=Math.ceil(e.y-o)+5;e.y=o,e.pdf.internal.write=i,e.pdf.margins_doc.bottom+=a;for(var u=function(t){var i=void 0!==t?t.pageNumber:1,o=e.y;e.y=e.pdf.internal.pageSize.height-e.pdf.margins_doc.bottom,e.pdf.margins_doc.bottom-=a;for(var u=s.getElementsByTagName("span"),c=0;c-1&&(u[c].innerHTML=i),(" "+u[c].className+" ").replace(/[\n\t]/g," ").indexOf(" totalPages ")>-1&&(u[c].innerHTML="###jsPDFVarTotalPages###");n(s,e,r),e.pdf.margins_doc.bottom+=a,e.y=o},c=s.getElementsByTagName("span"),l=0;l-1&&e.pdf.internal.events.subscribe("htmlRenderingFinished",e.pdf.putTotalPages.bind(e.pdf,"###jsPDFVarTotalPages###"),!0);e.pdf.internal.events.subscribe("addPage",u,!1),u(),b.FOOTER=1}},y=function(t,e,r,s,i,o){if(!e)return!1;"string"==typeof e||e.parentNode||(e=""+e.innerHTML),"string"==typeof e&&(e=function(t){var e,n,r,s;return r="jsPDFhtmlText"+Date.now().toString()+(1e3*Math.random()).toFixed(0),s="position: absolute !important;clip: rect(1px 1px 1px 1px); /* IE6, IE7 */clip: rect(1px, 1px, 1px, 1px);padding:0 !important;border:0 !important;height: 1px !important;width: 1px !important; top:auto;left:-100px;overflow: hidden;",n=document.createElement("div"),n.style.cssText=s,n.innerHTML='