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