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