diff --git a/.gitmodules b/.gitmodules
new file mode 100644
index 0000000..3fe4dc5
--- /dev/null
+++ b/.gitmodules
@@ -0,0 +1,3 @@
+[submodule "eve"]
+ path = source/eve
+ url = git://github.com/adobe-webplatform/eve.git
\ No newline at end of file
diff --git a/README.md b/README.md
index 3ed76f8..42d3a96 100644
--- a/README.md
+++ b/README.md
@@ -1,2 +1,4 @@
-RedRaphael beta
-==========================
\ No newline at end of file
+redraphael
+==========
+
+Red Raphael
\ No newline at end of file
diff --git a/docs/css/dr.css b/docs/css/dr.css
new file mode 100644
index 0000000..169fa17
--- /dev/null
+++ b/docs/css/dr.css
@@ -0,0 +1,140 @@
+#content section.code {
+ display: block;
+ font-weight: 400;
+ background: #181818;
+ -moz-border-radius: 10px;
+ -webkit-border-radius: 10px;
+ border-radius: 10px;
+}
+#content section.code pre code {
+ font-size: 14px;
+}
+code {
+ font-family: source-code-pro, Menlo, "Arial Unicode MS", sans-serif;
+}
+a.dr-hash,
+a.dr-sourceline {
+ -webkit-transition: opacity 0.2s linear;
+ color: #333;
+ font-family: Menlo, "Arial Unicode MS", sans-serif;
+ margin: 0 0 0 .3em;
+ opacity: 0;
+ text-decoration: none;
+}
+h2:hover a.dr-hash,
+h3:hover a.dr-hash,
+h4:hover a.dr-hash,
+h5:hover a.dr-hash,
+h2:hover a.dr-sourceline,
+h3:hover a.dr-sourceline,
+h4:hover a.dr-sourceline,
+h5:hover a.dr-sourceline {
+ opacity: 1;
+}
+.dr-param {
+ float: left;
+ min-width: 8em;
+}
+.dr-type {
+ float: left;
+}
+.dr-type em,
+.dr-returns em,
+.dr-property em {
+ -moz-border-radius: 5px;
+ -webkit-border-radius: 5px;
+ background: #ccc;
+ border-radius: 5px;
+ float: left;
+ font-size: .75em;
+ font-style: normal;
+ font-weight: 700;
+ margin: 0 8px 0 0;
+ min-width: 80px;
+ padding: 2px 5px;
+ text-align: center;
+}
+.dr-type em.amp,
+.dr-returns em.amp,
+.dr-property em.amp {
+ float: none;
+ background: none;
+ font-size: 1em;
+ font-weight: 400;
+ font-style: italic;
+ margin: 0;
+ padding: 0;
+ min-width: 0;
+}
+.dr-property em.dr-type {
+ margin: 4px 16px 0 0;
+}
+em.dr-type-string {
+ background: #e1edb1;
+ color: #3d4c00;
+}
+em.dr-type-object {
+ background: #edb1b1;
+ color: #4c0000;
+}
+em.dr-type-function {
+ background: #cfb1ed;
+ color: #26004c;
+}
+em.dr-type-number {
+ background: #b1c9ed;
+ color: #001e4c;
+}
+em.dr-type-boolean {
+ background: #b1edc9;
+ color: #004c1e;
+}
+em.dr-type-array {
+ background: #edd5b1;
+ color: #4c2d00;
+}
+.dr-optional {
+ display: none;
+}
+ol.dr-json {
+ background: #ddd;
+ list-style: none;
+ margin: 0 -30px;
+ padding: 16px 30px;
+ line-height: 1.5;
+}
+ol.dr-json .dr-json-key {
+ float: left;
+ min-width: 50px;
+ margin-right: 16px;
+}
+ol.dr-json .dr-json-description {
+ display: table;
+}
+ol.dr-json ol.dr-json {
+ margin: 0;
+ padding: 0 0 0 50px;
+}
+#pageNav li.dr-lvl1 a {
+ padding-left: 1em;
+}
+#pageNav li.dr-lvl2 a {
+ padding-left: 2em;
+}
+#pageNav li.dr-lvl3 a {
+ padding-left: 3em;
+}
+#pageNav li.dr-lvl4 a {
+ padding-left: 4em;
+}
+#pageNav li.dr-lvl5 a {
+ padding-left: 5em;
+}
+#pageNav li.dr-lvl6 a {
+ padding-left: 6em;
+}
+#pageNav ol {
+ list-style: none;
+ margin: 0;
+ padding: 0;
+}
\ No newline at end of file
diff --git a/docs/css/main.css b/docs/css/main.css
new file mode 100644
index 0000000..2d0bda3
--- /dev/null
+++ b/docs/css/main.css
@@ -0,0 +1,508 @@
+html,body{
+ margin:0;
+ padding:0;
+ height: 100%;
+}
+body {
+ font-family: source-sans-pro, sans-serif;
+ position: relative;
+ -webkit-font-smoothing: antialiased;
+}
+body.light {
+ background: #F4F4F4;
+}
+body.dark {
+ color: #F0F1F1;
+ background: #4A4D4E;
+}
+body.light {
+ color: #181919;
+}
+
+h1 {
+ font-weight: 600;
+}
+#wrapper {
+ width: 100%;
+ overflow-x: hidden;
+ background: inherit;
+ position: relative;
+}
+#site {
+ width: 100%;
+ position: relative;
+ z-index: 10;
+ background: inherit;
+ left: 0;
+ transition: all 0.2s ease-out;
+ -webkit-transition: all 0.2s ease-out;
+ transform: translate3d(0, 0, 0);
+ -webkit-transform: translate3d(0, 0, 0);
+}
+#site:before{
+ position: absolute;
+ content: '';
+ left: -4px;
+ height: 100%;
+ width: 4px;
+ background: #3B3E3E;
+}
+#site.open {
+ transform: translate3d(250px, 0, 0);
+ -webkit-transform: translate3d(250px, 0, 0);
+}
+pre {
+ font-family: source-code-pro, sans-serif;
+ font-size: 12px;
+}
+/* Main Header */
+#main-header {
+ color: #373435;
+ background: #fff;
+ height: 98px;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 10px 20px;
+ position: relative;
+}
+#main-header hgroup {
+ text-align: center;
+}
+#main-header hgroup h1 {
+ font-size: 40px;
+ margin: 5px 0 0;
+ letter-spacing: -.065em;
+ line-height: 1.1em;
+}
+#main-header hgroup a {
+ color: #464646;
+ text-decoration: none;
+}
+#main-header hgroup a:hover {
+ color: #000;
+}
+#main-header hgroup p {
+ font-size: 13px;
+ color: #999;
+ margin: 0;
+}
+#main-header nav {
+ display: none;
+}
+#slide-menu-button {
+ position: absolute;
+ top: 20px;
+ left: 20px;
+ display: inline-block;
+ vertical-align: top;
+ -webkit-box-sizing: border-box;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ -webkit-background-clip: padding;
+ -moz-background-clip: padding;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ -o-text-overflow: ellipsis;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+ padding: 0 0.5rem;
+ line-height: 2rem;
+ letter-spacing: 1px;
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ vertical-align: baseline;
+ -webkit-box-shadow: inset 0 1px #fff;
+ box-shadow: inset 0 1px #fff;
+ -webkit-border-radius: 3px;
+ border-radius: 3px;
+ width: 2.6rem;
+ height: 2.6rem;
+ line-height: 2.6rem;
+ border: 1px solid transparent;
+ -webkit-box-shadow: none;
+ box-shadow: none;
+}
+#slide-menu:disabled,
+#slide-menu.is-disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+#slide-menu-button:active,
+#slide-menu-button.is-active {
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ background-color: #d3d7d7;
+ border: 1px solid #a5a8a8;
+ -webkit-box-shadow: inset 0 1px rgba(0,0,0,0.12);
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+#slide-menu-button span {
+ background-repeat: no-repeat;
+ background-image: url(data:image/svg+xml;base64,<?xml version="1.0" encoding="utf-8"?>
<!-- Generator: Adobe Illustrator 16.0.3, SVG Export Plug-In . SVG Version: 6.00 Build 0)  -->
<!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd" [
	<!ENTITY ns_extend "http://ns.adobe.com/Extensibility/1.0/">
	<!ENTITY ns_ai "http://ns.adobe.com/AdobeIllustrator/10.0/">
	<!ENTITY ns_graphs "http://ns.adobe.com/Graphs/1.0/">
	<!ENTITY ns_vars "http://ns.adobe.com/Variables/1.0/">
	<!ENTITY ns_imrep "http://ns.adobe.com/ImageReplacement/1.0/">
	<!ENTITY ns_sfw "http://ns.adobe.com/SaveForWeb/1.0/">
	<!ENTITY ns_custom "http://ns.adobe.com/GenericCustomNamespace/1.0/">
	<!ENTITY ns_adobe_xpath "http://ns.adobe.com/XPath/1.0/">
]>
<svg version="1.1" id="Layer_1" xmlns:x="&ns_extend;" xmlns:i="&ns_ai;" xmlns:graph="&ns_graphs;"
	 xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" width="40px" height="40px"
	 viewBox="0 0 40 40" enable-background="new 0 0 40 40" xml:space="preserve">
<switch>
	<foreignObject requiredExtensions="&ns_ai;" x="0" y="0" width="1" height="1">
		<i:pgfRef  xlink:href="#adobe_illustrator_pgf">
		</i:pgfRef>
	</foreignObject>
	<g i:extraneous="self">
		<g opacity="0.7">
			<g opacity="0.75">
				<path fill-rule="evenodd" clip-rule="evenodd" fill="#FFFFFF" d="M39,11v-1c0-1.47-0.48-2-2-2H3c-1.53,0-2,0.52-2,2v1
					c0,1.55,0.52,2,2,2h34C38.5,13,39,12.52,39,11z"/>
			</g>
			<g>
				<path fill-rule="evenodd" clip-rule="evenodd" d="M39,10V9c0-1.47-0.48-2-2-2H3C1.47,7,1,7.52,1,9v1c0,1.55,0.52,2,2,2h34
					C38.5,12,39,11.52,39,10z"/>
			</g>
		</g>
		<g opacity="0.7">
			<g opacity="0.75">
				<path fill-rule="evenodd" clip-rule="evenodd" fill="#FFFFFF" d="M39,22v-1c0-1.47-0.48-2-2-2H3c-1.53,0-2,0.52-2,2v1
					c0,1.55,0.52,2,2,2h34C38.5,24,39,23.52,39,22z"/>
			</g>
			<g>
				<path fill-rule="evenodd" clip-rule="evenodd" d="M39,21v-1c0-1.47-0.48-2-2-2H3c-1.53,0-2,0.52-2,2v1c0,1.55,0.52,2,2,2h34
					C38.5,23,39,22.52,39,21z"/>
			</g>
		</g>
		<g opacity="0.7">
			<g opacity="0.75">
				<path fill-rule="evenodd" clip-rule="evenodd" fill="#FFFFFF" d="M39,33v-1c0-1.471-0.48-2-2-2H3c-1.53,0-2,0.52-2,2v1
					c0,1.55,0.52,2,2,2h34C38.5,35,39,34.52,39,33z"/>
			</g>
			<g>
				<path fill-rule="evenodd" clip-rule="evenodd" d="M39,32v-1c0-1.471-0.48-2-2-2H3c-1.53,0-2,0.52-2,2v1c0,1.55,0.52,2,2,2h34
					C38.5,34,39,33.52,39,32z"/>
			</g>
		</g>
	</g>
</switch>
<i:pgf  id="adobe_illustrator_pgf">
	<![CDATA[
	eJzsvemSJLmRJvgE9g6+P1qEHJkKGgCDAcYdWZHwOHo4Q3aXkOxjpWWkJJkVTeZ0HiVZWeRyn371
+1QVgLl7ZEUeRVK2K1CVEa4Og5nhUOjxqeLv/o8vf/PF9ddvfvfwRbqaD9Pf/d3N24dn7968/fmB
1MMvXr787tt3b0H6ya9/egjr1SyVrn9Rv7KK//zw9tsXb17/nF9dJfnyHlf/5BffPvvdw8uXD4df
Pnv99bs/PHv58Panh5/8VL7/7Yt3Lx+kxh+evfrdd29///D26ts//v6nfmtp6/bZO/l++Vn8WUiH
+PM5Hr78Fb5+9vqPz7799sX/+4CbpYp7Hd989/rrF69/f3zz/wjx8EXKh7Qdvijy1X9/8euHb9/z
/e2b59+9enj97su3b54/fPvtzZuXb95++/PDzZ+fvT786tnv5Ztnh/9b3uDNnw7Hl8+e/8ckb52/
un/x8kFe8NWzd4cQ0RfXvwjxq+N3L15+/Q/fvfrdg7z6upGcvmKT//SttCXN4m+Qy1e/eCWU3zy8
eyfPJTdEf/7674/jYwiR5Sf/9uuH379g90u3/K+fWrNv33zz6tnb/8C1hy+W+SD/2R1/+/Dqm5fS
e3zdOF/lwxf8t/9p9eQtWOeLuFbplbIe0lwPKVf9vvfNwx9fPPzp54d/ePP6QTvg+u273+gQLMs8
67/6za+/kzH+p9cv3smDrSBt2gO/evP1w0up366/f/mML84S+r9a4bfPZE68k7F78/K7d5xZ1e8g
PfzLZ39+wDAFvcE/fvPw+rdv/pnP+EUM0k7crrZDXOSNQtyWQ6hsP+JRt7ndM/R/tWk0hGa8fcyQ
L2Wc/vHti9+/eP3zL1KVvk5x1iH8+7cvvu4jGA7V/pfGr+rw/+b/69PKi7979/Danl7mzc2vhnkw
X/3qN3LXu9df37x5hd7/FhNcJsBrmRsv3/xev2t/8xu5/Ltv9A34+SsZqC/fvniNNqd/4Df1qy9f
fidf/f3bN99984vX//5m+oku7X9+eC7rV8by68M//u5/ywdZn5ybh9++ffZcGpDPrc7Vsxff/PS9
zcnLvX046JdyJT/67++/+vbh32VV9cuVevf6jw8v33wzNNsowlUO//Ls7Tff3/SXL5+9fvb2QHpr
+Zcv/ijfPJOe6m132hMalVnyjXQOL2GVkxu8p8Lw1RNu9OzdH4QBPbz++tvWtn7cP7jSvr+93zzH
HHx7OL797ts/HH775s3L1uz+q9a6kUlF/b+Ne3zJC17/42vtoPM7WYXTOwn/+Zu7i9R+/A7y5d9y
6zfPXr588fu3z775w4vnl25w4ft2J/3uQybWn1/97s3LF9++6vNpoHz57O27F89fPvzmz9++e3j1
5ME93H39QpjcI8v4vXV+86dn757/4Zcvfvf22dsXD+9dfRiAf3/x+muZ+7/57sW7h95Bb159Aynl
8Js/PPvmga/x7g/3rPmb1mD+Slj/yNy/+OI9XD+sh+Pr4fu/f/vs6xeyoYjQ9fdvXn798Prwa7D3
afdJtq9yOH49/ds0Wwksy+Hvvjq+nf7bNJ/+hJMSh5KsLFYy//ey4v9Jfo2ltFLnbVeuh3KUcmPl
dih38/0033vZPdj4E4eShrLsStYy8VcJq5TC33Uo265ch2MrN60cw60U/3Q7yYc7km7D/VjGruPD
+U8ayjKUHFeUSX/FMpQaNyn67/VQjlqm/+vSWD42mmk3nsswqn1spUxteJdhkHVUx9GtUvAbo1p3
47sNIyz/TjbUPtjHNsz6G6N8N462/Ks/88WxjtNuoJOMK35jZDHGK0vlKJeTEb62sh9bGUv8umO5
PRlFv2niGPYxQ9FB8iHbbLBkgCYbI4zVjZVbljsr9yi+svAy41gGjtJiHV/ZoUcuknv2TeTLr3y9
ja9yyycO9oQrp4w8wsT74j4xpbSkNRVRtrZ0lHKb7mW0w5KWZclLWeqyLdfLcbld7pZ76ciYU17y
mkve8nU+5pt8l+8neaAgs3RZ87qudd3W6/W43qy3673MhSDdsJRc1lLLVo7lptyWe5knoca61FzX
WupWr+ux3ta7ei/zJU5b2pZN2tnqtm3H7Wa72+5l0oTrdL1c5+v1ul5v19fXRyk317fXd5xJUV5g
OeZjOdbjJl8djzfHu+O9zK4wsbfTzXKTb+QBbrab65vjzc3N7c29TLbAUVhu8+16Kw93W2+32+Pt
ze3d7f3dLOOS7pa79a7c1bvtTu51d3N3O93d3d1jRtxLh93ne3nTe3n2+2spcst7ufL+5EfGcpbR
bJ/vhnI7lJuhHHfleiibl0n+qUMpQ1l3JQ9lGUoaikzA6T62MrKwznzn8dHvxp/bXbnRMvHXcSjX
Q9l2pQ6lDGUdSp7ucivLrqShxKGEocxj0S6frO/Hpx9/boZy1MKVuRvH/ejtx8xHaz9E+2GRoZiG
0dgPwb7j991t3XzSt+zPqXXpviP3vbfvsZN+2nfNdNIj2hvXVjaWaqVYWVmylYUlWYkT155tmcr6
2/T3N9N7+c+R5drKxlKtyNqe5J/VSraysCQrynqd0QsH9bG84/gdOV6V45M5IBiEmd1+y26+Zq8W
9uPCfgvspzt2zJEdUSe+eea74gVnvtAtX+Caj1z4hAufSJ5DZsuddOaN8K1r4V9V+Ngq3GwRrhaF
m8+TTKQ76eUbYXrXwv7qdRFGmIUhJtlRguyw9zLit9Idx+1a2GbdijDQLIw0yRYUZD++l4lwK910
FG671ToJ412FAS81CSsOsnHfywy5lT48lmth1bUUYdpZmHcSFh5ko7+XuXMrHXsUBr8Joy/C7rOw
/STMP6zzJNP4Tsb4RjaGa9kgqmwTa86yYSTZOILIDvcy3W5lLI6yqWyyuZRllW1mke0myqYzy7y/
k1lxIxvRtWxIdZKdaU1Z9qgke1UQWeVeZuitDB92UWyr2GizbG7YiEOcA7hsE2itFup5LdaTBaWb
OuQ37PwbJQOVFER24Hx45HZshg345f1iiB2xM4b7gdUrS9hkUmWZUEEm06WpFG2P4O6DH92BAtcl
dqHM1es7ke1F2IpktxEhgPfGftc2u1VmWb2w3aW23WGzu8YNeXGTGJLJoMWlBgoNdxQbggkOuYkO
12O3oxcgp8332qR8oDB3y49YLbpWfENsuyi/55vzrXUYrimV3KXdyFIkDE3oUQlPBR+V5G4oiN9J
08GEbH0dl5pNSqZawaa8od4MG2Fv6MUXe0IfeRwoHaSdsEBJIcnIF5kBR86FE+nAuyLbBuAcZ+Q5
fap0vjNyHvIe7eIL7GjPjb6PF2nXn6zGT16Mxyb+eumKTN2pOMXUHi15pxmZtjTZH/6jAvlOR92r
YL7HNAWtSfosx2mn2N1QgnalAL+3k1JPSlci9a91cjWzleWspLMSLxT7mXaa7plChxU3Cmnt9+3F
Inxiapr2jSljR9XOLpbt0VJd75sGJV/L+sTSVco8lmn/sRkb+t/nJT1SOCmmOT6xXOrgC2V6YsUn
/1zS3T/p57M16PtiaD2qMqqL2bdWVMJyo4QbKarpwm7EsJU+mdrsy9vNIn36n65ml1B99Y4L9qjr
2FX689VamglgtZKbgaDbhtK4QKfvX5nyVzNLdPPUyXL0BXk7DUat0ch1vLgcjzvT2PmCNGPK6fqs
Oytbt864hebSEm2fp7OVeboIzxfle8t0ZjC8tFLft35Pluc02CA/eQ2/bx3/9ZfdZ2/wkog6aj/n
+o/KI+f6j0shM3ToO0obm2l+q8kbsWk/94PEUXYSRxw0oBvRgagBTaICQQlSFQhK0EwNSHWgGxp/
VA+qTQ9KpgdBE7oRLehIPajSlLRMVIUCF8c9teJb3EGur6IJqS4EbSiqNkR96Nb0oSP1oUp9CBoR
OFeYTCm6p4YLtUgVI6hGlaoRlKOFylHgUrqnhnRDDQk6ErSkIndFVVGEJlGUEtlhoGnmjurSrahL
UJigMkFpgtpUlkzFCapTIqeaqT7dm2J91DKJ3AbJDbJbofy2on7CT6D81Lk4ZFodiy6lqWRWqUCh
LFPj1GdS1yBjuSVVf48MeBSRKA9NgxC0Y6pnjHMnuex45I4jTjuZBOWEt11ka++RIM4lhqexiUfZ
x+dcx6YtahnNVOPP3iw3mBr3tjDfVyf7I+5K2pW92S2flHVXyrQz5ZWdka9SUdqX65NyPC1Ts3l5
uT0r5z/358Xn2DTu1PfhQokXS7pYzJJ3XvIjZX20NFvutDPsFpqb31e27ynX086cfLkcn1jI3Ked
5fp95fZpZbpAvDSITy7To1995M/5JvqxDZnhZm7qZzSPhZbFipoz3bxZrLj1c7NittHJTKX+47yh
LxsfD+esriK7JO9WWrXZLhONRlrWVtzMW1vZrFy3crzAo+RnOmNQl1hTN91f5kYDH5oeYUKPcZ/H
uM7AaY604TzOa97DY3ZMxRjL9F6ecoGTfB8Lmd7PN57AMk5YwzRyiY9mDcN6f4wxPJENnK/8x9fx
+8tfaB1/kA11Z0UVke1GjZ10wkLOSssyiaS3Dr7Yo3AC9cfO5pGFTzbTK1tFoHTP7C18s7K5qHM2
URSFg7ZMlE/dSws/7R09teqrTZAu6a/FD3y21xSA4bfFAlLfLby3ECzpwZ3oxK2QtOnIPQo3um3u
3FnEP0iS8Oku8niQkAvF8k1E9Gv6d2+EX9yZl3e+DpPI8ZHyPNy9cPiu14WS/jY4fun6lYLpCEEw
0MCY6AaGIxgCeaE7uE7HjT5h9QrfkBdhad/TPzw3luvM1tmss9fOVo2jTo2VdhbqrLMzzc4rO4/s
vHHkiLdT44Mj/+t8b8/rRhY3srWBmU0DH9uzrhNmdUHyOedHg0xzWW55jMuciB1jmd7HNb6XTTxx
g//YdWyG63+bRlBxOET5az3U+WqR5XIo6UoWSETV30x/99VTqh6/PW/xKkvFPB9quBI9cH20udN6
aOsqrGVbgnwZoN1GRynnWuMif5Qc7esY8rodUEEbiVdpzku72Sc3xKeZRTPNq9SaZZGvuE6Ua1G+
2da25HkjdlqY4pLlj4jfYWgrXc3CVvpDfab29NnWWRgq2sphzoXXVWHE2paoiFXbj8JR0ZaMW952
beV53oZn+zztffwoxnJVU9rQUs1l+8ShPG3twky1J9+2K+Hnj03TfSVp5XjbII6OVLwMXvzHt8//
8OLrn07+B4D9DbJ4ta5B9kmMY4qlrAQwyuJi9+JrjPxu2doVB79C4xDsioNdYctUHliU9eGVPuJi
7bCnXohuWtarlDC9P+LG++s/pJt/8x9/lj7Gv3Lhoh3833z/ubddJ9p+k82fWWx3ubYN5XZ0PNtm
sRgWSnYDSkKyFwTDeS0UEYD0OlI0uKdVSEWClcLANcWAO+4SAHph39c9/5rWvttTYKOCGN0jpPZt
NZNjL54NAKouXUdw0ivcQHH5MazkpZ/H0bCj8TqbUYZWQkVOEDvR0BOnhsJiKArFUQ==
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
5+8WHNujgn37Jvf02foBF49O17EtRgdL6PPOr2X3eTnApGRT1pjj5tvyKZydOhfd8sG4HeMKh/VO
7013OIJF5f3+3OsnvTc1vMU23l1vVJ8w217gxTs3bjan2vPDR86ahVpD/2JkTVDjJsJv+qe1Z/an
yOaVZeNGuLhUMMvf2+8wpQ+OD548yfVhiTMCf1R/efbHf3725Y9ffXK/+fLzb9I/Pqvvfz14fP76
i1e9333z6Hef3/3koNNsDP9gvv/rQ/P7L77+/vGfn6q/fHH0r0+efj24bGWPHsqn+R2h1O+TZHj4
8+Grz5M/fH3/340/ff3l5xfDr4d/k/f27nx9/+h3Ay70zWj/1V//fvT1l6o8Puh89efWYaPxyaup
po7a/4D2ssNHd77I/vvx6PDnn/bVf9c/f3DePxrC4I5ef/Zn8/vLR4fqDz/u/9z95Me9O4dnyTen
M5H9ISvOsh/+/s9/PfjuoPHD/Ebjcl/89PWXbx799PUXw8b5Z4ef37l8dPdx+2zvDjHr0f+cPLs8
PPvpx2y/+3X35Rdn+69HB6+z/xZj7PifPx62xNF/vv7yL5/8aPFAl4cH/3717z58+uN/Dp+0n/xu
v57//IcHx/Xf92wfXjbbl3t3ip/vftZ62NJ/v3vwWp3c//LBnfSPn+1/+/n/fPb1wSffPzooL//0
5x+++f3r+61W8w1+6nz28OzotW1ZJPea2aDzh//5ovPvb9r73Tt/+aQ++Oxflw+Ojv/4H+z/p1/f
/+Z1unfH3P/hp68f9FqfnH/21dP797Lzf33VybJ7w7P0waD1RHz25gvhMbYOvxn+AGzLPimzH9Ok
/UXn4F4Txlc8/epu/fNyv5s9P7cU/OPoztcHT778/Y8PPy/0EMblyT/N7/+cHfT//dmXP7T/+YU8
/f1PhPbPvTtA0J/Nn36PQ/JP86P5ew/59Of9N5+aupuaP7SPEvHT758e3mt++cdHv/vsvwfYisEf
/k1YqMjeneT0/z5R9PmzPz/60n368seHf7PFDz5/+D8WmfyHfAJT92Xy2Z///PBzefiXV185PD9+
9eX99s/f/ptG0ncY8D3b164VKLT/je/AT6ED4u5XL7BQqQimf7d/eEKsPiyHf1Hmv83PrQffHf78
2eHZvb/952Gz+ckf9s3p938vnt95+f2DZwf7zw/Pjjv/+fo/P33xau/O/2/vy7fTZ3IFn4B3CPsO
ttmXsGMgQBIIECAJO4EEwk739Pxxn32qyrvxCnx3zp0z5zv9a4KNVKWSVJJKKuWCvdaQImY/POsX
h7i7kwt2sy9k4fd9mP/5DfsT33/WBZn/duGAgKlRJNKcbTl80bf9XzX7UnNVyYJzVqVowxCa4n2w
+qedr+FOd8Z7akKpcHScSbROpmzr6XS+nJqIsjw6MAvRPRgZUG9Acl7yJ4O12J9ZF8QomS5g5Gcm
gFggOUqSBcAdLo87t40NxGslpCx/YZmFoDgnvTyeEZXAXPh0eqqVssMnHHFM0r1LjEjHd9WXxZKt
PuG0DJLUQITkCJ+bsTnpsO2c+WW4uSq6az6S41QgAL0t1DBvxRnk0AwQqj8bmJrZmV8si8dIYtpu
ZiN94l28Bq/ldUcA21Qqej2TmNSSxFbzat5gzbbqMzfQMKlYIVfrraRGi97kvVfqRb6B0BQJjCgF
65ecc3olna/rBFkI9Ql3qjTyGqzcvMCspt9ksRCK5MLelw5SOD683PEgpAX/aOfK/Z5mf7n1prPN
tpZdCwBRdbMAdkXP9pkgbd7IR7b5vbSDn33ZcmF3bUlpS2fh214Og9Ful5QWLHa6ZQ==
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
	]]>
	<![CDATA[
	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
wknerg==
	]]>
</i:pgf>
</svg>
);
+ -webkit-background-size: cover;
+ -moz-background-size: cover;
+ background-size: cover;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ overflow: hidden;
+ vertical-align: middle;
+ width: 1.3rem;
+ height: 1.3rem;
+}
+#download-btn {
+ display: none;
+}
+/* Content */
+#content {
+ width: 100%;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 20px;
+}
+/* Code */
+#content section.code {
+ display: none;
+ background: #FFF;
+ border: 1px solid #E0E0E0;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 15px;
+ font-size: 12px;
+ -moz-border-radius: 1px;
+ -webkit-border-radius: 1px;
+ border-radius: 1px;
+ font-weight: 400;
+}
+article.component {
+ padding: 0 0 10px;
+}
+#content section.code h3 {
+ margin: 0;
+ font-size: 12px;
+ color: #000;
+ font-weight: 400;
+}
+#content header h2 {
+ font-weight: 300;
+ margin: 10px 0 25px;
+ font-size: 20px;
+ position: relative;
+ display: inline-block;
+ padding-right: 10px;
+}
+body.light #content header h2 {
+ background: #F4F4F4;
+}
+body.dark #content header h2 {
+ background: #4A4D4E;
+}
+#content header{
+ position: relative;
+}
+#content header:before {
+ content: '';
+ width: 100%;
+ display: block;
+ position: absolute;
+ left: 0;
+ top: 23px;
+}
+body.light #content header:before {
+ border-bottom: 1px solid #E0E0E0;
+}
+body.dark #content header:before {
+ border-bottom: 1px solid #58595A;
+}
+#content pre {
+ padding: 0;
+ margin: 2px 0 10px;
+}
+.showcode {
+ margin: 10px 0;
+}
+.showcode a, section.examples a {
+ color: #288edf;
+ text-decoration: none;
+}
+.showcode a:hover, section.examples a:hover {
+ text-decoration: underline;
+}
+section.examples ul {
+ margin: 0 0 20px;
+ padding: 0 0 0 20px;
+}
+section.examples h4 {
+ margin-bottom: 5px;
+}
+section.examples li {
+ color: #58595A;
+}
+/* Side Nav */
+#sideNav {
+ background: #4A4D4E;
+ position: absolute;
+ width: 100%;
+ z-index: 1;
+ height: 100%;
+ left: 0;
+}
+#sideNav ul {
+ list-style: none;
+ margin: 0;
+ padding: 0;
+}
+#sideNav li a {
+ color: #F0F1F1;
+ display: block;
+ height: 46px;
+ font-size: 16px;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 12px 0 0 20px;
+ text-decoration: none;
+}
+#sideNav nav.site, #sideNav .combo {
+ border-bottom: 1px solid #58595A;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 10px;
+ display: block;
+}
+#pageNav li {
+ border-bottom: 1px solid #58595A;
+}
+
+select.docNav {
+ -moz-border-radius: 3px;
+ -webkit-border-radius: 3px;
+ border-radius: 3px;
+ background: #595B5B;
+ background-image: none;
+ box-shadow: 0 0 0 1px #303233;
+ border: none;
+ border-top: 2px solid #666767;
+ color: #FFF;
+ text-shadow: 0 -1px 0 #000;
+ overflow: hidden;
+ font-size: 14px;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ padding: 20px;
+ -webkit-appearance: none;
+ -moz-appearance: button;
+}
+@media screen and (min-width: 650px) {
+ #site.open {
+ transform: translate3d(0, 0, 0);
+ -webkit-transform: translate3d(0, 0, 0);
+ }
+ #main-header nav {
+ display: inline-block;
+ position: absolute;
+ right: 0;
+ top: 40px;
+ }
+ #main-header ul {
+ list-style: none;
+ }
+ #main-header nav li {
+ display: inline-block;
+ margin: 0 18px;
+ }
+ #main-header nav li#download-btn {
+ display: none;
+ }
+ #main-header nav li a {
+ text-decoration: none;
+ font-size: 20px;
+ color: #7F7F7F;
+ }
+ #main-header nav li.selected a {
+ color: #373435;
+ }
+ #slide-menu-button{
+ display: none;
+ }
+ #main-header hgroup {
+ text-align: left;
+ position: absolute;
+ display: inline-block;
+ top: 24px;
+ }
+ #main-header hgroup h1 {
+ font-size: 60px;
+ }
+ #main-header hgroup p {
+ font-size: 15px;
+ }
+ #main-header {
+ color: #373435;
+ background: #fff;
+ height: 148px;
+ }
+ #content {
+ padding-left: 240px;
+ }
+ /* Side Nav */
+ #sideNav {
+ background: transparent;
+ width: 220px;
+ z-index: 20;
+ left: 10px;
+ top: 150px;
+ height: auto;
+ }
+ #sideNav nav.site {
+ display: none;
+ }
+ #sideNav .combo {
+ border-bottom: none;
+ padding: 36px 0;
+ }
+ #sideNav li a {
+ padding: 12px 0 0 10px;
+ }
+ body.light #sideNav li a {
+ color: #797B7B;
+ }
+ body.light #pageNav li {
+ border-bottom: 1px solid #E0E0E0;
+ }
+ select.docNav {
+ background: #595B5B;
+ box-shadow: 0 0 0 1px #303233;
+ border: none;
+ border-top: 2px solid #666767;
+ color: #FFF;
+ text-shadow: 0 -1px 0 #000;
+ padding: 3px 20px 4px 8px;
+ -webkit-appearance: none;
+ }
+ body.light select.docNav {
+ box-shadow: 0 0 0 1px #949696;
+ background: #DDE1E1;
+ border-top: 1px solid #FFF;
+ color: #454545;
+ text-shadow: 0 -1px 0 #FFF;
+ width: 192px;
+ }
+}
+@media screen and (min-width: 880px) {
+ #content {
+ padding-left: 300px;
+ }
+ #sideNav li a {
+ display: block;
+ height: 60px;
+ padding: 22px 0 0 10px;
+ text-decoration: none;
+ }
+ #content header h2 {
+ font-size: 28px;
+ }
+ #content header:before {
+ top: 30px;
+ }
+ section.code div {
+ display: inline-block;
+ width: 48%;
+ vertical-align: top;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ }
+ section.code div:first-child {
+ padding-right: 5px;
+ }
+ section.code div:last-child {
+ padding-left: 5px;
+ }
+ .max-width {
+ max-width: 1180px;
+ position: relative;
+ margin: 0 auto;
+ }
+ header#main-header .max-width {
+ top: -10px;
+ }
+ #main-header nav li a {
+ font-size: 22px;
+ }
+ #main-header nav {
+ display: inline-block;
+ }
+
+ #main-header nav li {
+ margin: 0 25px;
+ }
+ #main-header nav li:last-child {
+ margin-right: 0;
+ }
+}
+@media screen and (min-width: 940px) {
+ #main-header nav li#download-btn {
+ display: inline-block;
+ }
+ #main-header nav li a#download-btn{
+ position:relative;
+ top: -15px;
+ display:inline-block;
+ box-sizing:border-box;
+ -moz-box-sizing:border-box;
+ background-clip:padding-box;
+ font:inherit;
+ background:transparent;
+ -webkit-user-select:none;
+ -moz-user-select:none;
+ user-select:none;
+ text-overflow:ellipsis;
+ white-space:nowrap;
+ overflow:hidden;
+ font-size:16px;
+ line-height:3rem;
+ letter-spacing:1px;
+ color:#454545;
+ text-shadow:0 1px #fff;
+ vertical-align:top;
+ background-color:#e5e9e8;
+ box-shadow:inset 0 1px #fff;
+ border:1px solid #a5a8a8;
+ border-radius:6px;
+ margin:0;
+ padding:0 1.25rem;
+ }
+ #main-header nav li a#download-btn, #main-header nav li a#download-btn:hover {
+ border:1px solid #143250;
+ background-color:#288edf;
+ box-shadow:inset 0 1px rgba(255,255,255,0.36);
+ color:#fff;
+ font-weight:500;
+ text-shadow:0 -1px rgba(0,0,0,0.36);
+ }
+ #main-header nav li a#download-btn:hover {
+ background-color:#2f9cf3;
+ }
+ #main-header nav li a#download-btn:active, #main-header nav li a#download-btn.is-active {
+ background-color:#0380e8;
+ box-shadow:inset 0 1px rgba(0,0,0,0.12);
+ }
+ #main-header nav li a#download-btn:disabled, #main-header nav li a#download-btn.is-disabled {
+ opacity:.3;
+ cursor:default;
+ pointer-events:none;
+ }
+}
+
+
+
diff --git a/docs/css/prism.css b/docs/css/prism.css
new file mode 100644
index 0000000..b508b61
--- /dev/null
+++ b/docs/css/prism.css
@@ -0,0 +1,168 @@
+/**
+ * prism.js Twilight theme
+ * Based (more or less) on the Twilight theme originally of Textmate fame.
+ * @author Remy Bach
+ */
+code[class*="language-"],
+pre[class*="language-"] {
+ color: white;
+ direction: ltr;
+ font-family: source-code-pro, Consolas, Monaco, 'Andale Mono', monospace;
+ text-align: left;
+ text-shadow: 0 -.1em .2em black;
+ white-space: pre;
+ word-spacing: normal;
+
+ -moz-tab-size: 4;
+ -o-tab-size: 4;
+ tab-size: 4;
+
+ -webkit-hyphens: none;
+ -moz-hyphens: none;
+ -ms-hyphens: none;
+ hyphens: none;
+}
+
+pre[class*="language-"],
+:not(pre) > code[class*="language-"] {
+ background:#181818;
+}
+
+/* Code blocks */
+pre[class*="language-"] {
+ margin: .5em 0;
+ overflow: auto;
+}
+pre[class*="language-"]::selection { /* Safari */
+ background:hsl(200, 4%, 16%); /* #282A2B */
+}
+pre[class*="language-"]::selection { /* Firefox */
+ background:hsl(200, 4%, 16%); /* #282A2B */
+}
+
+/* Inline code */
+:not(pre) > code[class*="language-"] {
+ border-radius: .3em;
+ border: .13em solid hsl(0,0%,33%); /* #545454 */
+ box-shadow: 1px 1px .3em -.1em black inset;
+ padding: .15em .2em .05em;
+}
+
+.token.comment,
+.token.prolog,
+.token.doctype,
+.token.cdata {
+ color: hsl(0, 0%, 47%); /* #777777 */
+}
+
+.token.punctuation {
+ opacity: .7;
+}
+
+.namespace {
+ opacity: .7;
+}
+
+.token.tag,
+.token.boolean,
+.token.number {
+ color: hsl(14, 58%, 55%); /* #CF6A4C */
+}
+
+.token.keyword,
+.token.property,
+.token.selector {
+ color:hsl(53, 89%, 79%); /* #F9EE98 */
+}
+.token.attr-name,
+.token.attr-value,
+.token.string,
+.token.operator,
+.token.entity,
+.token.url,
+.language-css .token.string,
+.style .token.string {
+ color:hsl(76, 21%, 52%); /* #8F9D6A */
+}
+
+.token.atrule {
+ color:hsl(218, 22%, 55%); /* #7587A6 */
+}
+
+.token.regex,
+.token.important {
+ color: hsl(42, 75%, 65%); /* #E9C062 */
+}
+
+.token.important {
+ font-weight: bold;
+}
+
+.token.entity {
+ cursor: help;
+}
+pre[data-line] {
+ padding: 1em 0 1em 3em;
+ position: relative;
+}
+
+/* Markup */
+.language-markup .token.tag,
+.language-markup .token.attr-name,
+.language-markup .token.punctuation {
+ color: hsl(33, 33%, 52%); /* #AC885B */
+}
+
+/* Text Selection colour */
+::selection {
+ background: hsla(0,0%,93%,0.15); /* #EDEDED */
+}
+::-moz-selection {
+ background: hsla(0,0%,93%,0.15); /* #EDEDED */
+}
+
+/* Make the tokens sit above the line highlight so the colours don't look faded. */
+.token {
+ position:relative;
+ z-index:1;
+}
+.line-highlight {
+ background: -moz-linear-gradient(left, hsla(0, 0%, 33%,.1) 70%, hsla(0, 0%, 33%,0)); /* #545454 */
+ background: -o-linear-gradient(left, hsla(0, 0%, 33%,.1) 70%, hsla(0, 0%, 33%,0)); /* #545454 */
+ background: -webkit-linear-gradient(left, hsla(0, 0%, 33%,.1) 70%, hsla(0, 0%, 33%,0)); /* #545454 */
+ background: hsla(0, 0%, 33%, 0.25); /* #545454 */
+ background: linear-gradient(left, hsla(0, 0%, 33%,.1) 70%, hsla(0, 0%, 33%,0)); /* #545454 */
+ border-bottom:1px dashed hsl(0, 0%, 33%); /* #545454 */
+ border-top:1px dashed hsl(0, 0%, 33%); /* #545454 */
+ left: 0;
+ line-height: inherit;
+ margin-top: 0.75em; /* Same as .prism’s padding-top */
+ padding: inherit 0;
+ pointer-events: none;
+ position: absolute;
+ right: 0;
+ white-space: pre;
+ z-index:0;
+}
+.line-highlight:before,
+.line-highlight[data-end]:after {
+ background-color: hsl(215, 15%, 59%); /* #8794A6 */
+ border-radius: 999px;
+ box-shadow: 0 1px white;
+ color: hsl(24, 20%, 95%); /* #F5F2F0 */
+ content: attr(data-start);
+ font: bold 65%/1.5 sans-serif;
+ left: .6em;
+ min-width: 1em;
+ padding: 0 .5em;
+ position: absolute;
+ text-align: center;
+ text-shadow: none;
+ top: .4em;
+ vertical-align: .3em;
+}
+.line-highlight[data-end]:after {
+ bottom: .4em;
+ content: attr(data-end);
+ top: auto;
+}
\ No newline at end of file
diff --git a/docs/css/topcoat-desktop-light.css b/docs/css/topcoat-desktop-light.css
new file mode 100644
index 0000000..1e221af
--- /dev/null
+++ b/docs/css/topcoat-desktop-light.css
@@ -0,0 +1,2883 @@
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.button-bar {
+ display: table;
+ table-layout: fixed;
+ white-space: no-wrap;
+ margin: 0;
+ padding: 0;
+}
+
+.button-bar__item {
+ display: table-cell;
+ width: auto;
+ border-radius: 0;
+}
+
+.button_bar__item > input {
+ position: absolute;
+ overflow: hidden;
+ padding: 0;
+ border: 0;
+ opacity: 0.001;
+ z-index: 1;
+ vertical-align: top;
+ outline: none;
+}
+
+.button-bar__button {
+ border-radius: inherit;
+}
+
+.button-bar__item:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.button,
+.topcoat-button,
+.topcoat-button--quiet,
+.topcoat-button--large,
+.topcoat-button--large--quiet,
+.topcoat-button--cta,
+.topcoat-button--large--cta,
+.topcoat-button-bar__button,
+.topcoat-button-bar__button--large {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+ text-decoration: none;
+}
+
+.button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.button--disabled,
+.topcoat-button:disabled,
+.topcoat-button--quiet:disabled,
+.topcoat-button--large:disabled,
+.topcoat-button--large--quiet:disabled,
+.topcoat-button--cta:disabled,
+.topcoat-button--large--cta:disabled,
+.topcoat-button-bar__button:disabled,
+.topcoat-button-bar__button--large:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+.topcoat-button,
+.topcoat-button--quiet,
+.topcoat-button--large,
+.topcoat-button--large--quiet,
+.topcoat-button--cta,
+.topcoat-button--large--cta,
+.topcoat-button-bar__button,
+.topcoat-button-bar__button--large {
+ padding: 0 1.25rem;
+ font-size: 16px;
+ line-height: 3rem;
+ letter-spacing: 1px;
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ vertical-align: top;
+ background-color: #e5e9e8;
+ box-shadow: inset 0 1px #fff;
+ border: 1px solid #a5a8a8;
+ border-radius: 6px;
+}
+
+.topcoat-button:hover,
+.topcoat-button--quiet:hover,
+.topcoat-button--large:hover,
+.topcoat-button--large--quiet:hover,
+.topcoat-button-bar__button:hover,
+.topcoat-button-bar__button--large:hover {
+ background-color: #edf1f1;
+}
+
+.topcoat-button:active,
+.topcoat-button--large:active,
+.topcoat-button-bar__button:active,
+.topcoat-button-bar__button--large:active,
+:checked + .topcoat-button-bar__button {
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+.topcoat-button:focus,
+.topcoat-button--quiet:focus,
+.topcoat-button--large:focus,
+.topcoat-button--large--quiet:focus,
+.topcoat-button--cta:focus,
+.topcoat-button--large--cta:focus,
+.topcoat-button-bar__button:focus,
+.topcoat-button-bar__button--large:focus {
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+ outline: 0;
+}
+
+.topcoat-button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.topcoat-button--quiet:hover,
+.topcoat-button--large--quiet:hover {
+ text-shadow: 0 1px #fff;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px #fff;
+}
+
+.topcoat-button--quiet:active,
+.topcoat-button--large--quiet:active {
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ background-color: #d3d7d7;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+.topcoat-button--large,
+.topcoat-button--large--quiet,
+.topcoat-button-bar__button--large {
+ font-size: 1.3rem;
+ font-weight: 400;
+ line-height: 4.375rem;
+ padding: 0 1.25rem;
+}
+
+.topcoat-button--large--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.topcoat-button--cta,
+.topcoat-button--large--cta {
+ border: 1px solid #143250;
+ background-color: #288edf;
+ box-shadow: inset 0 1px rgba(255,255,255,0.36);
+ color: #fff;
+ font-weight: 500;
+ text-shadow: 0 -1px rgba(0,0,0,0.36);
+}
+
+.topcoat-button--cta:hover,
+.topcoat-button--large--cta:hover {
+ background-color: #509bef;
+}
+
+.topcoat-button--cta:active,
+.topcoat-button--large--cta:active {
+ background-color: #0380e8;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+.topcoat-button--large--cta {
+ font-size: 1.3rem;
+ font-weight: 400;
+ line-height: 4.375rem;
+ padding: 0 1.25rem;
+}
+
+.button-bar,
+.topcoat-button-bar {
+ display: table;
+ table-layout: fixed;
+ white-space: no-wrap;
+ margin: 0;
+ padding: 0;
+}
+
+.button-bar__item,
+.topcoat-button-bar__item {
+ display: table-cell;
+ width: auto;
+ border-radius: 0;
+}
+
+.button_bar__item > input,
+.topcoat-button-bar__item > input {
+ position: absolute;
+ overflow: hidden;
+ padding: 0;
+ border: 0;
+ opacity: 0.001;
+ z-index: 1;
+ vertical-align: top;
+ outline: none;
+}
+
+.button-bar__button {
+ border-radius: inherit;
+}
+
+.button-bar__item:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Button Bar
+ description: Component of grouped buttons
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+ examples:
+ mobile button bar: http://codepen.io/Topcoat/pen/kdKyg
+ tags:
+ - desktop
+ - light
+ - dark
+ - mobile
+ - button
+ - group
+ - bar
+*/
+
+.topcoat-button-bar > .topcoat-button-bar__item:first-child {
+ border-top-left-radius: 6px;
+ border-bottom-left-radius: 6px;
+}
+
+.topcoat-button-bar > .topcoat-button-bar__item:last-child {
+ border-top-right-radius: 6px;
+ border-bottom-right-radius: 6px;
+}
+
+.topcoat-button-bar__item:first-child > .topcoat-button-bar__button,
+.topcoat-button-bar__item:first-child > .topcoat-button-bar__button--large {
+ border-right: none;
+}
+
+.topcoat-button-bar__item:last-child > .topcoat-button-bar__button,
+.topcoat-button-bar__item:last-child > .topcoat-button-bar__button--large {
+ border-left: none;
+}
+
+.topcoat-button-bar__button {
+ border-radius: inherit;
+}
+
+.topcoat-button-bar__button:focus,
+.topcoat-button-bar__button--large:focus {
+ z-index: 1;
+}
+
+/* topdoc
+ name: Large Button Bar
+ description: A button bar, only larger
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+ tags:
+ - desktop
+ - light
+ - dark
+ - mobile
+ - button
+ - group
+ - bar
+ - large
+*/
+
+.topcoat-button-bar__button--large {
+ border-radius: inherit;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.button {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+ text-decoration: none;
+}
+
+.button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.button--disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.button,
+.topcoat-button,
+.topcoat-button--quiet,
+.topcoat-button--large,
+.topcoat-button--large--quiet,
+.topcoat-button--cta,
+.topcoat-button--large--cta {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+ text-decoration: none;
+}
+
+.button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.button--disabled,
+.topcoat-button:disabled,
+.topcoat-button--quiet:disabled,
+.topcoat-button--large:disabled,
+.topcoat-button--large--quiet:disabled,
+.topcoat-button--cta:disabled,
+.topcoat-button--large--cta:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Button
+ description: A simple button
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+ Button
+ Button
+ examples:
+ mobile button: http://codepen.io/Topcoat/pen/DpKtf
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+*/
+
+.topcoat-button,
+.topcoat-button--quiet,
+.topcoat-button--large,
+.topcoat-button--large--quiet,
+.topcoat-button--cta,
+.topcoat-button--large--cta {
+ padding: 0 1.25rem;
+ font-size: 16px;
+ line-height: 3rem;
+ letter-spacing: 1px;
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ vertical-align: top;
+ background-color: #e5e9e8;
+ box-shadow: inset 0 1px #fff;
+ border: 1px solid #a5a8a8;
+ border-radius: 6px;
+}
+
+.topcoat-button:hover,
+.topcoat-button--quiet:hover,
+.topcoat-button--large:hover,
+.topcoat-button--large--quiet:hover {
+ background-color: #edf1f1;
+}
+
+.topcoat-button:active,
+.topcoat-button--large:active {
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+.topcoat-button:focus,
+.topcoat-button--quiet:focus,
+.topcoat-button--large:focus,
+.topcoat-button--large--quiet:focus,
+.topcoat-button--cta:focus,
+.topcoat-button--large--cta:focus {
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+ outline: 0;
+}
+
+/* topdoc
+ name: Quiet Button
+ description: A simple, yet quiet button
+ modifiers:
+ :active: Quiet button active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+ Button
+ Button
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - quiet
+*/
+
+.topcoat-button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.topcoat-button--quiet:hover,
+.topcoat-button--large--quiet:hover {
+ text-shadow: 0 1px #fff;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px #fff;
+}
+
+.topcoat-button--quiet:active,
+.topcoat-button--large--quiet:active {
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ background-color: #d3d7d7;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+/* topdoc
+ name: Large Button
+ description: A big ol button
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+ Button
+ Button
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - large
+*/
+
+.topcoat-button--large,
+.topcoat-button--large--quiet {
+ font-size: 1.3rem;
+ font-weight: 400;
+ line-height: 4.375rem;
+ padding: 0 1.25rem;
+}
+
+/* topdoc
+ name: Large Quiet Button
+ description: A large, yet quiet button
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+ Button
+ Button
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - large
+ - quiet
+*/
+
+.topcoat-button--large--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+/* topdoc
+ name: Call To Action Button
+ description: A CALL TO ARMS, er, ACTION!
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+ Button
+ Button
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - call to action
+*/
+
+.topcoat-button--cta,
+.topcoat-button--large--cta {
+ border: 1px solid #143250;
+ background-color: #288edf;
+ box-shadow: inset 0 1px rgba(255,255,255,0.36);
+ color: #fff;
+ font-weight: 500;
+ text-shadow: 0 -1px rgba(0,0,0,0.36);
+}
+
+.topcoat-button--cta:hover,
+.topcoat-button--large--cta:hover {
+ background-color: #509bef;
+}
+
+.topcoat-button--cta:active,
+.topcoat-button--large--cta:active {
+ background-color: #0380e8;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+/* topdoc
+ name: Large Call To Action Button
+ description: Like call to action, but bigger
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+ Button
+ Button
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - large
+ - call to action
+*/
+
+.topcoat-button--large--cta {
+ font-size: 1.3rem;
+ font-weight: 400;
+ line-height: 4.375rem;
+ padding: 0 1.25rem;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+input[type="checkbox"] {
+ position: absolute;
+ overflow: hidden;
+ padding: 0;
+ border: 0;
+ opacity: 0.001;
+ z-index: 1;
+ vertical-align: top;
+ outline: none;
+}
+
+.checkbox {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.checkbox__label {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.checkbox--disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+.checkbox:before,
+.checkbox:after {
+ content: '';
+ position: absolute;
+}
+
+.checkbox:before {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+input[type="checkbox"] {
+ position: absolute;
+ overflow: hidden;
+ padding: 0;
+ border: 0;
+ opacity: 0.001;
+ z-index: 1;
+ vertical-align: top;
+ outline: none;
+}
+
+.checkbox,
+.topcoat-checkbox__checkmark {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.checkbox__label,
+.topcoat-checkbox {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.checkbox--disabled,
+input[type="checkbox"]:disabled + .topcoat-checkbox__checkmark {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+.checkbox:before,
+.checkbox:after,
+.topcoat-checkbox__checkmark:before,
+.topcoat-checkbox__checkmark:after {
+ content: '';
+ position: absolute;
+}
+
+.checkbox:before,
+.topcoat-checkbox__checkmark:before {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+}
+
+/* topdoc
+ name: Checkbox
+ description: Default skin for Topcoat checkbox
+ modifiers:
+ :focus: Focus state
+ :disabled: Disabled state
+ markup:
+
+
+
+ Default
+
+
+
+
+
+
+ Disabled
+
+ examples:
+ mobile checkbox: http://codepen.io/Topcoat/pen/piHcs
+ tags:
+ - desktop
+ - light
+ - mobile
+ - checkbox
+*/
+
+.topcoat-checkbox__checkmark {
+ height: 2rem;
+}
+
+input[type="checkbox"] {
+ height: 2rem;
+ width: 2rem;
+ margin-top: 0;
+ margin-right: -2rem;
+ margin-bottom: -2rem;
+ margin-left: 0;
+}
+
+input[type="checkbox"]:checked + .topcoat-checkbox__checkmark:after {
+ opacity: 1;
+}
+
+.topcoat-checkbox {
+ line-height: 2rem;
+}
+
+.topcoat-checkbox__checkmark:before {
+ width: 2rem;
+ height: 2rem;
+ background: #e5e9e8;
+ border: 1px solid #a5a8a8;
+ border-radius: 3px;
+ box-shadow: inset 0 1px #fff;
+}
+
+.topcoat-checkbox__checkmark {
+ width: 2rem;
+ height: 2rem;
+}
+
+.topcoat-checkbox__checkmark:after {
+ top: 1px;
+ left: 2px;
+ opacity: 0;
+ width: 28px;
+ height: 11px;
+ background: transparent;
+ border: 7px solid #666;
+ border-width: 7px;
+ border-top: none;
+ border-right: none;
+ border-radius: 2px;
+ -webkit-transform: rotate(-50deg);
+ -ms-transform: rotate(-50deg);
+ transform: rotate(-50deg);
+}
+
+input[type="checkbox"]:focus + .topcoat-checkbox__checkmark:before {
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.button,
+.topcoat-icon-button,
+.topcoat-icon-button--quiet,
+.topcoat-icon-button--large,
+.topcoat-icon-button--large--quiet {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+ text-decoration: none;
+}
+
+.button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.button--disabled,
+.topcoat-icon-button:disabled,
+.topcoat-icon-button--quiet:disabled,
+.topcoat-icon-button--large:disabled,
+.topcoat-icon-button--large--quiet:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Icon Button
+ description: Like button, but it has an icon.
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+
+
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - icon
+*/
+
+.topcoat-icon-button,
+.topcoat-icon-button--quiet,
+.topcoat-icon-button--large,
+.topcoat-icon-button--large--quiet {
+ padding: 0 0.75rem;
+ line-height: 3rem;
+ letter-spacing: 1px;
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ vertical-align: baseline;
+ background-color: #e5e9e8;
+ box-shadow: inset 0 1px #fff;
+ border: 1px solid #a5a8a8;
+ border-radius: 6px;
+}
+
+.topcoat-icon-button:hover,
+.topcoat-icon-button--quiet:hover,
+.topcoat-icon-button--large:hover,
+.topcoat-icon-button--large--quiet:hover {
+ background-color: #edf1f1;
+}
+
+.topcoat-icon-button:active {
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+.topcoat-icon-button:focus,
+.topcoat-icon-button--quiet:focus,
+.topcoat-icon-button--quiet:hover:focus,
+.topcoat-icon-button--large:focus,
+.topcoat-icon-button--large--quiet:focus,
+.topcoat-icon-button--large--quiet:hover:focus {
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+ outline: 0;
+}
+
+/* topdoc
+ name: Quiet Icon Button
+ description: Like quiet button, but it has an icon.
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+
+
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - icon
+ - quiet
+*/
+
+.topcoat-icon-button--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.topcoat-icon-button--quiet:hover,
+.topcoat-icon-button--large--quiet:hover {
+ text-shadow: 0 1px #fff;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px #fff;
+}
+
+.topcoat-icon-button--quiet:active,
+.topcoat-icon-button--large--quiet:active {
+ color: #454545;
+ text-shadow: 0 1px #fff;
+ background-color: #d3d7d7;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+/* topdoc
+ name: Large Icon Button
+ description: Like large button, but it has an icon.
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ :focus: Focused
+ markup:
+
+
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - icon
+ - large
+*/
+
+.topcoat-icon-button--large,
+.topcoat-icon-button--large--quiet {
+ width: 4.375rem;
+ height: 4.375rem;
+ line-height: 4.375rem;
+}
+
+.topcoat-icon-button--large:active {
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+}
+
+/* topdoc
+ name: Large Quiet Icon Button
+ description: Like large button, but it has an icon and this one is quiet.
+ modifiers:
+ :active: Active state
+ :disabled: Disabled state
+ :hover: Hover state
+ markup:
+
+
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - button
+ - icon
+ - large
+ - quiet
+*/
+
+.topcoat-icon-button--large--quiet {
+ background: transparent;
+ border: 1px solid transparent;
+ box-shadow: none;
+}
+
+.topcoat-icon,
+.topcoat-icon--large {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ overflow: hidden;
+ width: 1.62rem;
+ height: 1.62rem;
+ vertical-align: middle;
+ top: -1px;
+}
+
+.topcoat-icon--large {
+ width: 2.499999998125rem;
+ height: 2.499999998125rem;
+ top: -2px;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.input {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ vertical-align: top;
+ outline: none;
+}
+
+.input:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.list {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ overflow: auto;
+ -webkit-overflow-scrolling: touch;
+}
+
+.list__header {
+ margin: 0;
+}
+
+.list__container {
+ padding: 0;
+ margin: 0;
+ list-style-type: none;
+}
+
+.list__item {
+ margin: 0;
+ padding: 0;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.list,
+.topcoat-list {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ overflow: auto;
+ -webkit-overflow-scrolling: touch;
+}
+
+.list__header,
+.topcoat-list__header {
+ margin: 0;
+}
+
+.list__container,
+.topcoat-list__container {
+ padding: 0;
+ margin: 0;
+ list-style-type: none;
+}
+
+.list__item,
+.topcoat-list__item {
+ margin: 0;
+ padding: 0;
+}
+
+/* topdoc
+ name: List
+ description: Topcoat default list skin
+ markup:
+
+
+
+
+ Item
+
+
+ Item
+
+
+ Item
+
+
+
+ tags:
+ - mobile
+ - list
+*/
+
+.topcoat-list {
+ border-top: 1px solid #bcbfbf;
+ border-bottom: 1px solid #eff1f1;
+ background-color: #dfe2e2;
+}
+
+.topcoat-list__header {
+ padding: 4px 20px;
+ font-size: 0.9em;
+ font-weight: 400;
+ background-color: #cccfcf;
+ color: #656565;
+ text-shadow: 0 1px 0 rgba(255,255,255,0.5);
+ border-top: 1px solid rgba(255,255,255,0.5);
+ border-bottom: 1px solid rgba(255,255,255,0.23);
+}
+
+.topcoat-list__container {
+ border-top: 1px solid #bcbfbf;
+ color: #454545;
+}
+
+.topcoat-list__item {
+ padding: 1.25rem;
+ border-top: 1px solid #eff1f1;
+ border-bottom: 1px solid #bcbfbf;
+}
+
+.topcoat-list__item:first-child {
+ border-top: 1px solid rgba(0,0,0,0.05);
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.navigation-bar {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ white-space: nowrap;
+ overflow: hidden;
+ word-spacing: 0;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.navigation-bar__item {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+}
+
+.navigation-bar__title {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.navigation-bar,
+.topcoat-navigation-bar {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ white-space: nowrap;
+ overflow: hidden;
+ word-spacing: 0;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.navigation-bar__item,
+.topcoat-navigation-bar__item {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+}
+
+.navigation-bar__title,
+.topcoat-navigation-bar__title {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+}
+
+/* topdoc
+ name: Navigation Bar
+ description: A place where navigation goes to drink
+ markup:
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - navigation
+ - bar
+*/
+
+.topcoat-navigation-bar {
+ height: 4.375rem;
+ padding-left: 1rem;
+ padding-right: 1rem;
+ background: #e5e9e8;
+ color: #000;
+ box-shadow: inset 0 -1px #b9bcbc, 0 1px #d4d6d6;
+}
+
+.topcoat-navigation-bar__item {
+ margin: 0;
+ line-height: 4.375rem;
+ vertical-align: top;
+}
+
+.topcoat-navigation-bar__title {
+ font-size: 1.3rem;
+ font-weight: 400;
+ color: #000;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+input[type="radio"] {
+ position: absolute;
+ overflow: hidden;
+ padding: 0;
+ border: 0;
+ opacity: 0.001;
+ z-index: 1;
+ vertical-align: top;
+ outline: none;
+}
+
+.radio-button {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.radio-button__label {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.radio-button:before,
+.radio-button:after {
+ content: '';
+ position: absolute;
+ border-radius: 100%;
+}
+
+.radio-button:after {
+ top: 50%;
+ left: 50%;
+ -webkit-transform: translate(-50%, -50%);
+ -ms-transform: translate(-50%, -50%);
+ transform: translate(-50%, -50%);
+}
+
+.radio-button:before {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+}
+
+.radio-button--disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+input[type="radio"] {
+ position: absolute;
+ overflow: hidden;
+ padding: 0;
+ border: 0;
+ opacity: 0.001;
+ z-index: 1;
+ vertical-align: top;
+ outline: none;
+}
+
+.radio-button,
+.topcoat-radio-button__checkmark {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.radio-button__label,
+.topcoat-radio-button {
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+}
+
+.radio-button:before,
+.radio-button:after,
+.topcoat-radio-button__checkmark:before,
+.topcoat-radio-button__checkmark:after {
+ content: '';
+ position: absolute;
+ border-radius: 100%;
+}
+
+.radio-button:after,
+.topcoat-radio-button__checkmark:after {
+ top: 50%;
+ left: 50%;
+ -webkit-transform: translate(-50%, -50%);
+ -ms-transform: translate(-50%, -50%);
+ transform: translate(-50%, -50%);
+}
+
+.radio-button:before,
+.topcoat-radio-button__checkmark:before {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+}
+
+.radio-button--disabled,
+input[type="radio"]:disabled + .topcoat-radio-button__checkmark {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Radio Button
+ description: A button that can play music, but usually just plays ads.
+ modifiers:
+ markup:
+
+
+
+
+
+
+
+
+
+ Left label
+
+
+
+
+
+
+
+
+
+ Right label
+
+
+
+
+
+
+
+ Disabled
+
+ examples:
+ Mobile Radio Button: http://codepen.io/Topcoat/pen/HDcJj
+ tags:
+ - desktop
+ - light
+ - mobile
+ - Radio
+*/
+
+input[type="radio"] {
+ height: 1.875rem;
+ width: 1.875rem;
+ margin-top: 0;
+ margin-right: -1.875rem;
+ margin-bottom: -1.875rem;
+ margin-left: 0;
+}
+
+input[type="radio"]:checked + .topcoat-radio-button__checkmark:after {
+ opacity: 1;
+}
+
+.topcoat-radio-button {
+ color: #454545;
+ line-height: 1.875rem;
+}
+
+.topcoat-radio-button__checkmark:before {
+ width: 1.875rem;
+ height: 1.875rem;
+ background: #e5e9e8;
+ border: 1px solid #a5a8a8;
+ box-shadow: inset 0 1px #fff;
+}
+
+.topcoat-radio-button__checkmark {
+ position: relative;
+ width: 1.875rem;
+ height: 1.875rem;
+}
+
+.topcoat-radio-button__checkmark:after {
+ opacity: 0;
+ width: 0.875rem;
+ height: 0.875rem;
+ background: #666;
+ border: 1px solid rgba(0,0,0,0.1);
+ box-shadow: 0 1px rgba(255,255,255,0.5);
+ -webkit-transform: none;
+ -ms-transform: none;
+ transform: none;
+ top: 7px;
+ left: 7px;
+}
+
+input[type="radio"]:focus + .topcoat-radio-button__checkmark:before {
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.search-input {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ vertical-align: top;
+ outline: none;
+ -webkit-appearance: none;
+}
+
+input[type="search"]::-webkit-search-cancel-button {
+ -webkit-appearance: none;
+}
+
+.search-input:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.search-input,
+.topcoat-search-input,
+.topcoat-search-input--large {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ vertical-align: top;
+ outline: none;
+ -webkit-appearance: none;
+}
+
+input[type="search"]::-webkit-search-cancel-button {
+ -webkit-appearance: none;
+}
+
+.search-input:disabled,
+.topcoat-search-input:disabled,
+.topcoat-search-input--large:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Search Input
+ description: A text input designed for searching.
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - text
+ - input
+ - search
+ - form
+*/
+
+.topcoat-search-input,
+.topcoat-search-input--large {
+ line-height: 3rem;
+ font-size: 16px;
+ border: 1px solid #a5a8a8;
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+ color: #454545;
+ padding: 0 0 0 2rem;
+ border-radius: 30px;
+ background-image: url("../img/search.svg");
+ background-position: 1em center;
+ background-repeat: no-repeat;
+ background-size: 16px;
+}
+
+.topcoat-search-input:focus,
+.topcoat-search-input--large:focus {
+ background-image: url("../img/search_dark.svg");
+ background-color: #edf1f1;
+ color: #000;
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+}
+
+.topcoat-search-input::-webkit-search-cancel-button,
+.topcoat-search-input::-webkit-search-decoration,
+.topcoat-search-input--large::-webkit-search-cancel-button,
+.topcoat-search-input--large::-webkit-search-decoration {
+ margin-right: 5px;
+}
+
+.topcoat-search-input:focus::-webkit-input-placeholder,
+.topcoat-search-input:focus::-webkit-input-placeholder {
+ color: #c6c8c8;
+}
+
+.topcoat-search-input:disabled::-webkit-input-placeholder {
+ color: #000;
+}
+
+.topcoat-search-input:disabled::-moz-placeholder {
+ color: #000;
+}
+
+.topcoat-search-input:disabled:-ms-input-placeholder {
+ color: #000;
+}
+
+/* topdoc
+ name: Large Search Input
+ description: A large text input designed for searching.
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - text
+ - input
+ - search
+ - form
+ - large
+*/
+
+.topcoat-search-input--large {
+ line-height: 4.375rem;
+ font-size: 1.3rem;
+ font-weight: 200;
+ padding: 0 0 0 2.9rem;
+ border-radius: 40px;
+ background-position: 1.2em center;
+ background-size: 1.3rem;
+}
+
+.topcoat-search-input--large:disabled {
+ color: #000;
+}
+
+.topcoat-search-input--large:disabled::-webkit-input-placeholder {
+ color: #000;
+}
+
+.topcoat-search-input--large:disabled::-moz-placeholder {
+ color: #000;
+}
+
+.topcoat-search-input--large:disabled:-ms-input-placeholder {
+ color: #000;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.input,
+.topcoat-text-input,
+.topcoat-text-input--large {
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ vertical-align: top;
+ outline: none;
+}
+
+.input:disabled,
+.topcoat-text-input:disabled,
+.topcoat-text-input--large:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Text input
+ description: Topdoc text input
+ markup:
+
+
+
+
+ tags:
+ - desktop
+ - mobile
+ - text
+ - input
+*/
+
+.topcoat-text-input,
+.topcoat-text-input--large {
+ line-height: 3rem;
+ font-size: 16px;
+ letter-spacing: 1px;
+ padding: 0 1.25rem;
+ border: 1px solid #a5a8a8;
+ border-radius: 6px;
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+ color: #454545;
+ vertical-align: top;
+}
+
+.topcoat-text-input:focus,
+.topcoat-text-input--large:focus {
+ background-color: #edf1f1;
+ color: #000;
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+}
+
+.topcoat-text-input:disabled::-webkit-input-placeholder {
+ color: #000;
+}
+
+.topcoat-text-input:disabled::-moz-placeholder {
+ color: #000;
+}
+
+.topcoat-text-input:disabled:-ms-input-placeholder {
+ color: #000;
+}
+
+/* topdoc
+ name: Large Text Input
+ description: A bigger input, still for text.
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - form
+ - input
+ - large
+*/
+
+.topcoat-text-input--large {
+ line-height: 4.375rem;
+ font-size: 1.3rem;
+}
+
+.topcoat-text-input--large:disabled {
+ color: #000;
+}
+
+.topcoat-text-input--large:disabled::-webkit-input-placeholder {
+ color: #000;
+}
+
+.topcoat-text-input--large:disabled::-moz-placeholder {
+ color: #000;
+}
+
+.topcoat-text-input--large:disabled:-ms-input-placeholder {
+ color: #000;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.textarea {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ vertical-align: top;
+ resize: none;
+ outline: none;
+}
+
+.textarea:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+/**
+*
+* Copyright 2012 Adobe Systems Inc.;
+*
+* Licensed under the Apache License, Version 2.0 (the "License");
+* you may not use this file except in compliance with the License.
+* You may obtain a copy of the License at
+*
+* http://www.apache.org/licenses/LICENSE-2.0
+*
+* Unless required by applicable law or agreed to in writing, software
+* distributed under the License is distributed on an "AS IS" BASIS,
+* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+* See the License for the specific language governing permissions and
+* limitations under the License.
+*
+*/
+
+.textarea,
+.topcoat-textarea,
+.topcoat-textarea--large {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ vertical-align: top;
+ resize: none;
+ outline: none;
+}
+
+.textarea:disabled,
+.topcoat-textarea:disabled,
+.topcoat-textarea--large:disabled {
+ opacity: 0.3;
+ cursor: default;
+ pointer-events: none;
+}
+
+/* topdoc
+ name: Textarea
+ description: A whole area, just for text.
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - form
+ - input
+ - textarea
+*/
+
+.topcoat-textarea,
+.topcoat-textarea--large {
+ padding: 2rem;
+ font-size: 2.5rem;
+ font-weight: 200;
+ border-radius: 6px;
+ line-height: 3rem;
+ border: 1px solid #a5a8a8;
+ background-color: #d3d7d7;
+ box-shadow: inset 0 1px rgba(0,0,0,0.12);
+ color: #454545;
+ letter-spacing: 1px;
+}
+
+.topcoat-textarea:focus,
+.topcoat-textarea--large:focus {
+ background-color: #edf1f1;
+ color: #000;
+ border: 1px solid #0940fd;
+ box-shadow: 0 0 0 2px #6fb5f1;
+}
+
+.topcoat-textarea:disabled::-webkit-input-placeholder {
+ color: #000;
+}
+
+.topcoat-textarea:disabled::-moz-placeholder {
+ color: #000;
+}
+
+.topcoat-textarea:disabled:-ms-input-placeholder {
+ color: #000;
+}
+
+/* topdoc
+ name: Large Textarea
+ description: A whole area, just for text; now available in large.
+ modifiers:
+ :disabled: Disabled state
+ markup:
+
+
+
+
+ tags:
+ - desktop
+ - light
+ - mobile
+ - form
+ - input
+ - textarea
+*/
+
+.topcoat-textarea--large {
+ font-size: 3rem;
+ line-height: 4.375rem;
+}
+
+.topcoat-textarea--large:disabled {
+ color: #000;
+}
+
+.topcoat-textarea--large:disabled::-webkit-input-placeholder {
+ color: #000;
+}
+
+.topcoat-textarea--large:disabled::-moz-placeholder {
+ color: #000;
+}
+
+.topcoat-textarea--large:disabled:-ms-input-placeholder {
+ color: #000;
+}
+
+@font-face {
+ font-family: "Source Sans";
+ src: url("../font/SourceSansPro-Regular.otf");
+}
+
+@font-face {
+ font-family: "Source Sans";
+ src: url("../font/SourceSansPro-Light.otf");
+ font-weight: 200;
+}
+
+@font-face {
+ font-family: "Source Sans";
+ src: url("../font/SourceSansPro-Semibold.otf");
+ font-weight: 600;
+}
+
+body {
+ margin: 0;
+ padding: 0;
+ background: #dfe2e2;
+ color: #000;
+ font: 16px "Source Sans", helvetica, arial, sans-serif;
+ font-weight: 200;
+ text-rendering: optimizeLegibility;
+}
+
+:focus {
+ outline-color: transparent;
+ outline-style: none;
+}
+
+.topcoat-icon--menu-stack {
+ background: url("../img/hamburger_dark.svg") no-repeat;
+ background-size: cover;
+}
+
+.quarter {
+ width: 25%;
+}
+
+.half {
+ width: 50%;
+}
+
+.three-quarters {
+ width: 75%;
+}
+
+.third {
+ width: 33.333%;
+}
+
+.two-thirds {
+ width: 66.666%;
+}
+
+.full {
+ width: 100%;
+}
+
+.left {
+ text-align: left;
+}
+
+.center {
+ text-align: center;
+}
+
+.right {
+ text-align: right;
+}
+
+.reset-ui {
+ -moz-box-sizing: border-box;
+ box-sizing: border-box;
+ background-clip: padding-box;
+ position: relative;
+ display: inline-block;
+ vertical-align: top;
+ padding: 0;
+ margin: 0;
+ font: inherit;
+ color: inherit;
+ background: transparent;
+ border: none;
+ cursor: default;
+ -webkit-user-select: none;
+ -moz-user-select: none;
+ -ms-user-select: none;
+ user-select: none;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ overflow: hidden;
+}
+
+/* Call To Action */
+
+/* Icons */
+
+/* Navigation Bar */
+
+/* Text Input */
+
+/* Search Input */
+
+/* List */
+
+/* Checkbox */
+
+/* Overlay */
+
+/* Progress bar */
+
+/* Checkbox */
+
+/* Radio Button */
+
+/* Icon Button */
+
+/* Navigation bar */
+
+/* List */
+
+/* Search Input */
+
+/* Textarea */
+
+/* Checkbox */
+
+/* Radio */
+
+/* Search Input */
+
+/* Call To Action */
+
+/* Icons */
+
+/* Navigation Bar */
+
+/* Text Input */
+
+/* List */
+
+/*Overlay*/
+
+/* Progress bar */
+
+/* Checkbox */
+
+/* Range input */
+
+/* Containers */
+
+/* Icon Button */
+
+/* Navigation bar */
+
+/* List */
+
+/* Search Input */
+
+/* Text Area */
+
+/* Checkbox */
+
+/* Radio */
+
+/* Range input */
+
+/* Search Input */
+
+/* Text Input */
+
+/* Radio input */
+
+/* Overlay */
+
+/* Textarea */
+
+/* Progress bar container */
+
+/* Progress bar progress */
+
+/* Search input */
\ No newline at end of file
diff --git a/docs/fonts/sourcecodepro-regular-webfont.eot b/docs/fonts/sourcecodepro-regular-webfont.eot
new file mode 100644
index 0000000..9e9e4de
Binary files /dev/null and b/docs/fonts/sourcecodepro-regular-webfont.eot differ
diff --git a/docs/fonts/sourcecodepro-regular-webfont.svg b/docs/fonts/sourcecodepro-regular-webfont.svg
new file mode 100644
index 0000000..f4a1739
--- /dev/null
+++ b/docs/fonts/sourcecodepro-regular-webfont.svg
@@ -0,0 +1,242 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/docs/fonts/sourcecodepro-regular-webfont.ttf b/docs/fonts/sourcecodepro-regular-webfont.ttf
new file mode 100644
index 0000000..6eb48e7
Binary files /dev/null and b/docs/fonts/sourcecodepro-regular-webfont.ttf differ
diff --git a/docs/fonts/sourcecodepro-regular-webfont.woff b/docs/fonts/sourcecodepro-regular-webfont.woff
new file mode 100644
index 0000000..2383f47
Binary files /dev/null and b/docs/fonts/sourcecodepro-regular-webfont.woff differ
diff --git a/docs/fonts/sourcesanspro-light-webfont.eot b/docs/fonts/sourcesanspro-light-webfont.eot
new file mode 100644
index 0000000..bda2005
Binary files /dev/null and b/docs/fonts/sourcesanspro-light-webfont.eot differ
diff --git a/docs/fonts/sourcesanspro-light-webfont.svg b/docs/fonts/sourcesanspro-light-webfont.svg
new file mode 100644
index 0000000..e031390
--- /dev/null
+++ b/docs/fonts/sourcesanspro-light-webfont.svg
@@ -0,0 +1,243 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/docs/fonts/sourcesanspro-light-webfont.ttf b/docs/fonts/sourcesanspro-light-webfont.ttf
new file mode 100644
index 0000000..0959ece
Binary files /dev/null and b/docs/fonts/sourcesanspro-light-webfont.ttf differ
diff --git a/docs/fonts/sourcesanspro-light-webfont.woff b/docs/fonts/sourcesanspro-light-webfont.woff
new file mode 100644
index 0000000..522d5ab
Binary files /dev/null and b/docs/fonts/sourcesanspro-light-webfont.woff differ
diff --git a/docs/fonts/sourcesanspro-regular-webfont.eot b/docs/fonts/sourcesanspro-regular-webfont.eot
new file mode 100644
index 0000000..2b75abb
Binary files /dev/null and b/docs/fonts/sourcesanspro-regular-webfont.eot differ
diff --git a/docs/fonts/sourcesanspro-regular-webfont.svg b/docs/fonts/sourcesanspro-regular-webfont.svg
new file mode 100644
index 0000000..581a849
--- /dev/null
+++ b/docs/fonts/sourcesanspro-regular-webfont.svg
@@ -0,0 +1,243 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/docs/fonts/sourcesanspro-regular-webfont.ttf b/docs/fonts/sourcesanspro-regular-webfont.ttf
new file mode 100644
index 0000000..e166286
Binary files /dev/null and b/docs/fonts/sourcesanspro-regular-webfont.ttf differ
diff --git a/docs/fonts/sourcesanspro-regular-webfont.woff b/docs/fonts/sourcesanspro-regular-webfont.woff
new file mode 100644
index 0000000..315c98a
Binary files /dev/null and b/docs/fonts/sourcesanspro-regular-webfont.woff differ
diff --git a/docs/fonts/sourcesanspro-semibold-webfont.eot b/docs/fonts/sourcesanspro-semibold-webfont.eot
new file mode 100644
index 0000000..ddf5d11
Binary files /dev/null and b/docs/fonts/sourcesanspro-semibold-webfont.eot differ
diff --git a/docs/fonts/sourcesanspro-semibold-webfont.svg b/docs/fonts/sourcesanspro-semibold-webfont.svg
new file mode 100644
index 0000000..317e536
--- /dev/null
+++ b/docs/fonts/sourcesanspro-semibold-webfont.svg
@@ -0,0 +1,243 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/docs/fonts/sourcesanspro-semibold-webfont.ttf b/docs/fonts/sourcesanspro-semibold-webfont.ttf
new file mode 100644
index 0000000..6d97e7b
Binary files /dev/null and b/docs/fonts/sourcesanspro-semibold-webfont.ttf differ
diff --git a/docs/fonts/sourcesanspro-semibold-webfont.woff b/docs/fonts/sourcesanspro-semibold-webfont.woff
new file mode 100644
index 0000000..d844315
Binary files /dev/null and b/docs/fonts/sourcesanspro-semibold-webfont.woff differ
diff --git a/docs/fonts/stylesheet.css b/docs/fonts/stylesheet.css
new file mode 100644
index 0000000..2d66502
--- /dev/null
+++ b/docs/fonts/stylesheet.css
@@ -0,0 +1,57 @@
+
+@font-face {
+ font-family: 'source-sans-pro';
+ src: url('sourcesanspro-light-webfont.eot');
+ src: url('sourcesanspro-light-webfont.eot?#iefix') format('embedded-opentype'),
+ url('sourcesanspro-light-webfont.woff') format('woff'),
+ url('sourcesanspro-light-webfont.ttf') format('truetype'),
+ url('sourcesanspro-light-webfont.svg#source_sans_prolight') format('svg');
+ font-weight: 300;
+ font-style: normal;
+
+}
+
+
+
+
+@font-face {
+ font-family: 'source-sans-pro';
+ src: url('sourcesanspro-regular-webfont.eot');
+ src: url('sourcesanspro-regular-webfont.eot?#iefix') format('embedded-opentype'),
+ url('sourcesanspro-regular-webfont.woff') format('woff'),
+ url('sourcesanspro-regular-webfont.ttf') format('truetype'),
+ url('sourcesanspro-regular-webfont.svg#source_sans_proregular') format('svg');
+ font-weight: 400;
+ font-style: normal;
+
+}
+
+
+
+
+@font-face {
+ font-family: 'source-sans-pro';
+ src: url('sourcesanspro-semibold-webfont.eot');
+ src: url('sourcesanspro-semibold-webfont.eot?#iefix') format('embedded-opentype'),
+ url('sourcesanspro-semibold-webfont.woff') format('woff'),
+ url('sourcesanspro-semibold-webfont.ttf') format('truetype'),
+ url('sourcesanspro-semibold-webfont.svg#source_sans_prosemibold') format('svg');
+ font-weight: 600;
+ font-style: normal;
+
+}
+
+
+
+
+@font-face {
+ font-family: 'source-code-pro';
+ src: url('sourcecodepro-regular-webfont.eot');
+ src: url('sourcecodepro-regular-webfont.eot?#iefix') format('embedded-opentype'),
+ url('sourcecodepro-regular-webfont.woff') format('woff'),
+ url('sourcecodepro-regular-webfont.ttf') format('truetype'),
+ url('sourcecodepro-regular-webfont.svg#source_code_proregular') format('svg');
+ font-weight: normal;
+ font-style: normal;
+
+}
\ No newline at end of file
diff --git a/docs/img/search.svg b/docs/img/search.svg
new file mode 100644
index 0000000..d18a4fa
--- /dev/null
+++ b/docs/img/search.svg
@@ -0,0 +1,11 @@
+
+
+ Slice 1
+ Created with Sketch (http://www.bohemiancoding.com/sketch)
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/docs/img/search_dark.svg b/docs/img/search_dark.svg
new file mode 100644
index 0000000..cbfae91
--- /dev/null
+++ b/docs/img/search_dark.svg
@@ -0,0 +1,16 @@
+
+
+
+
+
+
+]>
+
+
+
diff --git a/docs/index.html b/docs/index.html
new file mode 100644
index 0000000..f68187f
--- /dev/null
+++ b/docs/index.html
@@ -0,0 +1,2393 @@
+
+
+
+
+
+ RedRaphaël API Reference
+
+
+
+
+
+
+
+
+
+
+
+
+
+ RedRaphaël
+ API Reference
+
+
+
+
+
+*
+ Creates a copy of existing animation object with given delay.
+*
+
+
Parameters *
+
+
delay
+number
+number of ms to pass between animation start and actual animation
+
+
*
+
+
Returns: object new altered Animation object
+
var anim = Raphael.animation({cx: 10, cy: 20}, 2e3);
+ circle1.animate(anim); // run the given animation immediately
+ circle2.animate(anim.delay(500)); // run the given animation after 500 ms
+
+
+*
+ Creates a copy of existing animation object with given repetition.
+*
+
+
Parameters *
+
+
repeat
+number
+number iterations of animation. For infinite animation pass Infinity
+
+
*
+
+
Returns: object new altered Animation object
+
+
+*
+ Creates and starts animation for given element.
+*
+
+
Parameters *
+
+
params
+object
+final attributes for the element, see also Element.attr
+ ms
+number
+number of milliseconds for animation to run
+ easing
+optional
+string
+easing type. Accept one of Raphael.easing_formulas or CSS format: cubic‐bezier(XX, XX, XX, XX)
+ callback
+optional
+function
+callback function. Will be called at the end of animation.
+
+
or
+
+
+
*
+
+
Returns: object original element
+
+*
+ Acts similar to Element.animate , but ensure that given animation runs in sync with another given element.
+*
+
+
Parameters *
+
+
el
+object
+element to sync with
+ anim
+object
+animation to sync with
+ params
+optional
+object
+final attributes for the element, see also Element.attr
+ ms
+optional
+number
+number of milliseconds for animation to run
+ easing
+optional
+string
+easing type. Accept on of Raphael.easing_formulas or CSS format: cubic‐bezier(XX, XX, XX, XX)
+ callback
+optional
+function
+callback function. Will be called at the end of animation.
+
+
or
+
+
element
+object
+element to sync with
+ anim
+object
+animation to sync with
+ animation
+optional
+object
+animation object, see Raphael.animation
+
+
*
+
+
Returns: object original element
+
+*
+ Adds event handler for click for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+
+
Returns: object clone of a given element
+
*
+
+
+*
+ Adds or retrieves given value asociated with given key.
+*
+ See also Element.removeData
+
+
Parameters key
+string
+key to store data
+ value
+optional
+any
+value to store
+
+
Returns: object Element
+
or, if value is not specified:
+
+
Returns: any value
+
Usage for (var i = 0, i < 5, i++) {
+ paper.circle(10 + 15 * i, 10, 10)
+ .attr({fill: "#000"})
+ .data("i", i)
+ .click(function () {
+ alert(this.data("i"));
+ });
+ }
+
+
+*
+ Adds event handler for double click for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handlers for drag of the element.
+
+
Parameters onmove
+function
+handler for moving
+ onstart
+function
+handler for drag start
+ onend
+function
+handler for drag end
+ mcontext
+optional
+object
+context for moving handler
+ scontext
+optional
+object
+context for drag start handler
+ econtext
+optional
+object
+context for drag end handler
+
+
Additionaly following drag
events will be triggered: drag.start.<id>
on start,
+ drag.end.<id>
on end and drag.move.<id>
on every move. When element will be dragged over another element
+ drag.over.<id>
will be fired as well.
+
+
Start event and start handler will be called in specified context or in context of the element with following parameters:
+
+
x number x position of the mouse
+y number y position of the mouse
+event object DOM event object
+
+
Move event and move handler will be called in specified context or in context of the element with following parameters:
+
+
dx number shift by x from the start point
+dy number shift by y from the start point
+x number x position of the mouse
+y number y position of the mouse
+event object DOM event object
+
+
End event and end handler will be called in specified context or in context of the element with following parameters:
+
+
event object DOM event object
+
+
Returns: object Element
+
+*
+ Return bounding box for a given element
+*
+
+
Parameters *
+
+
isWithoutTransform
+boolean
+flag, true
if you want to have bounding box before transformations. Default is false
.
+
+
Returns: object Bounding box object:
+
{x: number top left corner x
+y: number top left corner y
+x2: number bottom right corner x
+y2: number bottom right corner y
+width: number width
+height: number height
+ }
+
+*
+ Retrieves the element data
+
+
Returns: object data
+
+*
+ Return set of elements that create glow-like effect around given element. See Paper.set .
+
+
Note: Glow is not connected to the element. If you change element attributes it won’t adjust itself.
+*
+
+
Parameters *
+
+
glow
+optional
+object
+parameters object with all properties optional:
+
+
{width number size of the glow, default is 10
+fill boolean will it be filled, default is false
+opacity number opacity, default is 0.5
+offsetx number horizontal offset, default is 0
+offsety number vertical offset, default is 0
+color string glow colour, default is black
+ }
+
Returns: object Paper.set of elements that represents glow
+
+*
+ Adds event handlers for hover for the element.
+
+
Parameters f_in
+function
+handler for hover in
+ f_out
+function
+handler for hover out
+ icontext
+optional
+object
+context for hover in handler
+ ocontext
+optional
+object
+context for hover out handler
+
+
Returns: object Element
+
+*
+ Determine if given point is inside this element’s shape
+*
+
+
Parameters *
+
+
x
+number
+x coordinate of the point
+ y
+number
+y coordinate of the point
+
+
Returns: boolean true
if point inside the shape
+
+object *
+ Keeps Matrix object, which represents element transformation
+
+
+*
+ Adds event handler for mousedown for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for mousemove for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for mouseout for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for mouseover for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for mouseup for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Shortcut for assigning event handler for drag.over.<id>
event, where id is id of the element (see Element.id ).
+
+
Parameters f
+function
+handler for event, first argument would be the element you are dragging over
+
+
+*
+ Stops animation of the element with ability to resume it later on.
+*
+
+
Parameters *
+
+
anim
+optional
+object
+animation object
+
+
*
+
+
Returns: object original element
+
+*
+ Removes value associated with an element by given key.
+ If key is not provided, removes all the data of the element.
+
+
Parameters key
+optional
+string
+key
+
+
Returns: object Element
+
+*
+ Resumes animation if it was paused with Element.pause method.
+*
+
+
Parameters *
+
+
anim
+optional
+object
+animation object
+
+
*
+
+
Returns: object original element
+
+*
+ Sets the status of animation of the element in milliseconds. Similar to Element.status method.
+*
+
+
Parameters *
+
+
anim
+object
+animation object
+ value
+number
+number of milliseconds from the beginning of the animation
+
+
*
+
+
Returns: object original element if value
is specified
+
Note, that during animation following events are triggered:
+
+
On each animation frame event anim.frame.<id>
, on start anim.start.<id>
and on end anim.finish.<id>
.
+
+
+*
+ Gets or sets the status of animation of the element.
+*
+
+
Parameters *
+
+
anim
+optional
+object
+animation object
+ value
+optional
+number
+0 – 1. If specified, method works like a setter and sets the status of a given animation to the value. This will cause animation to jump to the given position.
+
+
*
+
+
Returns: number status
+
or
+
+
Returns: array status if anim
is not specified. Array of objects in format:
+
{anim: object animation object
+status: number status
+ }
+
or
+
+
Returns: object original element if value
is specified
+
+*
+ Stops animation of the element.
+*
+
+
Parameters *
+
+
anim
+optional
+object
+animation object
+
+
*
+
+
Returns: object original element
+
+*
+ Adds event handler for touchcancel for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for touchend for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for touchmove for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Adds event handler for touchstart for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for click for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for double click for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes all drag event handlers from given element.
+
+
+*
+ Removes event handlers for hover for the element.
+
+
Parameters f_in
+function
+handler for hover in
+ f_out
+function
+handler for hover out
+
+
Returns: object Element
+
+*
+ Removes event handler for mousedown for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for mousemove for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for mouseout for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for mouseover for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for mouseup for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for touchcancel for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for touchend for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for touchmove for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+*
+ Removes event handler for touchstart for the element.
+
+
Parameters handler
+function
+handler for the event
+
+
Returns: object Element
+
+
+*
+ Adds given matrix to existing one.
+
+
Parameters a
+number
+
+ b
+number
+
+ c
+number
+
+ d
+number
+
+ e
+number
+
+ f
+number
+
+
+
r
+
+
+*
+ Returns copy of the matrix
+
+
Returns: object Matrix
+
+*
+ Returns inverted version of the matrix
+
+
Returns: object Matrix
+
+*
+ Rotates the matrix
+
+
Parameters a
+number
+
+ x
+number
+
+ y
+number
+
+
+
+*
+ Scales the matrix
+
+
Parameters x
+number
+
+ y
+optional
+number
+
+ cx
+optional
+number
+
+ cy
+optional
+number
+
+
+
+*
+ Splits matrix into primitive transformations
+
+
Returns: object in format:
+
dx number translation by x
+dy number translation by y
+scalex number scale by x
+scaley number scale by y
+shear number shear
+rotate number rotation in deg
+isSimple boolean could it be represented via simple transformations
+
+
+*
+ Return transform string that represents given matrix
+
+
Returns: string transform string
+
+*
+ Translate the matrix
+
+
Parameters x
+number
+
+ y
+number
+
+
+
+*
+ Return x coordinate for given point after transformation described by the matrix. See also Matrix.y
+
+
Parameters x
+number
+
+ y
+number
+
+
+
Returns: number x
+
+*
+ Return y coordinate for given point after transformation described by the matrix. See also Matrix.x
+
+
Parameters x
+number
+
+ y
+number
+
+
+
Returns: number y
+
+
+*
+ Imports elements in JSON array in format {type: type, <attributes>}
+*
+
+
Parameters *
+
+
+
Returns: object resulting set of imported elements
+
Usage paper.add([
+ {
+ type: "circle",
+ cx: 10,
+ cy: 10,
+ r: 5
+ },
+ {
+ type: "rect",
+ x: 10,
+ y: 10,
+ width: 10,
+ height: 10,
+ fill: "#fc0"
+ }
+ ]);
+
+
+*
+ Points to the bottom element on the paper
+
+
+
+*
+ Draws a circle.
+*
+
+
Parameters *
+
+
x
+number
+x coordinate of the centre
+ y
+number
+y coordinate of the centre
+ r
+number
+radius
+
+
Returns: object Raphaël element object with type “circle”
+
*
+
+
Usage var c = paper.circle(50, 50, 40);
+
+
+object *
+ If you have a set of attributes that you would like to represent
+ as a function of some number you can do it easily with custom attributes:
+
+
Usage paper.customAttributes.hue = function (num) {
+ num = num % 1;
+ return {fill: "hsb(" + num + ", 0.75, 1)"};
+ };
+ // Custom attribute “hue” will change fill
+ // to be given hue with fixed saturation and brightness.
+ // Now you can use it like this:
+ var c = paper.circle(10, 10, 10).attr({hue: .45});
+ // or even like this:
+ c.animate({hue: 1}, 1e3);
+
+ // You could also create custom attribute
+ // with multiple parameters:
+ paper.customAttributes.hsb = function (h, s, b) {
+ return {fill: "hsb(" + [h, s, b].join(",") + ")"};
+ };
+ c.attr({hsb: "0.5 .8 1"});
+ c.animate({hsb: [1, 0, 0.5]}, 1e3);
+
+
+*
+ Draws an ellipse.
+*
+
+
Parameters *
+
+
x
+number
+x coordinate of the centre
+ y
+number
+y coordinate of the centre
+ rx
+number
+horizontal radius
+ ry
+number
+vertical radius
+
+
Returns: object Raphaël element object with type “ellipse”
+
*
+
+
Usage var c = paper.ellipse(50, 50, 40, 20);
+
+
+*
+ Executes given function for each element on the paper
+
+
If callback function returns false
it will stop loop running.
+*
+
+
Parameters *
+
+
callback
+function
+function to run
+ thisArg
+object
+context object for the callback
+
+
Returns: object Paper object
+
Usage paper.forEach(function (el) {
+ el.attr({ stroke: "blue" });
+ });
+
+
Paper.getElementByPoint⚓ ➭
+*
+ Returns you topmost element under given point.
+*
+
+
Returns: object Raphaël element object
+
Parameters *
+
+
x
+number
+x coordinate from the top left corner of the window
+ y
+number
+y coordinate from the top left corner of the window
+
+
Usage paper.getElementByPoint(mouseX, mouseY).attr({stroke: "#f00"});
+
+
Paper.getElementsByBBox⚓ ➭
+*
+ Returns set of elements that have an intersecting bounding box
+*
+
+
Parameters *
+
+
bbox
+object
+bbox to check with
+
+
Returns: object Set
+
Paper.getElementsByPoint⚓ ➭
+*
+ Returns set of elements that have common point inside
+*
+
+
Parameters *
+
+
x
+number
+x coordinate of the point
+ y
+number
+y coordinate of the point
+
+
Returns: object Set
+
+*
+ Finds font object in the registered fonts by given parameters. You could specify only one word from the font name, like “Myriad” for “Myriad Pro”.
+*
+
+
Parameters *
+
+
family
+string
+font family name or any word from it
+ weight
+optional
+string
+font weight
+ style
+optional
+string
+font style
+ stretch
+optional
+string
+font stretch
+
+
Returns: object the font object
+
Usage paper.print(100, 100, "Test string", paper.getFont("Times", 800), 30);
+
+
+*
+ Creates a group
+*
+
+
Parameters *
+
+
id
+number
+id of the group
+
+
Returns: object Raphaël element object with type “group”
+
*
+
+
Usage
+
+*
+ Hides a paper
+*
+
+
Usage
+
+*
+ Embeds an image into the surface.
+*
+
+
Parameters *
+
+
src
+string
+URI of the source image
+ x
+number
+x coordinate position
+ y
+number
+y coordinate position
+ width
+number
+width of the image
+ height
+number
+height of the image
+
+
Returns: object Raphaël element object with type “image”
+
*
+
+
Usage var c = paper.image("apple.png", 10, 10, 80, 80);
+
+
+*
+ Creates a path element by given path data string.
+
+
Parameters pathString
+optional
+string
+path string in SVG format.
+
+
Path string consists of one-letter commands, followed by comma seprarated arguments in numercal form. Example:
+
+
+
Here we can see two commands: “M”, with arguments (10, 20)
and “L” with arguments (30, 40)
. Upper case letter mean command is absolute, lower case—relative.
+
+
+
Here is short list of commands available, for more details see SVG path string format .
+
Command Name Parameters
+ M moveto (x y)+
+ Z closepath (none)
+ L lineto (x y)+
+ H horizontal lineto x+
+ V vertical lineto y+
+ C curveto (x1 y1 x2 y2 x y)+
+ S smooth curveto (x2 y2 x y)+
+ Q quadratic Bézier curveto (x1 y1 x y)+
+ T smooth quadratic Bézier curveto (x y)+
+ A elliptical arc (rx ry x-axis-rotation large-arc-flag sweep-flag x y)+
+ R Catmull-Rom curveto *x1 y1 (x y)+
+
“Catmull-Rom curveto” is a not standard SVG command and added in 2.0 to make life easier.
+ Note: there is a special case when path consist of just three commands: “M10,10R…z”. In this case path will smoothly connects to its beginning.
+
+
Usage var c = paper.path("M10 10L90 90");
+ // draw a diagonal line:
+ // move to 10,10, line to 90,90
+
+
For example of path strings, check out these icons: http://raphaeljs.com/icons/
+
+
+*
+ Creates path that represent given text written using given font at given position with given size.
+ Result of the method is path element that contains whole text as a separate path.
+*
+
+
Parameters *
+
+
x
+number
+x position of the text
+ y
+number
+y position of the text
+ string
+string
+text to print
+ font
+object
+font object, see Paper.getFont
+ size
+optional
+number
+size of the font, default is 16
+ origin
+optional
+string
+could be "baseline"
or "middle"
, default is "middle"
+ letter_spacing
+optional
+number
+number in range -1..1
, default is 0
+
+
Returns: object resulting path element, which consist of all letters
+
Usage var txt = r.print(10, 50, "print", r.getFont("Museo"), 30).attr({fill: "#fff"});
+
+
+*
+ Points to the Raphael object/function
+
+
+
+
Draws a rectangle.
+*
+
+
Parameters *
+
+
x
+number
+x coordinate of the top left corner
+ y
+number
+y coordinate of the top left corner
+ width
+number
+width
+ height
+number
+height
+ r
+optional
+number
+radius for rounded corners, default is 0
+
+
Returns: object Raphaël element object with type “rect”
+
*
+
+
Usage // regular rectangle
+ var c = paper.rect(10, 10, 50, 50);
+ // rectangle with rounded corners
+ var c = paper.rect(40, 40, 50, 50, 10);
+
+
+*
+ There is an inconvenient rendering bug in Safari (WebKit):
+ sometimes the rendering should be forced.
+ This method should help with dealing with this bug.
+
+
+*
+ Creates array-like object to keep and operate several elements at once.
+ Warning: it doesn’t create any elements for itself in the page, it just groups existing elements.
+ Sets act as pseudo elements — all methods available to an element can be used on a set.
+
+
Returns: object array-like object that represents set of elements
+
*
+
+
Usage var st = paper.set();
+ st.push(
+ paper.circle(10, 10, 5),
+ paper.circle(30, 10, 5)
+ );
+ st.attr({fill: "red"}); // changes the fill of both circles
+
+
+*
+ See Paper.setStart . This method finishes catching and returns resulting set.
+*
+
+
Returns: object set
+
+*
+ If you need to change dimensions of the canvas call this method
+*
+
+
Parameters *
+
+
width
+number
+new width of the canvas
+ height
+number
+new height of the canvas
+
+
+*
+ Creates Paper.set . All elements that will be created after calling this method and before calling
+ Paper.setFinish will be added to the set.
+*
+
+
Usage paper.setStart();
+ paper.circle(10, 10, 5),
+ paper.circle(30, 10, 5)
+ var st = paper.setFinish();
+ st.attr({fill: "red"}); // changes the fill of both circles
+
+
+*
+ Sets the view box of the paper. Practically it gives you ability to zoom and pan whole paper surface by
+ specifying new boundaries.
+*
+
+
Parameters *
+
+
x
+number
+new x position, default is 0
+ y
+number
+new y position, default is 0
+ w
+number
+new width of the canvas
+ h
+number
+new height of the canvas
+ fit
+boolean
+true
if you want graphics to fit into new boundary box
+
+
+*
+ Shows a hidden paper
+*
+
+
Usage
+
+*
+ Draws a text string. If you need line breaks, put “\n” in the string.
+*
+
+
Parameters *
+
+
x
+number
+x coordinate position
+ y
+number
+y coordinate position
+ text
+string
+The text string to draw
+
+
Returns: object Raphaël element object with type “text”
+
*
+
+
Usage var t = paper.text(50, 50, "Raphaël\nkicks\nbutt!");
+
+
+*
+ Points to the topmost element on the paper
+
+
+*
+ Creates a canvas object on which to draw.
+ You must do this first, as all future calls to drawing methods
+ from this instance will be bound to this canvas.
+
+
Parameters *
+
+
container
+HTMLElement string
+DOM element or its ID which is going to be a parent for drawing surface
+ width
+number
+
+ height
+number
+
+ callback
+optional
+function
+callback function which is going to be executed in the context of newly created paper
+
+
or
+
+
x
+number
+
+ y
+number
+
+ width
+number
+
+ height
+number
+
+ callback
+optional
+function
+callback function which is going to be executed in the context of newly created paper
+
+
or
+
+
all
+array
+(first 3 or 4 elements in the array are equal to [containerID, width, height] or [x, y, width, height]. The rest are element descriptions in format {type: type, <attributes>}). See Paper.add .
+ callback
+optional
+function
+callback function which is going to be executed in the context of newly created paper
+
+
or
+
+
onReadyCallback
+function
+function that is going to be called on DOM ready event. You can also subscribe to this event via Eve’s “DOMLoad” event. In this case method returns undefined
.
+
+
Returns: object Paper
+
Usage // Each of the following examples create a canvas
+ // that is 320px wide by 200px high.
+ // Canvas is created at the viewport’s 10,50 coordinate.
+ var paper = Raphael(10, 50, 320, 200);
+ // Canvas is created at the top left corner of the #notepad element
+ // (or its top right corner in dir="rtl" elements)
+ var paper = Raphael(document.getElementById("notepad"), 320, 200);
+ // Same as above
+ var paper = Raphael("notepad", 320, 200);
+ // Image dump
+ var set = Raphael(["notepad", 320, 200, {
+ type: "rect",
+ x: 10,
+ y: 10,
+ width: 25,
+ height: 25,
+ stroke: "#f00"
+ }, {
+ type: "text",
+ x: 30,
+ y: 40,
+ text: "Dump"
+ }]);
+
+
+*
+ Returns angle between two or three points
+
+
Parameters x1
+number
+x coord of first point
+ y1
+number
+y coord of first point
+ x2
+number
+x coord of second point
+ y2
+number
+y coord of second point
+ x3
+optional
+number
+x coord of third point
+ y3
+optional
+number
+y coord of third point
+
+
Returns: number angle in degrees.
+
+*
+ Creates an animation object that can be passed to the Element.animate or Element.animateWith methods.
+ See also Animation.delay and Animation.repeat methods.
+*
+
+
Parameters *
+
+
params
+object
+final attributes for the element, see also Element.attr
+ ms
+number
+number of milliseconds for animation to run
+ easing
+optional
+string
+easing type. Accept one of Raphael.easing_formulas or CSS format: cubic‐bezier(XX, XX, XX, XX)
+ callback
+optional
+function
+callback function. Will be called at the end of animation.
+
+
*
+
+
Returns: object Animation
+
+*
+ Utility method
+*
+ Return bounding box of a given cubic bezier curve
+
+
Parameters p1x
+number
+x of the first point of the curve
+ p1y
+number
+y of the first point of the curve
+ c1x
+number
+x of the first anchor of the curve
+ c1y
+number
+y of the first anchor of the curve
+ c2x
+number
+x of the second anchor of the curve
+ c2y
+number
+y of the second anchor of the curve
+ p2x
+number
+x of the second point of the curve
+ p2y
+number
+y of the second point of the curve
+
+
or
+
+
bez
+array
+array of six points for bezier curve
+
+
Returns: object point information in format:
+
{ min: {x: number x coordinate of the left point
+y: number y coordinate of the top point
+ } max: {x: number x coordinate of the right point
+y: number y coordinate of the bottom point
+ } }
+
+
+*
+ Returns a recursively cloned version of an object.
+
+
+*
+ Parses the color string and returns object with all values for the given color.
+
+
Parameters clr
+string
+color string in one of the supported formats (see Raphael.getRGB )
+
+
Returns: object Combined RGB & HSB object in format:
+
{r number red,
+g number green,
+b number blue,
+hex string color in HTML/CSS format: #••••••,
+error boolean true
if string can’t be parsed,
+h number hue,
+s number saturation,
+v number value (brightness),
+l number lightness
+ }
+
+*
+ Returns RFC4122, version 4 ID
+
+
Raphael.customAttributes⚓ ➭
+object *
+ If you have a set of attributes that you would like to represent
+ as a function of some number across all papers you can do it
+ easily with custom attributes:
+
+
Usage Raphael.customAttributes.hue = function (num) {
+ num = num % 1;
+ return {fill: "hsb(" + num + ", 0.75, 1)"};
+ };
+ // Custom attribute “hue” will change fill
+ // to be given hue with fixed saturation and brightness.
+ // Now you can use it like this:
+ var c = paper.circle(10, 10, 10).attr({hue: .45});
+ // or even like this:
+ c.animate({hue: 1}, 1e3);
+
+ // You could also create custom attribute
+ // with multiple parameters:
+ Raphael.customAttributes.hsb = function (h, s, b) {
+ return {fill: "hsb(" + [h, s, b].join(",") + ")"};
+ };
+ c.attr({hsb: "0.5 .8 1"});
+ c.animate({hsb: [1, 0, 0.5]}, 1e3);
+
+
+*
+ Allows a unified definition of composite shapes and other behaviours using
+ simple directives.
+*
+
+
Parameters *
+
+
definition
+object
+the shape definition
+
+
+*
+ Transform angle to degrees
+
+
Parameters deg
+number
+angle in radians
+
+
Returns: number angle in degrees.
+
+*
+ Object that contains easing formulas for animation. You could extend it with your own. By default it has following list of easing:
+
+
+ “linear”
+ “<” or “easeIn” or “ease-in”
+ “>” or “easeOut” or “ease-out”
+ “<>” or “easeInOut” or “ease-in-out”
+ “backIn” or “back-in”
+ “backOut” or “back-out”
+ “elastic”
+ “bounce”
+
+
See also Easing demo .
+
+object *
+ You can add your own method to elements. This is usefull when you want to hack default functionality or
+ want to wrap some common transformation or attributes in one method. In difference to canvas methods,
+ you can redefine element method at any time. Expending element methods wouldn’t affect set.
+
+
Usage Raphael.el.red = function () {
+ this.attr({fill: "#f00"});
+ };
+ // then use it
+ paper.circle(100, 100, 20).red();
+
+
Raphael.findDotsAtSegment⚓ ➭
+*
+ Utility method
+*
+ Find dot coordinates on the given cubic bezier curve at the given t.
+
+
Parameters p1x
+number
+x of the first point of the curve
+ p1y
+number
+y of the first point of the curve
+ c1x
+number
+x of the first anchor of the curve
+ c1y
+number
+y of the first anchor of the curve
+ c2x
+number
+x of the second anchor of the curve
+ c2y
+number
+y of the second anchor of the curve
+ p2x
+number
+x of the second point of the curve
+ p2y
+number
+y of the second point of the curve
+ t
+number
+position on the curve (0..1)
+
+
Returns: object point information in format:
+
{x: number x coordinate of the point
+y: number y coordinate of the point
+ m: {x: number x coordinate of the left anchor
+y: number y coordinate of the left anchor
+ } n: {x: number x coordinate of the right anchor
+y: number y coordinate of the right anchor
+ } start: {x: number x coordinate of the start of the curve
+y: number y coordinate of the start of the curve
+ } end: {x: number x coordinate of the end of the curve
+y: number y coordinate of the end of the curve
+ } alpha: number angle of the curve derivative at the point
+ }
+
+object *
+ You can add your own method to the canvas. For example if you want to draw a pie chart,
+ you can create your own pie chart function and ship it as a Raphaël plugin. To do this
+ you need to extend the Raphael.fn
object. You should modify the fn
object before a
+ Raphaël instance is created, otherwise it will take no effect. Please note that the
+ ability for namespaced plugins was removed in Raphael 2.0. It is up to the plugin to
+ ensure any namespacing ensures proper context.
+
+
Usage Raphael.fn.arrow = function (x1, y1, x2, y2, size) {
+ return this.path( ... );
+ };
+ // or create namespace
+ Raphael.fn.mystuff = {
+ arrow: function () {…},
+ star: function () {…},
+ // etc…
+ };
+ var paper = Raphael(10, 10, 630, 480);
+ // then use it
+ paper.arrow(10, 10, 30, 30, 5).attr({fill: "#f00"});
+ paper.mystuff.arrow();
+ paper.mystuff.star();
+
+
+*
+ Simple format function. Replaces construction of type “{<number>}
” to the corresponding argument.
+*
+
+
Parameters *
+
+
token
+string
+string to format
+ …
+string
+rest of arguments will be treated as parameters for replacement
+
+
Returns: string formated string
+
Usage var x = 10,
+ y = 20,
+ width = 40,
+ height = 50;
+ // this will draw a rectangular shape equivalent to "M10,20h40v50h-40z"
+ paper.path(Raphael.format("M{0},{1}h{2}v{3}h{4}z", x, y, width, height, -width));
+
+
+*
+ A little bit more advanced format function than Raphael.format . Replaces construction of type “{<name>}
” to the corresponding argument.
+*
+
+
Parameters *
+
+
token
+string
+string to format
+ json
+object
+object which properties will be used as a replacement
+
+
Returns: string formated string
+
Usage // this will draw a rectangular shape equivalent to "M10,20h40v50h-40z"
+ paper.path(Raphael.fullfill("M{x},{y}h{dim.width}v{dim.height}h{dim['negative width']}z", {
+ x: 10,
+ y: 20,
+ dim: {
+ width: 40,
+ height: 50,
+ "negative width": -40
+ }
+ }));
+
+
+*
+ On each call returns next colour in the spectrum. To reset it back to red call Raphael.getColor.reset
+
+
Parameters value
+optional
+number
+brightness, default is 0.75
+
+
Returns: string hex representation of the colour.
+
+ Raphael.getPointAtLength⚓ ➭
+*
+ Return coordinates of the point located at the given length on the given path.
+*
+
+
Parameters *
+
+
path
+string
+SVG path string
+ length
+number
+
+
+
*
+
+
Returns: object representation of the point:
+
{x: number x coordinate
+y: number y coordinate
+alpha: number angle of derivative
+ }
+
+*
+ Parses colour string as RGB object
+
+
Parameters colour
+string
+colour string in one of formats:
+
+
+ Colour name (“red
”, “green
”, “cornflowerblue
”, etc)
+ #••• — shortened HTML colour: (“#000
”, “#fc0
”, etc)
+ #•••••• — full length HTML colour: (“#000000
”, “#bd2300
”)
+ rgb(•••, •••, •••) — red, green and blue channels’ values: (“rgb(200, 100, 0)
”)
+ rgb(•••%, •••%, •••%) — same as above, but in %: (“rgb(100%, 175%, 0%)
”)
+ hsb(•••, •••, •••) — hue, saturation and brightness values: (“hsb(0.5, 0.25, 1)
”)
+ hsb(•••%, •••%, •••%) — same as above, but in %
+ hsl(•••, •••, •••) — same as hsb
+ hsl(•••%, •••%, •••%) — same as hsb
+
+
Returns: object RGB object in format:
+
{r number red,
+g number green,
+b number blue
+hex string color in HTML/CSS format: #••••••,
+error boolean true if string can’t be parsed
+ }
+
+*
+ Return subpath of a given path from given length to given length.
+*
+
+
Parameters *
+
+
path
+string
+SVG path string
+ from
+number
+position of the start of the segment
+ to
+number
+position of the end of the segment
+
+
*
+
+
Returns: string pathstring for the segment
+
+*
+ Returns length of the given path in pixels.
+*
+
+
Parameters *
+
+
path
+string
+SVG path string.
+
+
*
+
+
Returns: number length.
+
+*
+ Converts HSB values to hex representation of the colour.
+
+
Parameters h
+number
+hue
+ s
+number
+saturation
+ b
+number
+value or brightness
+
+
Returns: string hex representation of the colour.
+
+*
+ Converts HSB values to RGB object.
+
+
Parameters h
+number
+hue
+ s
+number
+saturation
+ v
+number
+value or brightness
+
+
Returns: object RGB object in format:
+
{r number red,
+g number green,
+b number blue,
+hex string color in HTML/CSS format: #••••••
+ }
+
+*
+ Converts HSL values to hex representation of the colour.
+
+
Parameters h
+number
+hue
+ s
+number
+saturation
+ l
+number
+luminosity
+
+
Returns: string hex representation of the colour.
+
+*
+ Converts HSL values to RGB object.
+
+
Parameters h
+number
+hue
+ s
+number
+saturation
+ l
+number
+luminosity
+
+
Returns: object RGB object in format:
+
{r number red,
+g number green,
+b number blue,
+hex string color in HTML/CSS format: #••••••
+ }
+
+*
+ Handfull replacement for typeof
operator.
+
+
Parameters o
+…
+any object or primitive
+ type
+string
+name of the type, i.e. “string”, “function”, “number”, etc.
+
+
Returns: boolean is given value is of given type
+
Raphael.isBBoxIntersect⚓ ➭
+*
+ Utility method
+*
+ Returns true
if two bounding boxes intersect
+
+
Parameters bbox1
+string
+first bounding box
+ bbox2
+string
+second bounding box
+
+
Returns: boolean true
if they intersect
+
Raphael.isPointInsideBBox⚓ ➭
+*
+ Utility method
+*
+ Returns true
if given point is inside bounding boxes.
+
+
Parameters bbox
+string
+bounding box
+ x
+string
+x coordinate of the point
+ y
+string
+y coordinate of the point
+
+
Returns: boolean true
if point inside
+
Raphael.isPointInsidePath⚓ ➭
+*
+ Utility method
+*
+ Returns true
if given point is inside a given closed path.
+
+
Parameters path
+string
+path string
+ x
+number
+x of the point
+ y
+number
+y of the point
+
+
Returns: boolean true, if point is inside the path
+
+*
+ Transform the path string with given matrix.
+
+
Parameters path
+string
+path string
+ matrix
+object
+see Matrix
+
+
Returns: string transformed path string
+
+*
+ Utility method
+*
+ Returns matrix based on given parameters.
+
+
Parameters a
+number
+
+ b
+number
+
+ c
+number
+
+ d
+number
+
+ e
+number
+
+ f
+number
+
+
+
Returns: object Matrix
+
+*
+ If you want to leave no trace of Raphaël (Well, Raphaël creates only one global variable Raphael
, but anyway.) You can use ninja
method.
+ Beware, that in this case plugins could stop working, because they are depending on global variable existance.
+*
+
+
Returns: object Raphael object
+
Usage (function (local_raphael) {
+ var paper = local_raphael(10, 10, 320, 200);
+ …
+ })(Raphael.ninja());
+
+
Raphael.parsePathString⚓ ➭
+*
+ Utility method
+*
+ Parses given path string into an array of arrays of path segments.
+
+
Parameters pathString
+string array
+path string or array of segments (in the last case it will be returned straight away)
+
+
Returns: array array of segments.
+
+*
+ Utility method
+*
+ Parses given path string into an array of transformations.
+
+
Parameters TString
+string array
+transform string or array of transformations (in the last case it will be returned straight away)
+
+
Returns: array array of transformations.
+
+*
+ Utility method
+*
+ Converts path to a new path where all segments are cubic bezier curves.
+
+
Parameters pathString
+string array
+path string or array of segments
+
+
Returns: array array of segments.
+
+*
+ Utility method
+*
+ Return bounding box of a given path
+
+
Parameters path
+string
+path string
+
+
Returns: object bounding box
+
{x: number x coordinate of the left top point of the box
+y: number y coordinate of the left top point of the box
+x2: number x coordinate of the right bottom point of the box
+y2: number y coordinate of the right bottom point of the box
+width: number width of the box
+height: number height of the box
+cx: number x coordinate of the center of the box
+cy: number y coordinate of the center of the box
+ }
+
Raphael.pathIntersection⚓ ➭
+*
+ Utility method
+*
+ Finds intersections of two paths
+
+
Parameters path1
+string
+path string
+ path2
+string
+path string
+
+
Returns: array dots of intersection
+
[ {x: number x coordinate of the point
+y: number y coordinate of the point
+t1: number t value for segment of path1
+t2: number t value for segment of path2
+segment1: number order number for segment of path1
+segment2: number order number for segment of path2
+bez1: array eight coordinates representing beziér curve for the segment of path1
+bez2: array eight coordinates representing beziér curve for the segment of path2
+ } ]
+
+*
+ Utility method
+*
+ Converts path to relative form
+
+
Parameters pathString
+string array
+path string or array of segments
+
+
Returns: array array of segments.
+
+*
+ Returns the first truthy argument.
+
+
+*
+ Transform angle to radians
+
+
Parameters deg
+number
+angle in degrees
+
+
Returns: number angle in radians.
+
+*
+ Adds given font to the registered set of fonts for Raphaël. Should be used as an internal call from within Cufón’s font file.
+ Returns original parameter, so it could be used with chaining.
+
+
More about Cufón and how to convert your font form TTF, OTF, etc to JavaScript file.
+
*
+
+
Parameters *
+
+
font
+object
+the font to register
+
+
Returns: object the font you passed in
+
Usage Cufon.registerFont(Raphael.registerFont({…}));
+
+
+*
+ Converts RGB values to hex representation of the colour.
+
+
Parameters r
+number
+red
+ g
+number
+green
+ b
+number
+blue
+
+
Returns: string hex representation of the colour.
+
+*
+ Converts RGB values to HSB object.
+
+
Parameters r
+number
+red
+ g
+number
+green
+ b
+number
+blue
+
+
Returns: object HSB object in format:
+
{h number hue
+s number saturation
+b number brightness
+ }
+
+*
+ Converts RGB values to HSL object.
+
+
Parameters r
+number
+red
+ g
+number
+green
+ b
+number
+blue
+
+
Returns: object HSL object in format:
+
{h number hue
+s number saturation
+l number luminosity
+ }
+
+*
+ Used when you need to draw in <iframe>
. Switched window to the iframe one.
+
+
Parameters newwin
+window
+new window object
+
+
+*
+ Snaps given value to given grid.
+
+
Parameters values
+array number
+given array of values or step of the grid
+ value
+number
+value to adjust
+ tolerance
+optional
+number
+tolerance for snapping. Default is 10
.
+
+
Returns: number adjusted value.
+
+object *
+ You can add your own method to elements and sets. It is wise to add a set method for each element method
+ you added, so you will be able to call the same method on sets too.
+*
+ See also Raphael.el .
+
+
Usage Raphael.el.red = function () {
+ this.attr({fill: "#f00"});
+ };
+ Raphael.st.red = function () {
+ this.forEach(function (el) {
+ el.red();
+ });
+ };
+ // then use it
+ paper.set(paper.circle(100, 100, 20), paper.circle(110, 100, 20)).red();
+
+
+boolean *
+ true
if browser supports SVG.
+
+
+*
+ Utility method
+*
+ Returns matrix of transformations applied to a given path
+
+
Parameters path
+string
+path string
+ transform
+string array
+transformation string
+
+
Returns: object Matrix
+
+*
+ Utility method
+*
+ Returns path transformed by a given transformation
+
+
Parameters path
+string
+path string
+ transform
+string array
+transformation string
+
+
Returns: string path
+
+string *
+ Can be “SVG”, “VML” or empty, depending on browser support.
+
+
+boolean *
+ true
if browser supports VML.
+
+
+
+*
+ Removeds all elements from the set
+
+
+*
+ Removes given element from the set
+*
+
+
Parameters *
+
+
element
+object
+element to remove
+
+
Returns: boolean true
if object was found & removed from the set
+
+*
+ Executes given function for each element in the set.
+
+
If function returns false
it will stop loop running.
+*
+
+
Parameters *
+
+
callback
+function
+function to run
+ thisArg
+object
+context object for the callback
+
+
Returns: object Set object
+
+*
+ Removes last element and returns it.
+
+
Returns: object element
+
+*
+ Adds each argument to the current set.
+
+
Returns: object original element
+
+*
+ Removes given element from the set
+*
+
+
Parameters *
+
+
index
+number
+position of the deletion
+ count
+number
+number of element to remove
+ insertion…
+optional
+object
+elements to insert
+
+
Returns: object set elements that were deleted
+
+ Fires event with given name
, given scope and other parameters.
+
+
Arguments name
+string
+name of the event , dot (.
) or slash (/
) separated
+ scope
+object
+context for the event handlers
+ varargs
+...
+the rest of arguments will be sent to event handlers
+
+
Returns: object array of returned values from the listeners
+
+*
+ Returns function that will fire given event with optional arguments.
+ Arguments that will be passed to the result function will be also
+ concated to the list of final arguments.
+
+
el.onclick = eve.f("click", 1, 2);
+ eve.on("click", function (a, b, c) {
+ console.log(a, b, c); // 1, 2, [event object]
+ });
+
+
Arguments event
+string
+event name
+ varargs
+…
+and any other arguments
+
+
Returns: function possible event handler function
+
+ Internal method which gives you array of all event handlers that will be triggered by the given name
.
+
+
Arguments name
+string
+name of the event, dot (.
) or slash (/
) separated
+
+
Returns: array array of event handlers
+
+*
+ Could be used inside event handler to figure out actual name of the event.
+*
+
+
Arguments *
+
+
subname
+optional
+string
+subname of the event
+
+
*
+
+
Returns: string name of the event, if subname
is not specified
+
or
+
+
Returns: boolean true
, if current event’s name contains subname
+
+*
+ Could be used inside event handler to figure out actual name of the event.
+*
+*
+
+
Returns: array names of the event
+
+*
+ Removes given function from the list of event listeners assigned to given name.
+ If no arguments specified all the events will be cleared.
+*
+
+
Arguments *
+
+
name
+string
+name of the event, dot (.
) or slash (/
) separated, with optional wildcards
+ f
+function
+event handler function
+
+
+*
+ Binds given event handler with a given name. You can use wildcards “*
” for the names:
+
+
eve.on("*.under.*", f);
+ eve("mouse.under.floor"); // triggers f
+
+
Use eve to trigger the listener.
+*
+
+
Arguments *
+
+
name
+string
+name of the event, dot (.
) or slash (/
) separated, with optional wildcards
+ f
+function
+event handler function
+
+
*
+
+
Returns: function returned function accepts a single numeric parameter that represents z-index of the handler. It is an optional feature and only used when you need to ensure that some subset of handlers will be invoked in a given order, despite of the order of assignment.
+
Example: eve.on("mouse", eatIt)(2);
+ eve.on("mouse", scream);
+ eve.on("mouse", catchIt)(1);
+
+
This will ensure that catchIt()
function will be called before eatIt()
.
+
+
If you want to put your handler before non-indexed handlers, specify a negative value.
+ Note: I assume most of the time you don’t need to worry about z-index, but it’s nice to have this feature “just in case”.
+
+
+*
+ Binds given event handler with a given name to only run once then unbind itself.
+
+
eve.once("login", f);
+ eve("login"); // triggers f
+ eve("login"); // no listeners
+
+
Use eve to trigger the listener.
+*
+
+
Arguments *
+
+
name
+string
+name of the event, dot (.
) or slash (/
) separated, with optional wildcards
+ f
+function
+event handler function
+
+
*
+
+
Returns: function same return function as eve.on
+
+*
+ Is used inside an event handler to stop the event, preventing any subsequent listeners from firing.
+
+
+
+string *
+ Current version of the library.
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/docs/js/prism.js b/docs/js/prism.js
new file mode 100644
index 0000000..8f1a555
--- /dev/null
+++ b/docs/js/prism.js
@@ -0,0 +1,9 @@
+/**
+ * Prism: Lightweight, robust, elegant syntax highlighting
+ * MIT license http://www.opensource.org/licenses/mit-license.php/
+ * @author Lea Verou http://lea.verou.me
+ */(function(){var e=/\blang(?:uage)?-(?!\*)(\w+)\b/i,t=self.Prism={util:{type:function(e){return Object.prototype.toString.call(e).match(/\[object (\w+)\]/)[1]},clone:function(e){var n=t.util.type(e);switch(n){case"Object":var r={};for(var i in e)e.hasOwnProperty(i)&&(r[i]=t.util.clone(e[i]));return r;case"Array":return e.slice()}return e}},languages:{extend:function(e,n){var r=t.util.clone(t.languages[e]);for(var i in n)r[i]=n[i];return r},insertBefore:function(e,n,r,i){i=i||t.languages;var s=i[e],o={};for(var u in s)if(s.hasOwnProperty(u)){if(u==n)for(var a in r)r.hasOwnProperty(a)&&(o[a]=r[a]);o[u]=s[u]}return i[e]=o},DFS:function(e,n){for(var r in e){n.call(e,r,e[r]);t.util.type(e)==="Object"&&t.languages.DFS(e[r],n)}}},highlightAll:function(e,n){var r=document.querySelectorAll('code[class*="language-"], [class*="language-"] code, code[class*="lang-"], [class*="lang-"] code');for(var i=0,s;s=r[i++];)t.highlightElement(s,e===!0,n)},highlightElement:function(r,i,s){var o,u,a=r;while(a&&!e.test(a.className))a=a.parentNode;if(a){o=(a.className.match(e)||[,""])[1];u=t.languages[o]}if(!u)return;r.className=r.className.replace(e,"").replace(/\s+/g," ")+" language-"+o;a=r.parentNode;/pre/i.test(a.nodeName)&&(a.className=a.className.replace(e,"").replace(/\s+/g," ")+" language-"+o);var f=r.textContent;if(!f)return;f=f.replace(/&/g,"&").replace(/e.length)break e;if(p instanceof i)continue;a.lastIndex=0;var d=a.exec(p);if(d){l&&(c=d[1].length);var v=d.index-1+c,d=d[0].slice(c),m=d.length,g=v+m,y=p.slice(0,v+1),b=p.slice(g+1),w=[h,1];y&&w.push(y);var E=new i(u,f?t.tokenize(d,f):d);w.push(E);b&&w.push(b);Array.prototype.splice.apply(s,w)}}}return s},hooks:{all:{},add:function(e,n){var r=t.hooks.all;r[e]=r[e]||[];r[e].push(n)},run:function(e,n){var r=t.hooks.all[e];if(!r||!r.length)return;for(var i=0,s;s=r[i++];)s(n)}}},n=t.Token=function(e,t){this.type=e;this.content=t};n.stringify=function(e,r,i){if(typeof e=="string")return e;if(Object.prototype.toString.call(e)=="[object Array]")return e.map(function(t){return n.stringify(t,r,e)}).join("");var s={type:e.type,content:n.stringify(e.content,r,i),tag:"span",classes:["token",e.type],attributes:{},language:r,parent:i};s.type=="comment"&&(s.attributes.spellcheck="true");t.hooks.run("wrap",s);var o="";for(var u in s.attributes)o+=u+'="'+(s.attributes[u]||"")+'"';return"<"+s.tag+' class="'+s.classes.join(" ")+'" '+o+">"+s.content+""+s.tag+">"};if(!self.document){self.addEventListener("message",function(e){var n=JSON.parse(e.data),r=n.language,i=n.code;self.postMessage(JSON.stringify(t.tokenize(i,t.languages[r])));self.close()},!1);return}var r=document.getElementsByTagName("script");r=r[r.length-1];if(r){t.filename=r.src;document.addEventListener&&!r.hasAttribute("data-manual")&&document.addEventListener("DOMContentLoaded",t.highlightAll)}})();;
+Prism.languages.clike={comment:{pattern:/(^|[^\\])(\/\*[\w\W]*?\*\/|(^|[^:])\/\/.*?(\r?\n|$))/g,lookbehind:!0},string:/("|')(\\?.)*?\1/g,"class-name":{pattern:/((?:(?:class|interface|extends|implements|trait|instanceof|new)\s+)|(?:catch\s+\())[a-z0-9_\.\\]+/ig,lookbehind:!0,inside:{punctuation:/(\.|\\)/}},keyword:/\b(if|else|while|do|for|return|in|instanceof|function|new|try|throw|catch|finally|null|break|continue)\b/g,"boolean":/\b(true|false)\b/g,"function":{pattern:/[a-z0-9_]+\(/ig,inside:{punctuation:/\(/}}, number:/\b-?(0x[\dA-Fa-f]+|\d*\.?\d+([Ee]-?\d+)?)\b/g,operator:/[-+]{1,2}|!|<=?|>=?|={1,3}|(&){1,2}|\|?\||\?|\*|\/|\~|\^|\%/g,ignore:/&(lt|gt|amp);/gi,punctuation:/[{}[\];(),.:]/g};
+;
+Prism.languages.javascript=Prism.languages.extend("clike",{keyword:/\b(var|let|if|else|while|do|for|return|in|instanceof|function|new|with|typeof|try|throw|catch|finally|null|break|continue)\b/g,number:/\b-?(0x[\dA-Fa-f]+|\d*\.?\d+([Ee]-?\d+)?|NaN|-?Infinity)\b/g});Prism.languages.insertBefore("javascript","keyword",{regex:{pattern:/(^|[^/])\/(?!\/)(\[.+?]|\\.|[^/\r\n])+\/[gim]{0,3}(?=\s*($|[\r\n,.;})]))/g,lookbehind:!0}});Prism.languages.markup&&Prism.languages.insertBefore("markup","tag",{script:{pattern:/(<|<)script[\w\W]*?(>|>)[\w\W]*?(<|<)\/script(>|>)/ig,inside:{tag:{pattern:/(<|<)script[\w\W]*?(>|>)|(<|<)\/script(>|>)/ig,inside:Prism.languages.markup.tag.inside},rest:Prism.languages.javascript}}});
+;
\ No newline at end of file
diff --git a/package.json b/package.json
new file mode 100644
index 0000000..6815a97
--- /dev/null
+++ b/package.json
@@ -0,0 +1,19 @@
+{
+ "name": "redraphael",
+ "filename": "raphael",
+ "version": "1.0.0",
+ "rversion": "2.1.0",
+ "description": "RedRaphael",
+ "main": "index.js",
+ "scripts": {
+ "test": "init"
+ },
+ "repository": "",
+ "author": "FusionCharts",
+ "license": "MIT",
+ "devDependencies": {
+ "grunt-contrib-concat": "~0.3.0",
+ "grunt-contrib-uglify": "~0.2.2",
+ "grunt-contrib-jasmine": "~0.5.2"
+ }
+}
diff --git a/package/raphael-min.js b/package/raphael-min.js
new file mode 100644
index 0000000..dc832d4
--- /dev/null
+++ b/package/raphael-min.js
@@ -0,0 +1,15 @@
+/**!
+ * RedRaphael 1.0.0 - JavaScript Vector Library
+ * Copyright (c) 2012-2013 FusionCharts Technologies
+ *
+ * Raphael 2.1.0
+ * Copyright (c) 2008-2012 Dmitry Baranovskiy
+ * Copyright © 2008-2012 Sencha Labs
+ *
+ * Licensed under the MIT license.
+ */
+!function(a){var b,c,d="0.4.2",e="hasOwnProperty",f=/[\.\/]/,g="*",h=function(){},i=function(a,b){return a-b},j={n:{}},k=function(a,d){a=String(a);var e,f=c,g=Array.prototype.slice.call(arguments,2),h=k.listeners(a),j=0,l=[],m={},n=[],o=b;b=a,c=0;for(var p=0,q=h.length;q>p;p++)"zIndex"in h[p]&&(l.push(h[p].zIndex),h[p].zIndex<0&&(m[h[p].zIndex]=h[p]));for(l.sort(i);l[j]<0;)if(e=m[l[j++]],n.push(e.apply(d,g)),c)return c=f,n;for(p=0;q>p;p++)if(e=h[p],"zIndex"in e)if(e.zIndex==l[j]){if(n.push(e.apply(d,g)),c)break;do if(j++,e=m[l[j]],e&&n.push(e.apply(d,g)),c)break;while(e)}else m[e.zIndex]=e;else if(n.push(e.apply(d,g)),c)break;return c=f,b=o,n.length?n:null};k._events=j,k.listeners=function(a){var b,c,d,e,h,i,k,l,m=a.split(f),n=j,o=[n],p=[];for(e=0,h=m.length;h>e;e++){for(l=[],i=0,k=o.length;k>i;i++)for(n=o[i].n,c=[n[m[e]],n[g]],d=2;d--;)b=c[d],b&&(l.push(b),p=p.concat(b.f||[]));o=l}return p},k.on=function(a,b){if(a=String(a),"function"!=typeof b)return function(){};for(var c=a.split(f),d=j,e=0,g=c.length;g>e;e++)d=d.n,d=d.hasOwnProperty(c[e])&&d[c[e]]||(d[c[e]]={n:{}});for(d.f=d.f||[],e=0,g=d.f.length;g>e;e++)if(d.f[e]==b)return h;return d.f.push(b),function(a){+a==+a&&(b.zIndex=+a)}},k.f=function(a){var b=[].slice.call(arguments,1);return function(){k.apply(null,[a,null].concat(b).concat([].slice.call(arguments,0)))}},k.stop=function(){c=1},k.nt=function(a){return a?new RegExp("(?:\\.|\\/|^)"+a+"(?:\\.|\\/|$)").test(b):b},k.nts=function(){return b.split(f)},k.off=k.unbind=function(a,b){if(!a)return k._events=j={n:{}},void 0;var c,d,h,i,l,m,n,o=a.split(f),p=[j];for(i=0,l=o.length;l>i;i++)for(m=0;mi;i++)for(c=p[i];c.n;){if(b){if(c.f){for(m=0,n=c.f.length;n>m;m++)if(c.f[m]==b){c.f.splice(m,1);break}!c.f.length&&delete c.f}for(d in c.n)if(c.n[e](d)&&c.n[d].f){var q=c.n[d].f;for(m=0,n=q.length;n>m;m++)if(q[m]==b){q.splice(m,1);break}!q.length&&delete c.n[d].f}}else{delete c.f;for(d in c.n)c.n[e](d)&&c.n[d].f&&delete c.n[d].f}c=c.n}},k.once=function(a,b){var c=function(){return k.unbind(a,c),b.apply(this,arguments)};return k.on(a,c)},k.version=d,k.toString=function(){return"You are running Eve "+d},"undefined"!=typeof module&&module.exports?module.exports=k:"undefined"!=typeof define?define("eve",[],function(){return k}):a.eve=k}(this),function(a,b){"function"==typeof define&&define.amd?define(["eve"],function(c){return b(a,c)}):b(a,a.eve)}(this,function(a,b){function c(d){var e,f;return c._url&&(c._url=(c._g&&c._g.win||a).location.href.replace(/#.*?$/,"")),c.is(d,"function")?s?d():b.on("raphael.DOMload",d):c.is(d,B)?c._engine.create[x](c,d.splice(0,3+c.is(d[0],z))).add(d):(e=Array.prototype.slice.call(arguments,0),c.is(e[e.length-1],"function")?(f=e.pop(),s?f.call(c._engine.create[x](c,e)):b.on("raphael.DOMload",function(){f.call(c._engine.create[x](c,e))})):c._engine.create[x](c,arguments))}function d(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return a.push(a.splice(c,1)[0])}function e(){return this.hex}function f(a,b){for(var c=[],d=0,e=a.length;e-2*!b>d;d+=2){var f=[{x:+a[d-2],y:+a[d-1]},{x:+a[d],y:+a[d+1]},{x:+a[d+2],y:+a[d+3]},{x:+a[d+4],y:+a[d+5]}];b?d?e-4==d?f[3]={x:+a[0],y:+a[1]}:e-2==d&&(f[2]={x:+a[0],y:+a[1]},f[3]={x:+a[2],y:+a[3]}):f[0]={x:+a[e-2],y:+a[e-1]}:e-4==d?f[3]=f[2]:d||(f[0]={x:+a[d],y:+a[d+1]}),c.push(["C",(-f[0].x+6*f[1].x+f[2].x)/6,(-f[0].y+6*f[1].y+f[2].y)/6,(f[1].x+6*f[2].x-f[3].x)/6,(f[1].y+6*f[2].y-f[3].y)/6,f[2].x,f[2].y])}return c}function g(a,b,c,d,e){var f=-3*b+9*c-9*d+3*e,g=a*f+6*b-12*c+6*d;return a*g-3*b+3*c}function h(a,b,c,d,e,f,h,i,j){null==j&&(j=1),j=j>1?1:0>j?0:j;for(var k=j/2,l=12,m=[-.1252,.1252,-.3678,.3678,-.5873,.5873,-.7699,.7699,-.9041,.9041,-.9816,.9816],n=[.2491,.2491,.2335,.2335,.2032,.2032,.1601,.1601,.1069,.1069,.0472,.0472],o=0,p=0;l>p;p++){var q=k*m[p]+k,r=g(q,a,c,e,h),s=g(q,b,d,f,i),t=r*r+s*s;o+=n[p]*eb(t)}return k*o}function i(a,b,c,d,e,f,g,i,j){if(!(0>j||h(a,b,c,d,e,f,g,i)o;)m/=2,n+=(j>k?1:-1)*m,k=h(a,b,c,d,e,f,g,i,n);return n}}function j(a,b,c,d,e,f,g,h){if(!($(a,c)<_(e,g)||_(a,c)>$(e,g)||$(b,d)<_(f,h)||_(b,d)>$(f,h))){var i=(a*d-b*c)*(e-g)-(a-c)*(e*h-f*g),j=(a*d-b*c)*(f-h)-(b-d)*(e*h-f*g),k=(a-c)*(f-h)-(b-d)*(e-g);if(k){var l=i/k,m=j/k,n=+l.toFixed(2),o=+m.toFixed(2);if(!(n<+_(a,c).toFixed(2)||n>+$(a,c).toFixed(2)||n<+_(e,g).toFixed(2)||n>+$(e,g).toFixed(2)||o<+_(b,d).toFixed(2)||o>+$(b,d).toFixed(2)||o<+_(f,h).toFixed(2)||o>+$(f,h).toFixed(2)))return{x:l,y:m}}}}function k(a,b,d){var e=c.bezierBBox(a),f=c.bezierBBox(b);if(!c.isBBoxIntersect(e,f))return d?0:[];for(var g=h.apply(0,a),i=h.apply(0,b),k=$(~~(g/5),1),l=$(~~(i/5),1),m=[],n=[],o={},p=d?0:[],q=0;k+1>q;q++){var r=c.findDotsAtSegment.apply(c,a.concat(q/k));m.push({x:r.x,y:r.y,t:q/k})}for(q=0;l+1>q;q++)r=c.findDotsAtSegment.apply(c,b.concat(q/l)),n.push({x:r.x,y:r.y,t:q/l});for(q=0;k>q;q++)for(var s=0;l>s;s++){var t=m[q],u=m[q+1],v=n[s],w=n[s+1],x=ab(u.x-t.x)<.001?"y":"x",y=ab(w.x-v.x)<.001?"y":"x",z=j(t.x,t.y,u.x,u.y,v.x,v.y,w.x,w.y);if(z){if(o[z.x.toFixed(4)]==z.y.toFixed(4))continue;o[z.x.toFixed(4)]=z.y.toFixed(4);var A=t.t+ab((z[x]-t[x])/(u[x]-t[x]))*(u.t-t.t),B=v.t+ab((z[y]-v[y])/(w[y]-v[y]))*(w.t-v.t);A>=0&&1.001>=A&&B>=0&&1.001>=B&&(d?p++:p.push({x:z.x,y:z.y,t1:_(A,1),t2:_(B,1)}))}}return p}function l(a,b,d){a=c._path2curve(a),b=c._path2curve(b);for(var e,f,g,h,i,j,l,m,n,o,p=d?0:[],q=0,r=a.length;r>q;q++){var s=a[q];if("M"==s[0])e=i=s[1],f=j=s[2];else{"C"==s[0]?(n=[e,f].concat(s.slice(1)),e=n[6],f=n[7]):(n=[e,f,e,f,i,j,i,j],e=i,f=j);for(var t=0,u=b.length;u>t;t++){var v=b[t];if("M"==v[0])g=l=v[1],h=m=v[2];else{"C"==v[0]?(o=[g,h].concat(v.slice(1)),g=o[6],h=o[7]):(o=[g,h,g,h,l,m,l,m],g=l,h=m);var w=k(n,o,d);if(d)p+=w;else{for(var x=0,y=w.length;y>x;x++)w[x].segment1=q,w[x].segment2=t,w[x].bez1=n,w[x].bez2=o;p=p.concat(w)}}}}}return p}function m(a,b,c,d,e,f){null!=a?(this.a=+a,this.b=+b,this.c=+c,this.d=+d,this.e=+e,this.f=+f):(this.a=1,this.b=0,this.c=0,this.d=1,this.e=0,this.f=0)}function n(){return this.x+v+this.y+v+this.width+" × "+this.height}function o(a,b,c,d,e,f){function g(a){return((l*a+k)*a+j)*a}function h(a,b){var c=i(a,b);return((o*c+n)*c+m)*c}function i(a,b){var c,d,e,f,h,i;for(e=a,i=0;8>i;i++){if(f=g(e)-a,ab(f)e)return c;if(e>d)return d;for(;d>c;){if(f=g(e),ab(f-a)f?c=e:d=e,e=(d-c)/2+c}return e}var j=3*b,k=3*(d-b)-j,l=1-j-k,m=3*c,n=3*(e-c)-m,o=1-m-n;return h(a,1/(200*f))}function p(a,b){var c=[],d={};if(this.ms=b,this.times=1,a){for(var e in a)a[w](e)&&(d[X(e)]=a[e],c.push(X(e)));c.sort(Ab)}this.anim=d,this.top=c[c.length-1],this.percents=c}function q(a,d,e,f,g,h){e=X(e);var i,j,k,l,n,p,q=a.ms,r={},s={},t={};if(f)for(v=0,x=Ec.length;x>v;v++){var u=Ec[v];if(u.el.id==d.id&&u.anim==a){u.percent!=e?(Ec.splice(v,1),k=1):j=u,d.attr(u.totalOrigin);break}}else f=+s;for(var v=0,x=a.percents.length;x>v;v++){if(a.percents[v]==e||a.percents[v]>f*a.top){e=a.percents[v],n=a.percents[v-1]||0,q=q/a.top*(e-n),l=a.percents[v+1],i=a.anim[e];break}f&&d.attr(a.anim[a.percents[v]])}if(i){if(j)j.initstatus=f,j.start=new Date-j.ms*f;else{for(var A in i)if(i[w](A)&&(yb[w](A)||d.ca[A]))switch(r[A]=d.attr(A),null==r[A]&&(r[A]=xb[A]),s[A]=i[A],yb[A]){case z:t[A]=(s[A]-r[A])/q;break;case"colour":r[A]=c.getRGB(r[A]);var B=c.getRGB(s[A]);t[A]={r:(B.r-r[A].r)/q,g:(B.g-r[A].g)/q,b:(B.b-r[A].b)/q};break;case"path":var C=cc(r[A],s[A]),D=C[1];for(r[A]=C[0],t[A]=[],v=0,x=r[A].length;x>v;v++){t[A][v]=[0];for(var F=1,G=r[A][v].length;G>F;F++)t[A][v][F]=(D[v][F]-r[A][v][F])/q}break;case"transform":var H=d._,I=hc(H[A],s[A]);if(I)for(r[A]=I.from,s[A]=I.to,t[A]=[],t[A].real=!0,v=0,x=r[A].length;x>v;v++)for(t[A][v]=[r[A][v][0]],F=1,G=r[A][v].length;G>F;F++)t[A][v][F]=(s[A][v][F]-r[A][v][F])/q;else{var J=d.matrix||new m,K={_:{transform:H.transform},getBBox:function(){return d.getBBox(1)}};r[A]=[J.a,J.b,J.c,J.d,J.e,J.f],fc(K,s[A]),s[A]=K._.transform,t[A]=[(K.matrix.a-J.a)/q,(K.matrix.b-J.b)/q,(K.matrix.c-J.c)/q,(K.matrix.d-J.d)/q,(K.matrix.e-J.e)/q,(K.matrix.f-J.f)/q]}break;case"csv":var L=W(i[A])[E](mb),M=W(r[A])[E](mb);if("clip-rect"==A)for(r[A]=M,t[A]=[],v=M.length;v--;)t[A][v]=(L[v]-r[A][v])/q;s[A]=L;break;default:for(L=[][y](i[A]),M=[][y](r[A]),t[A]=[],v=d.ca[A].length;v--;)t[A][v]=((L[v]||0)-(M[v]||0))/q}var N=i.easing,O=c.easing_formulas[N];if(!O)if(O=W(N).match(pb),O&&5==O.length){var P=O;O=function(a){return o(a,+P[1],+P[2],+P[3],+P[4],q)}}else O=Cb;if(p=i.start||a.start||+new Date,u={anim:a,percent:e,timestamp:p,start:p+(a.del||0),status:0,initstatus:f||0,stop:!1,ms:q,easing:O,from:r,diff:t,to:s,el:d,callback:i.callback,prev:n,next:l,repeat:h||a.times,origin:d.attr(),totalOrigin:g},Ec.push(u),f&&!j&&!k&&(u.stop=!0,u.start=new Date-q*f,1==Ec.length))return Gc();k&&(u.start=new Date-u.ms*f),1==Ec.length&&Fc(Gc)}b("raphael.anim.start."+d.id,d,a)}}function r(a){for(var b=0;be;e++)for(i=a[e],f=1,h=i.length;h>f;f+=2)c=b.x(i[f],i[f+1]),d=b.y(i[f],i[f+1]),i[f]=c,i[f+1]=d;return a},Hb=(c.pick=function(){for(var a,b=0,c=arguments.length;c>b;b+=1)if(a=arguments[b],a||a===!1||0===a)return a;return t},c._lastArgIfGroup=function(a,b){var d=a.length-1,e=a[d];return e&&e.constructor===c.el.constructor&&"group"===e.type?(b&&K.call(a,d,1),e):void 0}),Ib=c.merge=function(a,b,c,d,e){var f,g,h,i,j;if(e?(d.push(a),e.push(b)):(d=[a],e=[b]),b instanceof Array)for(f=0;f ',Lb=Mb.firstChild,Lb.style.behavior="url(#default#VML)",!Lb||typeof Lb.adj!=C)return c.type=u;Mb=null}c.svg=!((c.vml="VML"==c.type)||(c.canvas="CANVAS"==c.type)),c._Paper=R,c._id=0,c._oid=0,c.angle=function(a,b,d,e,f,g){if(null==f){var h=a-d,i=b-e;return h||i?(180+Z.atan2(-i,-h)*ib+360)%360:0}return c.angle(a,b,f,g)-c.angle(d,e,f,g)},c.rad=function(a){return a%360*hb},c.deg=function(a){return a*ib%360},c.snapTo=function(a,b,c){var d,e;if(Jb(c,D)||(c=10),Jb(a,B)){for(e=a.length;e--;)if(ab(a[e]-b)<=c)return a[e]}else{if(a=+a,d=b%a,c>d)return b-d;if(d>a-c)return b-d+a}return b},c.setWindow=function(a){b("raphael.setWindow",c,L.win,a),O=L.win=a,N=L.doc=L.win.document,c._engine.initWin&&c._engine.initWin(L.win)};var Nb=function(a){if(c.vml){var b,d=/^\s+|\s+$/g;try{var e=new ActiveXObject("htmlfile");e.write(""),e.close(),b=e.body}catch(f){b=createPopup().document.body}var g=b.createTextRange();Nb=Tb(function(a){try{b.style.color=W(a).replace(d,u);var c=g.queryCommandValue("ForeColor");return c=(255&c)<<16|65280&c|(16711680&c)>>>16,"#"+("000000"+c.toString(16)).slice(-6)}catch(e){return F}})}else{var h=L.doc.createElement("i");h.title="Raphaël Colour Picker",h.style.display=F,L.doc.body.appendChild(h),Nb=Tb(function(a){return h.style.color=a,L.doc.defaultView.getComputedStyle(h,u).getPropertyValue("color")})}return Nb(a)},Ob=function(){return"hsb("+[this.h,this.s,this.b]+")"},Pb=function(){return"hsl("+[this.h,this.s,this.l]+")"},Qb=function(){return this.hex},Rb=function(a,b,d){if(null==b&&Jb(a,C)&&"r"in a&&"g"in a&&"b"in a&&(d=a.b,b=a.g,a=a.r),null==b&&Jb(a,A)){var e=c.getRGB(a);a=e.r,b=e.g,d=e.b}return(a>1||b>1||d>1)&&(a/=255,b/=255,d/=255),[a,b,d]},Sb=function(a,b,d,e){var f={r:a*=255,g:b*=255,b:d*=255,hex:c.rgb(a,b,d),toString:Qb};return Jb(e,"finite")&&(f.opacity=e),f};c.color=function(a){var b;return c.is(a,C)&&"h"in a&&"s"in a&&"b"in a?(b=c.hsb2rgb(a),a.r=b.r,a.g=b.g,a.b=b.b,a.hex=b.hex):c.is(a,C)&&"h"in a&&"s"in a&&"l"in a?(b=c.hsl2rgb(a),a.r=b.r,a.g=b.g,a.b=b.b,a.hex=b.hex):(c.is(a,"string")&&(a=c.getRGB(a)),c.is(a,C)&&"r"in a&&"g"in a&&"b"in a?(b=c.rgb2hsl(a),a.h=b.h,a.s=b.s,a.l=b.l,b=c.rgb2hsb(a),a.v=b.b):(a={hex:F},a.r=a.g=a.b=a.h=a.s=a.v=a.l=-1)),a.toString=Qb,a},c.hsb2rgb=function(a,b,c,d){this.is(a,C)&&"h"in a&&"s"in a&&"b"in a&&(c=a.b,b=a.s,a=a.h,d=a.o),a*=360;var e,f,g,h,i;return a=a%360/60,i=c*b,h=i*(1-ab(a%2-1)),e=f=g=c-i,a=~~a,e+=[i,h,0,0,h,i][a],f+=[h,i,i,h,0,0][a],g+=[0,0,h,i,i,h][a],Sb(e,f,g,d)},c.hsl2rgb=function(a,b,c,d){this.is(a,C)&&"h"in a&&"s"in a&&"l"in a&&(c=a.l,b=a.s,a=a.h),(a>1||b>1||c>1)&&(a/=360,b/=100,c/=100),a*=360;var e,f,g,h,i;return a=a%360/60,i=2*b*(.5>c?c:1-c),h=i*(1-ab(a%2-1)),e=f=g=c-i/2,a=~~a,e+=[i,h,0,0,h,i][a],f+=[h,i,i,h,0,0][a],g+=[0,0,h,i,i,h][a],Sb(e,f,g,d)},c.rgb2hsb=function(a,b,c){c=Rb(a,b,c),a=c[0],b=c[1],c=c[2];var d,e,f,g;return f=$(a,b,c),g=f-_(a,b,c),d=0==g?null:f==a?(b-c)/g:f==b?(c-a)/g+2:(a-b)/g+4,d=60*((d+360)%6)/360,e=0==g?0:g/f,{h:d,s:e,b:f,toString:Ob}},c.rgb2hsl=function(a,b,c){c=Rb(a,b,c),a=c[0],b=c[1],c=c[2];var d,e,f,g,h,i;return g=$(a,b,c),h=_(a,b,c),i=g-h,d=0==i?null:g==a?(b-c)/i:g==b?(c-a)/i+2:(a-b)/i+4,d=60*((d+360)%6)/360,f=(g+h)/2,e=0==i?0:.5>f?i/(2*f):i/(2-2*f),{h:d,s:e,l:f,toString:Pb}},c._path2string=function(){return this.join(",").replace(rb,"$1")};var Tb=c._cacher=function(a,b,c){function e(){var f=J.call(arguments,0),g=f.join("␀"),h=e.cache=e.cache||{},i=e.count=e.count||[];return h[w](g)?(d(i,g),c?c(h[g]):h[g]):(i.length>=1e3&&delete h[i.shift()],i.push(g),h[g]=a[x](b,f),c?c(h[g]):h[g])}return e};c._preload=function(a,b){var c=N.createElement("img");c.style.cssText="position:absolute;left:-9999em;top:-9999em",c.onload=function(){b.call(this),this.onload=null,N.body.removeChild(this)},c.onerror=function(){N.body.removeChild(this)},N.body.appendChild(c),c.src=a},c.getRGB=Tb(function(a){var b,d,f,g,h,i,j;return a&&Jb(a,"object")&&"opacity"in a&&(b=a.opacity),!a||(a=W(a)).indexOf("-")+1?{r:-1,g:-1,b:-1,hex:F,error:1,toString:e}:a==F?{r:-1,g:-1,b:-1,hex:F,toString:e}:(!(wb[w](a.toLowerCase().substring(0,2))||"#"===a.charAt())&&(a=Nb(a)),(j=a.match(ob))?(j[2]&&(g=Y(j[2].substring(5),16),f=Y(j[2].substring(3,5),16),d=Y(j[2].substring(1,3),16)),j[3]&&(g=Y((h=j[3].charAt(3))+h,16),f=Y((h=j[3].charAt(2))+h,16),d=Y((h=j[3].charAt(1))+h,16)),j[4]&&(i=j[4][E](qb),d=X(i[0]),"%"==i[0].slice(-1)&&(d*=2.55),f=X(i[1]),"%"==i[1].slice(-1)&&(f*=2.55),g=X(i[2]),"%"==i[2].slice(-1)&&(g*=2.55),"rgba"==j[1].toLowerCase().slice(0,4)&&(b=X(i[3])),i[3]&&"%"==i[3].slice(-1)&&(b/=100)),j[5]?(i=j[5][E](qb),d=X(i[0]),"%"==i[0].slice(-1)&&(d*=2.55),f=X(i[1]),"%"==i[1].slice(-1)&&(f*=2.55),g=X(i[2]),"%"==i[2].slice(-1)&&(g*=2.55),("deg"==i[0].slice(-3)||"°"==i[0].slice(-1))&&(d/=360),"hsba"==j[1].toLowerCase().slice(0,4)&&(b=X(i[3])),i[3]&&"%"==i[3].slice(-1)&&(b/=100),c.hsb2rgb(d,f,g,b)):j[6]?(i=j[6][E](qb),d=X(i[0]),"%"==i[0].slice(-1)&&(d*=2.55),f=X(i[1]),"%"==i[1].slice(-1)&&(f*=2.55),g=X(i[2]),"%"==i[2].slice(-1)&&(g*=2.55),("deg"==i[0].slice(-3)||"°"==i[0].slice(-1))&&(d/=360),"hsla"==j[1].toLowerCase().slice(0,4)&&(b=X(i[3])),i[3]&&"%"==i[3].slice(-1)&&(b/=100),c.hsl2rgb(d,f,g,b)):(j={r:d,g:f,b:g,toString:e},j.hex="#"+(16777216|g|f<<8|d<<16).toString(16).slice(1),c.is(b,"finite")&&(j.opacity=b),j)):{r:-1,g:-1,b:-1,hex:F,error:1,toString:e})},c),c.tintshade=Tb(function(a,b){var d,f=c.getRGB(a),g=255;return 0>b&&(b*=-1,g=0),b>1&&(b=1),d=0===b?f:{r:g-(g-f.r)*b,g:g-(g-f.g)*b,b:g-(g-f.b)*b,toString:e},d.hex=c.rgb(d.r,d.g,d.b),f.error&&(d.error=f.error),"opacity"in f?(d.rgba="rgba("+[d.r,d.g,d.b,f.opacity].join(",")+")",d.opacity=f.opacity):d.rgba="rgb("+[d.r,d.g,d.b].join(",")+")",d},c),c.hsb=Tb(function(a,b,d){return c.hsb2rgb(a,b,d).hex}),c.hsl=Tb(function(a,b,d){return c.hsl2rgb(a,b,d).hex}),c.rgb=Tb(function(a,b,c){return"#"+(16777216|c|b<<8|a<<16).toString(16).slice(1)}),c.getColor=function(a){var b=this.getColor.start=this.getColor.start||{h:0,s:1,b:a||.75},c=this.hsb2rgb(b.h,b.s,b.b);return b.h+=.075,b.h>1&&(b.h=0,b.s-=.2,b.s<=0&&(this.getColor.start={h:0,s:1,b:b.b})),c.hex},c.getColor.reset=function(){delete this.start},c.parsePathString=function(a){if(!a)return null;var b=Ub(a);if(b.arr)return Wb(b.arr);var d={a:7,c:6,h:1,l:2,m:2,r:4,q:4,s:4,t:2,v:1,z:0},e=[];return c.is(a,B)&&c.is(a[0],B)&&(e=Wb(a)),e.length||W(a).replace(sb,function(a,b,c){var f=[],g=b.toLowerCase();if(c.replace(ub,function(a,b){b&&f.push(+b)}),"m"==g&&f.length>2&&(e.push([b][y](f.splice(0,2))),g="l",b="m"==b?"l":"L"),"r"==g)e.push([b][y](f));else for(;f.length>=d[g]&&(e.push([b][y](f.splice(0,d[g]))),d[g]););}),e.toString=c._path2string,b.arr=Wb(e),e},c.parseTransformString=Tb(function(a){if(!a)return null;var b=[];return c.is(a,B)&&c.is(a[0],B)&&(b=Wb(a)),b.length||W(a).replace(tb,function(a,c,d){var e=[];jb.call(c),d.replace(ub,function(a,b){b&&e.push(+b)}),b.push([c][y](e))}),b.toString=c._path2string,b});var Ub=function(a){var b=Ub.ps=Ub.ps||{};return b[a]?b[a].sleep=100:b[a]={sleep:100},setTimeout(function(){for(var c in b)b[w](c)&&c!=a&&(b[c].sleep--,!b[c].sleep&&delete b[c])}),b[a]};c.findDotsAtSegment=function(a,b,c,d,e,f,g,h,i){var j=1-i,k=bb(j,3),l=bb(j,2),m=i*i,n=m*i,o=k*a+3*l*i*c+3*j*i*i*e+n*g,p=k*b+3*l*i*d+3*j*i*i*f+n*h,q=a+2*i*(c-a)+m*(e-2*c+a),r=b+2*i*(d-b)+m*(f-2*d+b),s=c+2*i*(e-c)+m*(g-2*e+c),t=d+2*i*(f-d)+m*(h-2*f+d),u=j*a+i*c,v=j*b+i*d,w=j*e+i*g,x=j*f+i*h,y=90-180*Z.atan2(q-s,r-t)/gb;return(q>s||t>r)&&(y+=180),{x:o,y:p,m:{x:q,y:r},n:{x:s,y:t},start:{x:u,y:v},end:{x:w,y:x},alpha:y}},c.bezierBBox=function(a,b,d,e,f,g,h,i){c.is(a,"array")||(a=[a,b,d,e,f,g,h,i]);var j=bc.apply(null,a);return{x:j.min.x,y:j.min.y,x2:j.max.x,y2:j.max.y,width:j.max.x-j.min.x,height:j.max.y-j.min.y}},c.isPointInsideBBox=function(a,b,c){return b>=a.x&&b<=a.x2&&c>=a.y&&c<=a.y2},c.isBBoxIntersect=function(a,b){var d=c.isPointInsideBBox;return d(b,a.x,a.y)||d(b,a.x2,a.y)||d(b,a.x,a.y2)||d(b,a.x2,a.y2)||d(a,b.x,b.y)||d(a,b.x2,b.y)||d(a,b.x,b.y2)||d(a,b.x2,b.y2)||(a.xb.x||b.xa.x)&&(a.yb.y||b.ya.y)},c.pathIntersection=function(a,b){return l(a,b)},c.pathIntersectionNumber=function(a,b){return l(a,b,1)},c.isPointInsidePath=function(a,b,d){var e=c.pathBBox(a);return c.isPointInsideBBox(e,b,d)&&(1==l(a,[["M",b,d],["H",e.x2+10]],1)%2||1==l(a,[["M",b,d],["V",e.y2+10]],1)%2)},c._removedFactory=function(a){return function(){b("raphael.log",null,"Raphaël: you are calling to method “"+a+"” of removed object",a)}};var Vb=c.pathBBox=function(a){var b=Ub(a);if(b.bbox)return b.bbox;if(!a)return{x:0,y:0,width:0,height:0,x2:0,y2:0};a=cc(a);for(var c,d=0,e=0,f=[],g=[],h=0,i=a.length;i>h;h++)if(c=a[h],"M"==c[0])d=c[1],e=c[2],f.push(d),g.push(e);else{var j=bc(d,e,c[1],c[2],c[3],c[4],c[5],c[6]);f=f[y](j.min.x,j.max.x),g=g[y](j.min.y,j.max.y),d=c[5],e=c[6]}var k=_[x](0,f),l=_[x](0,g),m=$[x](0,f),n=$[x](0,g),o={x:k,y:l,x2:m,y2:n,width:m-k,height:n-l};return b.bbox=Kb(o),o},Wb=function(a){var b=Kb(a);return b.toString=c._path2string,b},Xb=c._pathToRelative=function(a){var b=Ub(a);if(b.rel)return Wb(b.rel);c.is(a,B)&&c.is(a&&a[0],B)||(a=c.parsePathString(a));var d=[],e=0,f=0,g=0,h=0,i=0;"M"==a[0][0]&&(e=a[0][1],f=a[0][2],g=e,h=f,i++,d.push(["M",e,f]));for(var j=i,k=a.length;k>j;j++){var l=d[j]=[],m=a[j];if(m[0]!=jb.call(m[0]))switch(l[0]=jb.call(m[0]),l[0]){case"a":l[1]=m[1],l[2]=m[2],l[3]=m[3],l[4]=m[4],l[5]=m[5],l[6]=+(m[6]-e).toFixed(3),l[7]=+(m[7]-f).toFixed(3);break;case"v":l[1]=+(m[1]-f).toFixed(3);break;case"m":g=m[1],h=m[2];default:for(var n=1,o=m.length;o>n;n++)l[n]=+(m[n]-(n%2?e:f)).toFixed(3)}else{l=d[j]=[],"m"==m[0]&&(g=m[1]+e,h=m[2]+f);for(var p=0,q=m.length;q>p;p++)d[j][p]=m[p]}var r=d[j].length;switch(d[j][0]){case"z":e=g,f=h;break;case"h":e+=+d[j][r-1];break;case"v":f+=+d[j][r-1];break;default:e+=+d[j][r-2],f+=+d[j][r-1]}}return d.toString=c._path2string,b.rel=Wb(d),d},Yb=c._pathToAbsolute=function(a){var b,d=Ub(a);if(d.abs)return Wb(d.abs);if(c.is(a,B)&&c.is(a&&a[0],B)||(a=c.parsePathString(a)),!a||!a.length)return b=["M",0,0],b.toString=c._path2string,b;var e=0,g=0,h=0,i=0,j=0;b=[],"M"==a[0][0]&&(e=+a[0][1],g=+a[0][2],h=e,i=g,j++,b[0]=["M",e,g]);for(var k,l,m=3==a.length&&"M"==a[0][0]&&"R"==a[1][0].toUpperCase()&&"Z"==a[2][0].toUpperCase(),n=j,o=a.length;o>n;n++){if(b.push(k=[]),l=a[n],l[0]!=kb.call(l[0]))switch(k[0]=kb.call(l[0]),k[0]){case"A":k[1]=l[1],k[2]=l[2],k[3]=l[3],k[4]=l[4],k[5]=l[5],k[6]=+(l[6]+e),k[7]=+(l[7]+g);break;case"V":k[1]=+l[1]+g;break;case"H":k[1]=+l[1]+e;break;case"R":for(var p=[e,g][y](l.slice(1)),q=2,r=p.length;r>q;q++)p[q]=+p[q]+e,p[++q]=+p[q]+g;b.pop(),b=b[y](f(p,m));break;case"M":h=+l[1]+e,i=+l[2]+g;default:for(q=1,r=l.length;r>q;q++)k[q]=+l[q]+(q%2?e:g)}else if("R"==l[0])p=[e,g][y](l.slice(1)),b.pop(),b=b[y](f(p,m)),k=["R"][y](l.slice(-2));else for(var s=0,t=l.length;t>s;s++)k[s]=l[s];switch(k[0]){case"Z":e=h,g=i;break;case"H":e=k[1];break;case"V":g=k[1];break;case"M":h=k[k.length-2],i=k[k.length-1];default:e=k[k.length-2],g=k[k.length-1]}}return b.toString=c._path2string,d.abs=Wb(b),b},Zb=function(a,b,c,d){return[a,b,c,d,c,d]},$b=function(a,b,c,d,e,f){var g=1/3,h=2/3;return[g*a+h*c,g*b+h*d,g*e+h*c,g*f+h*d,e,f]},_b=function(a,b,c,d,e,f,g,h,i,j){var k,l=120*gb/180,m=hb*(+e||0),n=[],o=Tb(function(a,b,c){var d=a*cb(c)-b*db(c),e=a*db(c)+b*cb(c);return{x:d,y:e}});if(j)x=j[0],z=j[1],v=j[2],w=j[3];else{k=o(a,b,-m),a=k.x,b=k.y,k=o(h,i,-m),h=k.x,i=k.y;var p=(cb(hb*e),db(hb*e),(a-h)/2),q=(b-i)/2,r=p*p/(c*c)+q*q/(d*d);r>1&&(r=eb(r),c=r*c,d=r*d);var s=c*c,t=d*d,u=(f==g?-1:1)*eb(ab((s*t-s*q*q-t*p*p)/(s*q*q+t*p*p))),v=u*c*q/d+(a+h)/2,w=u*-d*p/c+(b+i)/2,x=Z.asin(((b-w)/d).toFixed(9)),z=Z.asin(((i-w)/d).toFixed(9));x=v>a?gb-x:x,z=v>h?gb-z:z,0>x&&(x=2*gb+x),0>z&&(z=2*gb+z),g&&x>z&&(x-=2*gb),!g&&z>x&&(z-=2*gb)}var A=z-x;if(ab(A)>l){var B=z,C=h,D=i;z=x+l*(g&&z>x?1:-1),h=v+c*cb(z),i=w+d*db(z),n=_b(h,i,c,d,e,0,g,C,D,[z,B,v,w])}A=z-x;var F=cb(x),G=db(x),H=cb(z),I=db(z),J=Z.tan(A/4),K=4/3*c*J,L=4/3*d*J,M=[a,b],N=[a+K*G,b-L*F],O=[h+K*I,i-L*H],P=[h,i];if(N[0]=2*M[0]-N[0],N[1]=2*M[1]-N[1],j)return[N,O,P][y](n);n=[N,O,P][y](n).join()[E](",");for(var Q=[],R=0,S=n.length;S>R;R++)Q[R]=R%2?o(n[R-1],n[R],m).y:o(n[R],n[R+1],m).x;return Q},ac=function(a,b,c,d,e,f,g,h,i){var j=1-i;return{x:bb(j,3)*a+3*bb(j,2)*i*c+3*j*i*i*e+bb(i,3)*g,y:bb(j,3)*b+3*bb(j,2)*i*d+3*j*i*i*f+bb(i,3)*h}},bc=Tb(function(a,b,c,d,e,f,g,h){var i,j=e-2*c+a-(g-2*e+c),k=2*(c-a)-2*(e-c),l=a-c,m=(-k+eb(k*k-4*j*l))/2/j,n=(-k-eb(k*k-4*j*l))/2/j,o=[b,h],p=[a,g];return ab(m)>"1e12"&&(m=.5),ab(n)>"1e12"&&(n=.5),m>0&&1>m&&(i=ac(a,b,c,d,e,f,g,h,m),p.push(i.x),o.push(i.y)),n>0&&1>n&&(i=ac(a,b,c,d,e,f,g,h,n),p.push(i.x),o.push(i.y)),j=f-2*d+b-(h-2*f+d),k=2*(d-b)-2*(f-d),l=b-d,m=(-k+eb(k*k-4*j*l))/2/j,n=(-k-eb(k*k-4*j*l))/2/j,ab(m)>"1e12"&&(m=.5),ab(n)>"1e12"&&(n=.5),m>0&&1>m&&(i=ac(a,b,c,d,e,f,g,h,m),p.push(i.x),o.push(i.y)),n>0&&1>n&&(i=ac(a,b,c,d,e,f,g,h,n),p.push(i.x),o.push(i.y)),{min:{x:_[x](0,p),y:_[x](0,o)},max:{x:$[x](0,p),y:$[x](0,o)}}}),cc=c._path2curve=Tb(function(a,b){var c=!b&&Ub(a);if(!b&&c.curve)return Wb(c.curve);for(var d=Yb(a),e=b&&Yb(b),f={x:0,y:0,bx:0,by:0,X:0,Y:0,qx:null,qy:null},g={x:0,y:0,bx:0,by:0,X:0,Y:0,qx:null,qy:null},h=(function(a,b){var c,d;if(!a)return["C",b.x,b.y,b.x,b.y,b.x,b.y];switch(!(a[0]in{T:1,Q:1})&&(b.qx=b.qy=null),a[0]){case"M":b.X=a[1],b.Y=a[2];break;case"A":a=["C"][y](_b[x](0,[b.x,b.y][y](a.slice(1))));break;case"S":c=b.x+(b.x-(b.bx||b.x)),d=b.y+(b.y-(b.by||b.y)),a=["C",c,d][y](a.slice(1));break;case"T":b.qx=b.x+(b.x-(b.qx||b.x)),b.qy=b.y+(b.y-(b.qy||b.y)),a=["C"][y]($b(b.x,b.y,b.qx,b.qy,a[1],a[2]));break;case"Q":b.qx=a[1],b.qy=a[2],a=["C"][y]($b(b.x,b.y,a[1],a[2],a[3],a[4]));break;case"L":a=["C"][y](Zb(b.x,b.y,a[1],a[2]));break;case"H":a=["C"][y](Zb(b.x,b.y,a[1],b.y));break;case"V":a=["C"][y](Zb(b.x,b.y,b.x,a[1]));break;case"Z":a=["C"][y](Zb(b.x,b.y,b.X,b.Y))}return a}),i=function(a,b){if(a[b].length>7){a[b].shift();for(var c=a[b];c.length;)a.splice(b++,0,["C"][y](c.splice(0,6)));a.splice(b,1),l=$(d.length,e&&e.length||0)}},j=function(a,b,c,f,g){a&&b&&"M"==a[g][0]&&"M"!=b[g][0]&&(b.splice(g,0,["M",f.x,f.y]),c.bx=0,c.by=0,c.x=a[g][1],c.y=a[g][2],l=$(d.length,e&&e.length||0))},k=0,l=$(d.length,e&&e.length||0);l>k;k++){d[k]=h(d[k],f),i(d,k),e&&(e[k]=h(e[k],g)),e&&i(e,k),j(d,e,f,g,k),j(e,d,g,f,k);var m=d[k],n=e&&e[k],o=m.length,p=e&&n.length;f.x=m[o-2],f.y=m[o-1],f.bx=X(m[o-4])||f.x,f.by=X(m[o-3])||f.y,g.bx=e&&(X(n[p-4])||g.x),g.by=e&&(X(n[p-3])||g.y),g.x=e&&n[p-2],g.y=e&&n[p-1]}return e||(c.curve=Wb(d)),e?[d,e]:d},null,Wb),dc=(c._parseDots=Tb(function(a){for(var b=[],d=0,e=a.length;e>d;d++){var f={},g=a[d].match(/^([^:]*):?([\d\.]*)/);if(f.color=c.getRGB(g[1]),f.color.error)return null;f.opacity=f.color.opacity,f.color=f.color.hex,g[2]&&(f.offset=g[2]+"%"),b.push(f)}for(d=1,e=b.length-1;e>d;d++)if(!b[d].offset){for(var h=X(b[d-1].offset||0),i=0,j=d+1;e>j;j++)if(b[j].offset){i=b[j].offset;break}i||(i=100,j=e),i=X(i);for(var k=(i-h)/(j-d+1);j>d;d++)h+=k,b[d].offset=h+"%"}return b}),c._tear=function(a,b){a==b.top&&(b.top=a.prev),a==b.bottom&&(b.bottom=a.next),a.next&&(a.next.prev=a.prev),a.prev&&(a.prev.next=a.next)}),ec=(c._tofront=function(a,b){return b.top===a?!1:(dc(a,b),a.next=null,a.prev=b.top,b.top.next=a,b.top=a,!0)},c._toback=function(a,b){return b.bottom===a?!1:(dc(a,b),a.next=b.bottom,a.prev=null,b.bottom.prev=a,b.bottom=a,!0)},c._insertafter=function(a,b,c,d){dc(a,c),a.parent=d,b===d.top&&(d.top=a),b.next&&(b.next.prev=a),a.next=b.next,a.prev=b,b.next=a},c._insertbefore=function(a,b,c,d){dc(a,c),a.parent=d,b===d.bottom&&(d.bottom=a),b.prev&&(b.prev.next=a),a.prev=b.prev,b.prev=a,a.next=b},c.toMatrix=function(a,b){var c=Vb(a),d={_:{transform:u},getBBox:function(){return c}};return fc(d,b),d.matrix}),fc=(c.transformPath=function(a,b){return Gb(a,ec(a,b))},c._extractTransform=function(a,b){if(null==b)return a._.transform;
+b=W(b).replace(/\.{3}|\u2026/g,a._.transform||u);var d=c.parseTransformString(b),e=0,f=0,g=0,h=1,i=1,j=a._,k=new m;if(j.transform=d||[],d)for(var l=0,n=d.length;n>l;l++){var o,p,q,r,s,t=d[l],v=t.length,w=W(t[0]).toLowerCase(),x=t[0]!=w,y=x?k.invert():0;"t"==w&&3==v?x?(o=y.x(0,0),p=y.y(0,0),q=y.x(t[1],t[2]),r=y.y(t[1],t[2]),k.translate(q-o,r-p)):k.translate(t[1],t[2]):"r"==w?2==v?(s=s||a.getBBox(1),k.rotate(t[1],s.x+s.width/2,s.y+s.height/2),e+=t[1]):4==v&&(x?(q=y.x(t[2],t[3]),r=y.y(t[2],t[3]),k.rotate(t[1],q,r)):k.rotate(t[1],t[2],t[3]),e+=t[1]):"s"==w?2==v||3==v?(s=s||a.getBBox(1),k.scale(t[1],t[v-1],s.x+s.width/2,s.y+s.height/2),h*=t[1],i*=t[v-1]):5==v&&(x?(q=y.x(t[3],t[4]),r=y.y(t[3],t[4]),k.scale(t[1],t[2],q,r)):k.scale(t[1],t[2],t[3],t[4]),h*=t[1],i*=t[2]):"m"==w&&7==v&&k.add(t[1],t[2],t[3],t[4],t[5],t[6]),j.dirtyT=1,a.matrix=k}a.matrix=k,j.sx=h,j.sy=i,j.deg=e,j.dx=f=k.e,j.dy=g=k.f,1==h&&1==i&&!e&&j.bbox?(j.bbox.x+=+f,j.bbox.y+=+g):j.dirtyT=1}),gc=function(a){var b=a[0];switch(b.toLowerCase()){case"t":return[b,0,0];case"m":return[b,1,0,0,1,0,0];case"r":return 4==a.length?[b,0,a[2],a[3]]:[b,0];case"s":return 5==a.length?[b,1,1,a[3],a[4]]:3==a.length?[b,1,1]:[b,1]}},hc=c._equaliseTransform=function(a,b){b=W(b).replace(/\.{3}|\u2026/g,a),a=c.parseTransformString(a)||[],b=c.parseTransformString(b)||[];for(var d,e,f,g,h=$(a.length,b.length),i=[],j=[],k=0;h>k;k++){if(f=a[k]||gc(b[k]),g=b[k]||gc(f),f[0]!=g[0]||"r"==f[0].toLowerCase()&&(f[2]!=g[2]||f[3]!=g[3])||"s"==f[0].toLowerCase()&&(f[3]!=g[3]||f[4]!=g[4]))return;for(i[k]=[],j[k]=[],d=0,e=$(f.length,g.length);e>d;d++)d in f&&(i[k][d]=f[d]),d in g&&(j[k][d]=g[d])}return{from:i,to:j}};c._getContainer=function(a,b,d,e){var f;return f=null!=e||c.is(a,C)?a:L.doc.getElementById(a),null!=f?f.tagName?null==b?{container:f,width:f.style.pixelWidth||f.offsetWidth,height:f.style.pixelHeight||f.offsetHeight}:{container:f,width:b,height:d}:{container:1,x:a,y:b,width:d,height:e}:void 0},c.pathToRelative=Xb,c._engine={},c.path2curve=cc,c.matrix=function(a,b,c,d,e,f){return new m(a,b,c,d,e,f)},function(a){function b(a){return a[0]*a[0]+a[1]*a[1]}function d(a){var c=eb(b(a));a[0]&&(a[0]/=c),a[1]&&(a[1]/=c)}a.add=function(a,b,c,d,e,f){var g,h,i,j,k=[[],[],[]],l=[[this.a,this.c,this.e],[this.b,this.d,this.f],[0,0,1]],n=[[a,c,e],[b,d,f],[0,0,1]];for(a&&a instanceof m&&(n=[[a.a,a.c,a.e],[a.b,a.d,a.f],[0,0,1]]),g=0;3>g;g++)for(h=0;3>h;h++){for(j=0,i=0;3>i;i++)j+=l[g][i]*n[i][h];k[g][h]=j}this.a=k[0][0],this.b=k[1][0],this.c=k[0][1],this.d=k[1][1],this.e=k[0][2],this.f=k[1][2]},a.invert=function(){var a=this,b=a.a*a.d-a.b*a.c;return new m(a.d/b,-a.b/b,-a.c/b,a.a/b,(a.c*a.f-a.d*a.e)/b,(a.b*a.e-a.a*a.f)/b)},a.clone=function(){return new m(this.a,this.b,this.c,this.d,this.e,this.f)},a.translate=function(a,b){this.add(1,0,0,1,a,b)},a.scale=function(a,b,c,d){null==b&&(b=a),(c||d)&&this.add(1,0,0,1,c,d),this.add(a,0,0,b,0,0),(c||d)&&this.add(1,0,0,1,-c,-d)},a.rotate=function(a,b,d){a=c.rad(a),b=b||0,d=d||0;var e=+cb(a).toFixed(9),f=+db(a).toFixed(9);this.add(e,f,-f,e,b,d),this.add(1,0,0,1,-b,-d)},a.x=function(a,b){return a*this.a+b*this.c+this.e},a.y=function(a,b){return a*this.b+b*this.d+this.f},a.get=function(a){return+this[W.fromCharCode(97+a)].toFixed(4)},a.toString=function(){return c.svg?"matrix("+[this.get(0),this.get(1),this.get(2),this.get(3),this.get(4),this.get(5)].join()+")":[this.get(0),this.get(2),this.get(1),this.get(3),0,0].join()},a.toMatrixString=function(){return"matrix("+[this.get(0),this.get(1),this.get(2),this.get(3),this.get(4),this.get(5)].join()+")"},a.toFilter=function(){return"progid:DXImageTransform.Microsoft.Matrix(M11="+this.get(0)+", M12="+this.get(2)+", M21="+this.get(1)+", M22="+this.get(3)+", Dx="+this.get(4)+", Dy="+this.get(5)+", sizingmethod='auto expand')"},a.offset=function(){return[this.e.toFixed(4),this.f.toFixed(4)]},a.split=function(){var a={};a.dx=this.e,a.dy=this.f;var e=[[this.a,this.c],[this.b,this.d]];a.scalex=eb(b(e[0])),d(e[0]),a.shear=e[0][0]*e[1][0]+e[0][1]*e[1][1],e[1]=[e[1][0]-e[0][0]*a.shear,e[1][1]-e[0][1]*a.shear],a.scaley=eb(b(e[1])),d(e[1]),a.shear/=a.scaley;var f=-e[0][1],g=e[1][1];return 0>g?(a.rotate=c.deg(Z.acos(g)),0>f&&(a.rotate=360-a.rotate)):a.rotate=c.deg(Z.asin(f)),a.isSimple=!(+a.shear.toFixed(9)||a.scalex.toFixed(9)!=a.scaley.toFixed(9)&&a.rotate),a.isSuperSimple=!+a.shear.toFixed(9)&&a.scalex.toFixed(9)==a.scaley.toFixed(9)&&!a.rotate,a.noRotation=!+a.shear.toFixed(9)&&!a.rotate,a},a.toTransformString=function(a){var b=a||this[E]();return b.isSimple?(b.scalex=+b.scalex.toFixed(4),b.scaley=+b.scaley.toFixed(4),b.rotate=+b.rotate.toFixed(4),(b.dx||b.dy?"t"+[b.dx,b.dy]:u)+(1!=b.scalex||1!=b.scaley?"s"+[b.scalex,b.scaley,0,0]:u)+(b.rotate?"r"+[b.rotate,0,0]:u)):"m"+[this.get(0),this.get(1),this.get(2),this.get(3),this.get(4),this.get(5)]}}(m.prototype);var ic=navigator.userAgent.match(/Version\/(.*?)\s/)||navigator.userAgent.match(/Chrome\/(\d+)/);S.safari="Apple Computer, Inc."==navigator.vendor&&(ic&&ic[1]<4||"iP"==navigator.platform.slice(0,2))||"Google Inc."==navigator.vendor&&ic&&ic[1]<8?function(){var a=this.rect(-99,-99,this.width+99,this.height+99).attr({stroke:"none"});return setTimeout(function(){a.remove()}),!0}:Bb;for(var jc=function(){this.returnValue=!1},kc=function(){return this.originalEvent.preventDefault()},lc=function(){this.cancelBubble=!0},mc=function(){return this.originalEvent.stopPropagation()},nc=c.addEvent=function(){return L.doc.addEventListener?function(a,b,c,d){var e=P&&V[b]?V[b]:b,f=function(e){var f=L.doc.documentElement.scrollTop||L.doc.body.scrollTop,g=L.doc.documentElement.scrollLeft||L.doc.body.scrollLeft;if(P&&V[w](b))for(var h=0,i=e.targetTouches&&e.targetTouches.length;i>h;h++)if(e.targetTouches[h].target==a){var j=e;e=e.targetTouches[h],e.originalEvent=j,e.preventDefault=kc,e.stopPropagation=mc;break}return c.call(d,e,e.clientX+g,e.clientY+f)};return a.addEventListener(e,f,!1),function(){return a.removeEventListener(e,f,!1),!0}}:L.doc.attachEvent?function(a,b,c,d){var e=function(a){a=a||L.win.event;var b=L.doc.documentElement.scrollTop||L.doc.body.scrollTop,e=L.doc.documentElement.scrollLeft||L.doc.body.scrollLeft,f=a.clientX+e,g=a.clientY+b;return a.preventDefault=a.preventDefault||jc,a.stopPropagation=a.stopPropagation||lc,c.call(d,a,f,g)};a.attachEvent("on"+b,e);var f=function(){return a.detachEvent("on"+b,e),!0};return f}:void 0}(),oc=[],pc=function(a){for(var c,d=a.clientX,e=a.clientY,f=L.doc.documentElement.scrollTop||L.doc.body.scrollTop,g=L.doc.documentElement.scrollLeft||L.doc.body.scrollLeft,h=oc.length;h--;){if(c=oc[h],P){for(var i,j=a.touches.length;j--;)if(i=a.touches[j],i.identifier==c.el._drag.id){d=i.clientX,e=i.clientY,(a.originalEvent?a.originalEvent:a).preventDefault();break}}else a.preventDefault();var k,l=c.el.node,m=l.nextSibling,n=l.parentNode,o=l.style.display;L.win.opera&&n.removeChild(l),l.style.display="none",k=c.el.paper.getElementByPoint(d,e),l.style.display=o,L.win.opera&&(m?n.insertBefore(l,m):n.appendChild(l)),k&&b("raphael.drag.over."+c.el.id,c.el,k),d+=g,e+=f,b("raphael.drag.move."+c.el.id,c.move_scope||c.el,d-c.el._drag.x,e-c.el._drag.y,d,e,a)}},qc=function(a){c.unmousemove(pc).unmouseup(qc);for(var d,e=oc.length;e--;)d=oc[e],d.el._drag={},b("raphael.drag.end."+d.el.id,d.end_scope||d.start_scope||d.move_scope||d.el,a);oc=[]},rc=c.el={},sc=U.length;sc--;)!function(a){c[a]=rc[a]=function(b,d){return c.is(b,"function")&&(this.events=this.events||[],this.events.push({name:a,f:b,unbind:nc(this.shape||this.node||L.doc,a,b,d||this)})),this},c["un"+a]=rc["un"+a]=function(b){for(var c=this.events||[],d=c.length;d--;)if(c[d].name==a&&c[d].f==b)return c[d].unbind(),c.splice(d,1),!c.length&&delete this.events,this;return this}}(U[sc]);rc.data=function(a,d){var e=zb[this.id]=zb[this.id]||{};if(1==arguments.length){if(c.is(a,C)){for(var f in a)a[w](f)&&this.data(f,a[f]);return this}return b("raphael.data.get."+this.id,this,e[a],a),e[a]}return e[a]=d,b("raphael.data.set."+this.id,this,d,a),this},rc.removeData=function(a){return null==a?zb[this.id]={}:zb[this.id]&&delete zb[this.id][a],this},rc.getData=function(){return Kb(zb[this.id]||{})};var tc=[],uc=function(){this.untrack=nc(L.doc,"mouseup",vc,this)},vc=function(){return this.untrack(),this.untrack=null,this.fn&&this.fn.apply(this.scope||this.el,arguments)};rc.mouseup=function(a,b,d){return d?(tc.push(d={el:this,fn:a,scope:b}),d.unbind=nc(this.shape||this.node||L.doc,"mousedown",uc,d),this):c.mouseup.apply(this,arguments)},rc.unmouseup=function(a){for(var b,d=tc.length;d--;)tc[d].el===this&&tc[d].fn===a&&(b=tc[d],b.unbind(),b.untrack&&b.untrack(),tc.splice(d,1));return b?this:c.unmouseup.apply(this,arguments)},rc.hover=function(a,b,c,d){return this.mouseover(a,c).mouseout(b,d||c)},rc.unhover=function(a,b){return this.unmouseover(a).unmouseout(b)};var wc=[];rc.drag=function(a,d,e,f,g,h){function i(i){(i.originalEvent||i).preventDefault();var j=L.doc.documentElement.scrollTop||L.doc.body.scrollTop,k=L.doc.documentElement.scrollLeft||L.doc.body.scrollLeft;this._drag.x=i.clientX+k,this._drag.y=i.clientY+j,this._drag.id=i.identifier,!oc.length&&c.mousemove(pc).mouseup(qc),oc.push({el:this,move_scope:f,start_scope:g,end_scope:h}),d&&b.on("raphael.drag.start."+this.id,d),a&&b.on("raphael.drag.move."+this.id,a),e&&b.on("raphael.drag.end."+this.id,e),b("raphael.drag.start."+this.id,g||f||this,i.clientX+k,i.clientY+j,i)}return this._drag={},wc.push({el:this,start:i}),this.mousedown(i),this},rc.onDragOver=function(a){a?b.on("raphael.drag.over."+this.id,a):b.unbind("raphael.drag.over."+this.id)},rc.undrag=function(){for(var a=wc.length;a--;)wc[a].el==this&&(this.unmousedown(wc[a].start),wc.splice(a,1),b.unbind("raphael.drag.*."+this.id));!wc.length&&c.unmousemove(pc).unmouseup(qc)},rc.follow=function(a,b,d){return a.removed||a.constructor!==c.el.constructor?this:(a.followers.push({el:this,stalk:d={before:"insertBefore",after:"insertAfter"}[d],cb:b}),d&&this[d](a),this)},rc.unfollow=function(a){if(a.removed||a.constructor!==c.el.constructor)return this;for(var b=0,d=a.followers.length;d>b;b++)if(a.followers[b].el===this){a.followers.splice(b,1);break}return this},S.hide=function(){var a=this;return a.canvas.style.visibility="hidden",a},S.show=function(){var a=this;return a.canvas.style.visibility=u,a},S.group=function(){var a,b=this,d=arguments,e=Hb(d,!0);return a=c._engine.group(b,d[0],e),b.__set__&&b.__set__.push(a),a},S.circle=function(){var a,b=this,d=arguments,e=Hb(d,!0);return a=c._engine.circle(b,d[0]||0,d[1]||0,d[2]||0,e),b.__set__&&b.__set__.push(a),a},S.rect=function(){var a,b=this,d=arguments,e=Hb(d,!0);return a=c._engine.rect(b,d[0]||0,d[1]||0,d[2]||0,d[3]||0,d[4]||0,e),b.__set__&&b.__set__.push(a),a},S.ellipse=function(){var a,b=this,d=arguments,e=Hb(d,!0);return a=c._engine.ellipse(this,d[0]||0,d[1]||0,d[2]||0,d[3]||0,e),b.__set__&&b.__set__.push(a),a},S.path=function(){var a,b,d=this,e=arguments,f=Hb(e,!0);return a=e[0],a&&!c.is(a,A)&&!c.is(a[0],B)&&(a+=u),b=c._engine.path(c.format[x](c,arguments),d,f),d.__set__&&d.__set__.push(b),b},S.image=function(){var a,b=this,d=arguments,e=Hb(d,!0);return a=c._engine.image(b,d[0]||"about:blank",d[1]||0,d[2]||0,d[3]||0,d[4]||0,e),b.__set__&&b.__set__.push(a),a},S.text=function(){var a,b=this,d=arguments,e=Hb(d,!0);return a=c._engine.text(b,d[0]||0,d[1]||0,W(d[2]||u),e),b.__set__&&b.__set__.push(a),a},S.set=function(a){!c.is(a,"array")&&(a=K.call(arguments,0,arguments.length));var b=new Ic(a);return this.__set__&&this.__set__.push(b),b},S.setStart=function(a){this.__set__=a||this.set()},S.setFinish=function(){var a=this.__set__;return delete this.__set__,a},S.setSize=function(a,b){return c._engine.setSize.call(this,a,b)},S.setViewBox=function(a,b,d,e,f){return c._engine.setViewBox.call(this,a,b,d,e,f)},S.top=S.bottom=null,S.raphael=c;var xc=function(a){var b=a.getBoundingClientRect(),c=a.ownerDocument,d=c.body,e=c.documentElement,f=e.clientTop||d.clientTop||0,g=e.clientLeft||d.clientLeft||0,h=b.top+(L.win.pageYOffset||e.scrollTop||d.scrollTop)-f,i=b.left+(L.win.pageXOffset||e.scrollLeft||d.scrollLeft)-g;return{y:h,x:i}};S.getElementByPoint=function(a,b){var c=this,d=c.canvas,e=L.doc.elementFromPoint(a,b);if(L.win.opera&&"svg"==e.tagName){var f=xc(d),g=d.createSVGRect();g.x=a-f.x,g.y=b-f.y,g.width=g.height=1;var h=d.getIntersectionList(g,null);h.length&&(e=h[h.length-1])}if(!e)return null;for(;e.parentNode&&e!=d.parentNode&&!e.raphael;)e=e.parentNode;return e==c.canvas.parentNode&&(e=d),e=e&&e.raphael?c.getById(e.raphaelid):null},S.getElementsByBBox=function(a){var b=this.set();return this.forEach(function(d){c.isBBoxIntersect(d.getBBox(),a)&&b.push(d)}),b},S.getById=function(a){for(var b=this.bottom;b;){if(b.id==a)return b;b=b.next}return null},S.forEach=function(a,b){for(var c=this.bottom;c;){if(a.call(b,c)===!1)return this;c=c.next}return this},S.getElementsByPoint=function(a,b){var c=this.set();return this.forEach(function(d){d.isPointInside(a,b)&&c.push(d)}),c},rc.isPointInside=function(a,b){var d,e=this.realPath=this.realPath||Fb[this.type](this);return c.isPointInsidePath((d=this.attr("transform"))&&d.length&&c.transformPath(e,d)||e,a,b)},rc.getBBox=function(a){if(this.removed)return{};var b=this._;return a?((b.dirty||!b.bboxwt)&&(this.realPath=Fb[this.type](this),b.bboxwt=Vb(this.realPath),b.bboxwt.toString=n,b.dirty=0),b.bboxwt):((b.dirty||b.dirtyT||!b.bbox)&&((b.dirty||!this.realPath)&&(b.bboxwt=0,this.realPath=Fb[this.type](this)),b.bbox=Vb(Gb(this.realPath,this.matrix)),b.bbox.toString=n,b.dirty=b.dirtyT=0),b.bbox)},rc.clone=function(){if(this.removed)return null;var a=this,b=a.paper[a.type]().attr(a.attr());return a.__set__&&a.__set__.push(b),b},rc.glow=function(a){if("text"==this.type)return null;a=a||{};var b={width:(a.width||10)+(+this.attr("stroke-width")||1),fill:a.fill||!1,opacity:a.opacity||.5,offsetx:a.offsetx||0,offsety:a.offsety||0,color:a.color||"#000"},c=b.width/2,d=this.paper,e=d.set(),f=this.realPath||Fb[this.type](this);f=this.matrix?Gb(f,this.matrix):f;for(var g=1;c+1>g;g++)e.push(d.path(f).attr({stroke:b.color,fill:b.fill?b.color:"none","stroke-linejoin":"round","stroke-linecap":"round","stroke-width":+(b.width/c*g).toFixed(3),opacity:+(b.opacity/c).toFixed(3)}));return e.insertBefore(this).translate(b.offsetx,b.offsety)};var yc=function(a,b,d,e,f,g,j,k,l){return null==l?h(a,b,d,e,f,g,j,k):c.findDotsAtSegment(a,b,d,e,f,g,j,k,i(a,b,d,e,f,g,j,k,l))},zc=function(a,b){return function(d,e,f){d=cc(d);for(var g,h,i,j,k,l="",m={},n=0,o=0,p=d.length;p>o;o++){if(i=d[o],"M"==i[0])g=+i[1],h=+i[2];else{if(j=yc(g,h,i[1],i[2],i[3],i[4],i[5],i[6]),n+j>e){if(b&&!m.start){if(k=yc(g,h,i[1],i[2],i[3],i[4],i[5],i[6],e-n),l+=["C"+k.start.x,k.start.y,k.m.x,k.m.y,k.x,k.y],f)return l;m.start=l,l=["M"+k.x,k.y+"C"+k.n.x,k.n.y,k.end.x,k.end.y,i[5],i[6]].join(),n+=j,g=+i[5],h=+i[6];continue}if(!a&&!b)return k=yc(g,h,i[1],i[2],i[3],i[4],i[5],i[6],e-n),{x:k.x,y:k.y,alpha:k.alpha}}n+=j,g=+i[5],h=+i[6]}l+=i.shift()+i}return m.end=l,k=a?n:b?m:c.findDotsAtSegment(g,h,i[0],i[1],i[2],i[3],i[4],i[5],1),k.alpha&&(k={x:k.x,y:k.y,alpha:k.alpha}),k}},Ac=zc(1),Bc=zc(),Cc=zc(0,1);c.getTotalLength=Ac,c.getPointAtLength=Bc,c.getSubpath=function(a,b,c){if(this.getTotalLength(a)-c<1e-6)return Cc(a,b).end;var d=Cc(a,c,1);return b?Cc(d,b).end:d},rc.getTotalLength=function(){return"path"==this.type?this.node.getTotalLength?this.node.getTotalLength():Ac(this.attrs.path):void 0},rc.getPointAtLength=function(a){return"path"==this.type?Bc(this.attrs.path,a):void 0},rc.getSubpath=function(a,b){return"path"==this.type?c.getSubpath(this.attrs.path,a,b):void 0};var Dc=c.easing_formulas={linear:function(a){return a},"<":function(a){return bb(a,1.7)},">":function(a){return bb(a,.48)},"<>":function(a){var b=.48-a/1.04,c=eb(.1734+b*b),d=c-b,e=bb(ab(d),1/3)*(0>d?-1:1),f=-c-b,g=bb(ab(f),1/3)*(0>f?-1:1),h=e+g+.5;return 3*(1-h)*h*h+h*h*h},backIn:function(a){var b=1.70158;return a*a*((b+1)*a-b)},backOut:function(a){a-=1;var b=1.70158;return a*a*((b+1)*a+b)+1},elastic:function(a){return a==!!a?a:bb(2,-10*a)*db((a-.075)*2*gb/.3)+1},bounce:function(a){var b,c=7.5625,d=2.75;return 1/d>a?b=c*a*a:2/d>a?(a-=1.5/d,b=c*a*a+.75):2.5/d>a?(a-=2.25/d,b=c*a*a+.9375):(a-=2.625/d,b=c*a*a+.984375),b}};Dc.easeIn=Dc["ease-in"]=Dc["<"],Dc.easeOut=Dc["ease-out"]=Dc[">"],Dc.easeInOut=Dc["ease-in-out"]=Dc["<>"],Dc["back-in"]=Dc.backIn,Dc["back-out"]=Dc.backOut;var Ec=[],Fc=a.requestAnimationFrame||a.webkitRequestAnimationFrame||a.mozRequestAnimationFrame||a.oRequestAnimationFrame||a.msRequestAnimationFrame||function(a){setTimeout(a,16)},Gc=function(){for(var a=+new Date,d=0;dh))if(i>h){var r=j(h/i);for(var s in k)if(k[w](s)){switch(yb[s]){case z:f=+k[s]+r*i*l[s];break;case"colour":f="rgb("+[Hc(fb(k[s].r+r*i*l[s].r)),Hc(fb(k[s].g+r*i*l[s].g)),Hc(fb(k[s].b+r*i*l[s].b))].join(",")+")";break;case"path":f=[];for(var t=0,u=k[s].length;u>t;t++){f[t]=[k[s][t][0]];for(var x=1,A=k[s][t].length;A>x;x++)f[t][x]=(+k[s][t][x]+r*i*l[s][t][x]).toFixed(4);f[t]=f[t].join(v)}f=f.join(v);break;case"transform":if(l[s].real)for(f=[],t=0,u=k[s].length;u>t;t++)for(f[t]=[k[s][t][0]],x=1,A=k[s][t].length;A>x;x++)f[t][x]=k[s][t][x]+r*i*l[s][t][x];else{var B=function(a){return+k[s][a]+r*i*l[s][a]};f=[["m",B(0),B(1),B(2),B(3),B(4),B(5)]]}break;case"csv":if("clip-rect"==s)for(f=[],t=4;t--;)f[t]=+k[s][t]+r*i*l[s][t];break;default:var C=[][y](k[s]);for(f=[],t=n.ca[s].length;t--;)f[t]=+C[t]+r*i*l[s][t]}o[s]=f}n.attr(o),function(a,c,d){setTimeout(function(){b("raphael.anim.frame."+a,c,d)})}(n.id,n,e.anim)}else{if(function(a,d,e){setTimeout(function(){b("raphael.anim.frame."+d.id,d,e),b("raphael.anim.finish."+d.id,d,e),c.is(a,"function")&&a.call(d)})}(e.callback,n,e.anim),n.attr(m),Ec.splice(d--,1),e.repeat>1&&!e.next){for(g in m)m[w](g)&&(p[g]=e.totalOrigin[g]);e.el.attr(p),q(e.anim,e.el,e.anim.percents[0],null,e.totalOrigin,e.repeat-1)}e.next&&!e.stop&&q(e.anim,e.el,e.next,null,e.totalOrigin,e.repeat)}}}c.svg&&n&&n.paper&&n.paper.safari(),Ec.length&&Fc(Gc)},Hc=function(a){return a>255?255:0>a?0:a};rc.animateWith=function(a,b,d,e,f,g){var h=this;if(h.removed)return g&&g.call(h),h;var i=d instanceof p?d:c.animation(d,e,f,g);q(i,h,i.percents[0],null,h.attr());for(var j=0,k=Ec.length;k>j;j++)if(Ec[j].anim==b&&Ec[j].el==a){Ec[k-1].start=Ec[j].start;break}return h},rc.onAnimation=function(a){return a?b.on("raphael.anim.frame."+this.id,a):b.unbind("raphael.anim.frame."+this.id),this},p.prototype.delay=function(a){var b=new p(this.anim,this.ms);return b.times=this.times,b.del=+a||0,b},p.prototype.repeat=function(a){var b=new p(this.anim,this.ms);return b.del=this.del,b.times=Z.floor($(a,0))||1,b},c.animation=function(a,b,d,e){if(a instanceof p)return a;(c.is(d,"function")||!d)&&(e=e||d||null,d=null),a=Object(a),b=+b||0;var f,g,h={};for(g in a)a[w](g)&&X(g)!=g&&X(g)+"%"!=g&&(f=!0,h[g]=a[g]);return f?(d&&(h.easing=d),e&&(h.callback=e),new p({100:h},b)):new p(a,b)},rc.animate=function(a,b,d,e){var f=this;if(f.removed)return e&&e.call(f),f;var g=a instanceof p?a:c.animation(a,b,d,e);return q(g,f,g.percents[0],null,f.attr()),f},rc.setTime=function(a,b){return a&&null!=b&&this.status(a,_(b,a.ms)/a.ms),this},rc.status=function(a,b){var c,d,e=[],f=0;if(null!=b)return q(a,this,-1,_(b,1)),this;for(c=Ec.length;c>f;f++)if(d=Ec[f],d.el.id==this.id&&(!a||d.anim==a)){if(a)return d.status;e.push({anim:d.anim,status:d.status})}return a?0:e},rc.pause=function(a){for(var c=0;cb;b++)(a=h[b]).stalk&&a.el[a.stalk](e);return e},rc.toBack=function(){if(this.removed)return this;var a,b,d,e=this,f=c._engine.getNode(e),g=e.parent,h=e.followers;for(c._toback(e,g)&&g.canvas.insertBefore(f,g.canvas.firstChild),b=0,d=h.length;d>b;b++)(a=h[b]).stalk&&a.el[a.stalk](e);return e},rc.insertAfter=function(a){if(this.removed)return this;var b,d,e,f=this,g=c._engine.getNode(f),h=c._engine.getLastNode(a),i=a.parent.canvas,j=f.followers;for(h.nextSibling?i.insertBefore(g,h.nextSibling):i.appendChild(g),c._insertafter(f,a,f.parent,a.parent),d=0,e=j.length;e>d;d++)(b=j[d]).stalk&&b.el[b.stalk](a);return f},rc.insertBefore=function(a){if(this.removed)return this;var b,d,e,f=this,g=c._engine.getNode(f),h=c._engine.getNode(a),i=f.followers;for(a.parent.canvas.insertBefore(g,h),c._insertbefore(f,a,f.parent,a.parent),f.parent=a.parent,d=0,e=i.length;e>d;d++)(b=i[d]).stalk&&b.el[b.stalk](a);return this},rc.appendChild=function(a){if(this.removed||"group"!==this.type)return this;var b,d,e,f,g=this,h=g.followers;if(a.parent===g)return a.toFront(),g;for(d=c._engine.getNode(a),c._tear(a,a.parent),g.canvas.appendChild(d),a.parent=g,!g.bottom&&(g.bottom=a),a.prev=g.top,a.next=null,g.top&&(g.top.next=a),g.top=a,e=0,f=h.length;f>e;e++)(b=h[e]).stalk&&b.el[b.stalk](a);return g},rc.removeChild=function(a){if(this.removed||"group"!==this.type||a.parent!==this)return this;var b=this,d=c._engine.getNode(a),e=b.paper;return c._tear(a,b),e.canvas.appendChild(d),b.parent=e,!e.bottom&&(e.bottom=b),b.prev=e.top,e.top&&(e.top.next=b),e.top=b,b.next=null,b};var Ic=function(a){if(this.items=[],this.length=0,this.type="set",a)for(var b=0,c=a.length;c>b;b++)!a[b]||a[b].constructor!=rc.constructor&&a[b].constructor!=Ic||(this[this.items.length]=this.items[this.items.length]=a[b],this.length++)},Jc=Ic.prototype;Jc.push=function(){for(var a,b,c=0,d=arguments.length;d>c;c++)a=arguments[c],!a||a.constructor!=rc.constructor&&a.constructor!=Ic||(b=this.items.length,this[b]=this.items[b]=a,this.length++);return this},Jc.pop=function(){return this.length&&delete this[this.length--],this.items.pop()},Jc.forEach=function(a,b){for(var c=0,d=this.items.length;d>c;c++)if(a.call(b,this.items[c],c)===!1)return this;return this};for(var Kc in rc)rc[w](Kc)&&(Jc[Kc]=function(a){return function(){var b=arguments;return this.forEach(function(c){c[a][x](c,b)})}}(Kc));Jc.attr=function(a,b){if(a&&c.is(a,B)&&c.is(a[0],C))for(var d=0,e=a.length;e>d;d++)this.items[d].attr(a[d]);else for(var f=0,g=this.items.length;g>f;f++)this.items[f].attr(a,b);return this},Jc.clear=function(){for(;this.length;)this.pop()},Jc.splice=function(a,b){a=0>a?$(this.length+a,0):a,b=$(0,_(this.length-a,isNaN(b)&&this.length||b));var c,d=[],e=[],f=[];for(c=2;cc;c++)e.push(this[a+c]);for(;cc?f[c]:d[c-g];for(c=this.items.length=this.length-=b-g;this[c];)delete this[c++];return new Ic(e)},Jc.exclude=function(a){for(var b=0,c=this.length;c>b;b++)if(this[b]==a)return this.splice(b,1),!0},Jc.animate=function(a,b,d,e){(c.is(d,"function")||!d)&&(e=d||null);var f,g,h=this.items.length,i=h,j=this;if(!h)return this;e&&(g=function(){!--h&&e.call(j)}),d=c.is(d,A)?d:g;var k=c.animation(a,b,d,g);for(f=this.items[--i].animate(k);i--;)this.items[i]&&!this.items[i].removed&&this.items[i].animateWith(f,k,k);return this},Jc.insertAfter=function(a){for(var b=this.items.length;b--;)this.items[b].insertAfter(a);return this},Jc.getBBox=function(){for(var a=[],b=[],c=[],d=[],e=this.items.length;e--;)if(!this.items[e].removed){var f=this.items[e].getBBox();a.push(f.x),b.push(f.y),c.push(f.x+f.width),d.push(f.y+f.height)}return a=_[x](0,a),b=_[x](0,b),c=$[x](0,c),d=$[x](0,d),{x:a,y:b,x2:c,y2:d,width:c-a,height:d-b}},Jc.clone=function(a){a=new Ic;for(var b=0,c=this.items.length;c>b;b++)a.push(this.items[b].clone());return a},Jc.toString=function(){return"Raphaël‘s set"},Jc.glow=function(a){var b=this.paper.set();return this.forEach(function(c){var d=c.glow(a);null!=d&&d.forEach(function(a){b.push(a)})}),b},c.registerFont=function(a){if(!a.face)return a;this.fonts=this.fonts||{};var b={w:a.w,face:{},glyphs:{}},c=a.face["font-family"];for(var d in a.face)a.face[w](d)&&(b.face[d]=a.face[d]);if(this.fonts[c]?this.fonts[c].push(b):this.fonts[c]=[b],!a.svg){b.face["units-per-em"]=Y(a.face["units-per-em"],10);for(var e in a.glyphs)if(a.glyphs[w](e)){var f=a.glyphs[e];if(b.glyphs[e]={w:f.w,k:{},d:f.d&&"M"+f.d.replace(/[mlcxtrv]/g,function(a){return{l:"L",c:"C",x:"z",t:"m",r:"l",v:"c"}[a]||"M"})+"z"},f.k)for(var g in f.k)f[w](g)&&(b.glyphs[e].k[g]=f.k[g])}}return a},S.getFont=function(a,b,d,e){if(e=e||"normal",d=d||"normal",b=+b||{normal:400,bold:700,lighter:300,bolder:800}[b]||400,c.fonts){var f=c.fonts[a];if(!f){var g=new RegExp("(^|\\s)"+a.replace(/[^\w\d\s+!~.:_-]/g,u)+"(\\s|$)","i");for(var h in c.fonts)if(c.fonts[w](h)&&g.test(h)){f=c.fonts[h];break}}var i;if(f)for(var j=0,k=f.length;k>j&&(i=f[j],i.face["font-weight"]!=b||i.face["font-style"]!=d&&i.face["font-style"]||i.face["font-stretch"]!=e);j++);return i}},S.print=function(a,b,d,e,f,g,h){g=g||"middle",h=$(_(h||0,1),-1);var i,j=W(d)[E](u),k=0,l=0,m=u;if(c.is(e,d)&&(e=this.getFont(e)),e){i=(f||16)/e.face["units-per-em"];for(var n=e.face.bbox[E](mb),o=+n[0],p=n[3]-n[1],q=0,r=+n[1]+("baseline"==g?p+ +e.face.descent:p/2),s=0,t=j.length;t>s;s++){if("\n"==j[s])k=0,w=0,l=0,q+=p;else{var v=l&&e.glyphs[j[s-1]]||{},w=e.glyphs[j[s]];k+=l?(v.w||e.w)+(v.k&&v.k[j[s]]||0)+e.w*h:0,l=1}w&&w.d&&(m+=c.transformPath(w.d,["t",k*i,q*i,"s",i,i,o,r,"t",(a-o)/i,(b-r)/i]))}}return this.path(m).attr({fill:"#000",stroke:"none"})},S.add=function(a){if(c.is(a,"array"))for(var b,d=this.set(),e=0,f=a.length;f>e;e++)b=a[e]||{},T[w](b.type)&&d.push(this[b.type]().attr(b));return d},c.format=function(a,b){var d=c.is(b,B)?[0][y](b):arguments;return a&&c.is(a,A)&&d.length-1&&(a=a.replace(nb,function(a,b){return null==d[++b]?u:d[b]})),a||u},c.fullfill=function(){var a=/\{([^\}]+)\}/g,b=/(?:(?:^|\.)(.+?)(?=\[|\.|$|\()|\[('|")(.+?)\2\])(\(\))?/g,c=function(a,c,d){var e=d;return c.replace(b,function(a,b,c,d,f){b=b||d,e&&(b in e&&(e=e[b]),"function"==typeof e&&f&&(e=e()))}),e=(null==e||e==d?a:e)+""};return function(b,d){return String(b).replace(a,function(a,b){return c(a,b,d)})}}(),c.ninja=function(){return M.was?L.win.Raphael=M.is:delete Raphael,c};var Lc=c.vml&&.5||0;return c.crispBound=Tb(function(a,b,c,d,e){var f,g={};return a=a||0,b=b||0,c=c||0,d=d||0,e=e||0,f=e%2/2+Lc,g.x=fb(a+f)-f,g.y=fb(b+f)-f,g.width=fb(a+c+f)-f-g.x,g.height=fb(b+d+f)-f-g.y,g["stroke-width"]=e,0===g.width&&0!==c&&(g.width=1),0===g.height&&0!==d&&(g.height=1),g},c),rc.crisp=function(){var a,b=this,d=b.attrs,e=b.attr(["x","y","width","height","stroke-width"]);e=c.crispBound(e.x,e.y,e.width,e.height,e["stroke-width"]);for(a in e)d[a]===e[a]&&delete e[a];return b.attr(e)},c.st=Jc,c.define=function(a,b,d,e,f){var g,h;if(c.is(a,B))for(g=0,h=a.length;h>g;g++)c.define(a[g]);else{if(c.is(a,C))return c.define(a.name,a[a.name],a.ca,a.fn,a.e),void 0;if(a&&!c.fn[a])return c.fn[a]=function(){var g,h=arguments,i=b.apply(this,h);if(e&&c.is(e,C))for(g in e)i[g]=e[g];if(f&&c.is(f,C))for(g in f)i[g]&&i[g](f[g]);if(d){if(c.is(d,"function"))i.ca[a]=d;else for(g in d)i.ca[g]=d[g];i.ca[a]&&(c._lastArgIfGroup(h,!0),i.attr(a,J.call(h)))}return i},d&&(c.fn[a].ca=d),e&&(c.fn[a].fn=e),f&&(c.fn[a].e=f),c.fn[a]}},function(a,b,d){function e(){/in/.test(a.readyState)?setTimeout(e,9):c.eve("raphael.DOMload")}null==a.readyState&&a.addEventListener&&(a.addEventListener(b,d=function(){a.removeEventListener(b,d,!1),a.readyState="complete"},!1),a.readyState="loading"),e()}(document,"DOMContentLoaded"),b.on("raphael.DOMload",function(){s=!0}),function(){if(c.svg){var b="hasOwnProperty",d=String,e=parseFloat,f=parseInt,g=Math,h=g.max,i=g.abs,j=g.pow,k=g.sqrt,l=/[, ]+/,m=!(!/AppleWebKit/.test(c._g.win.navigator.userAgent)||/Chrome/.test(c._g.win.navigator.userAgent)&&!(c._g.win.navigator.appVersion.match(/Chrome\/(\d+)\./)[1]<29)),n=c.eve,o="",p=" ",q="http://www.w3.org/1999/xlink",r={block:"M5,0 0,2.5 5,5z",classic:"M5,0 0,2.5 5,5 3.5,3 3.5,2z",diamond:"M2.5,0 5,2.5 2.5,5 0,2.5z",open:"M6,1 1,3.5 6,6",oval:"M2.5,0A2.5,2.5,0,0,1,2.5,5 2.5,2.5,0,0,1,2.5,0z"},s={},t=function(){return c._url=c._g.win.location.href.replace(/#.*?$/,o)};c.toString=function(){return"Your browser supports SVG.\nYou are running Raphaël "+this.version},c._url=/msie/i.test(navigator.userAgent)&&!a.opera?o:t(),c._url&&c._g.win.history.pushState&&(c._g.win.history.pushState=function(){var a=c._g.win.history.pushState;return function(){var b=a.apply(c._g.win.history,arguments);return t(),b}}(),c._g.win.addEventListener("popstate",t,!1));var u=c._createNode=function(a,e){if(e){"string"==typeof a&&(a=u(a));for(var f in e)e[b](f)&&("xlink:"==f.substring(0,6)?a.setAttributeNS(q,f.substring(6),d(e[f])):a.setAttribute(f,d(e[f])))}else a=c._g.doc.createElementNS("http://www.w3.org/2000/svg",a);return a},v={userSpaceOnUse:"userSpaceOnUse",objectBoundingBox:"objectBoundingBox"},w={pad:"pad",redlect:"reflect",repeat:"repeat"},x=function(a,b){var f,l,m,n,p,q="linear",r=a.id+b,s=.5,t=.5,x=a.node,y=a.paper,z=x.style,A=c._g.doc.getElementById(r);if(!A&&y.defs){if(b=d(b).replace(c._radial_gradient,function(a,b){q="radial",b=b&&b.split(",")||[],n=b[5],p=b[6];var c,d,g=b[0],h=b[1],i=b[2],r=b[3],u=b[4],w=g&&h;return i&&(f=/\%/.test(i)?i:e(i)),n===v.userSpaceOnUse?(w&&(s=g,t=h),r&&u&&(l=r,m=u,w||(s=l,t=m)),o):(w&&(s=e(g),t=e(h),c=2*(t>.5)-1,(d=j(s-.5,2))+j(t-.5,2)>.25&&.25>d&&(t=k(.25-d)*c+.5)&&.5!==t&&(t=t.toFixed(5)-1e-5*c)),r&&u&&(l=e(r),m=e(u),c=2*(m>.5)-1,(d=j(l-.5,2))+j(m-.5,2)>.25&&.25>d&&(m=k(.25-d)*c+.5)&&.5!==m&&(m=m.toFixed(5)-1e-5*c),w||(s=l,t=m)),o)}),b=b.split(/\s*\-\s*/),"linear"==q){var B,C,D=b.shift(),E=D.match(/\((.*)\)/);if(E=E&&E[1]&&E[1].split(/\s*\,\s*/),D=-e(D),isNaN(D))return null;E&&E.length?(E[0]in v?(n=E.shift(),E[0]in w&&(p=E.shift())):(E[4]&&(n=E[4]),E[5]&&(p=E[5])),B=[E[0]||"0%",E[1]||"0%",E[2]||"100%",E[3]||"0%"]):(B=[0,0,g.cos(c.rad(D)),g.sin(c.rad(D))],C=1/(h(i(B[2]),i(B[3]))||1),B[2]*=C,B[3]*=C,B[2]<0&&(B[0]=-B[2],B[2]=0),B[3]<0&&(B[1]=-B[3],B[3]=0))}var F=c._parseDots(b);if(!F)return null;if(r=r.replace(/[\(\)\s,\xb0#]/g,"_"),a.gradient&&r!==a.gradient.id&&(y.defs.removeChild(a.gradient),delete a.gradient),!a.gradient){A=u(q+"Gradient",{id:r}),a.gradient=A,n in v&&A.setAttribute("gradientUnits",d(n)),p in w&&A.setAttribute("spreadMethod",d(p)),"radial"===q?(void 0!==f&&A.setAttribute("r",d(f)),void 0!==l&&void 0!==m&&(A.setAttribute("cx",d(l)),A.setAttribute("cy",d(m))),A.setAttribute("fx",d(s)),A.setAttribute("fy",d(t))):u(A,{x1:B[0],y1:B[1],x2:B[2],y2:B[3],gradientTransform:a.matrix.invert()}),y.defs.appendChild(A);for(var G=0,H=F.length;H>G;G++)A.appendChild(u("stop",{offset:F[G].offset?F[G].offset:G?"100%":"0%","stop-color":F[G].color||"#fff","stop-opacity":void 0===F[G].opacity?1:F[G].opacity}))}}return u(x,{fill:"url('"+c._url+"#"+r+"')",opacity:1,"fill-opacity":1}),z.fill=o,z.opacity=1,z.fillOpacity=1,1},y=function(a){var b=a.getBBox(1);u(a.pattern,{patternTransform:a.matrix.invert()+" translate("+b.x+","+b.y+")"})},z=function(a,e,f){if("path"==a.type){for(var g,h,i,j,k,l=d(e).toLowerCase().split("-"),m=a.paper,n=f?"end":"start",p=a.node,q=a.attrs,t=q["stroke-width"],v=l.length,w="classic",x=3,y=3,z=5;v--;)switch(l[v]){case"block":case"classic":case"oval":case"diamond":case"open":case"none":w=l[v];
+break;case"wide":y=5;break;case"narrow":y=2;break;case"long":x=5;break;case"short":x=2}if("open"==w?(x+=2,y+=2,z+=2,i=1,j=f?4:1,k={fill:"none",stroke:q.stroke}):(j=i=x/2,k={fill:q.stroke,stroke:"none"}),a._.arrows?f?(a._.arrows.endPath&&s[a._.arrows.endPath]--,a._.arrows.endMarker&&s[a._.arrows.endMarker]--):(a._.arrows.startPath&&s[a._.arrows.startPath]--,a._.arrows.startMarker&&s[a._.arrows.startMarker]--):a._.arrows={},"none"!=w){var A="raphael-marker-"+w,B="raphael-marker-"+n+w+x+y+"-obj"+a.id;c._g.doc.getElementById(A)?s[A]++:(m.defs.appendChild(u(u("path"),{"stroke-linecap":"round",d:r[w],id:A})),s[A]=1);var C,D=c._g.doc.getElementById(B);D?(s[B]++,C=D.getElementsByTagName("use")[0]):(D=u(u("marker"),{id:B,markerHeight:y,markerWidth:x,orient:"auto",refX:j,refY:y/2}),C=u(u("use"),{"xlink:href":"#"+A,transform:(f?"rotate(180 "+x/2+" "+y/2+") ":o)+"scale("+x/z+","+y/z+")","stroke-width":(1/((x/z+y/z)/2)).toFixed(4)}),D.appendChild(C),m.defs.appendChild(D),s[B]=1),u(C,k);var E=i*("diamond"!=w&&"oval"!=w);f?(g=a._.arrows.startdx*t||0,h=c.getTotalLength(q.path)-E*t):(g=E*t,h=c.getTotalLength(q.path)-(a._.arrows.enddx*t||0)),k={},k["marker-"+n]="url('"+c._url+"#"+B+"')",(h||g)&&(k.d=Raphael.getSubpath(q.path,g,h)),u(p,k),a._.arrows[n+"Path"]=A,a._.arrows[n+"Marker"]=B,a._.arrows[n+"dx"]=E,a._.arrows[n+"Type"]=w,a._.arrows[n+"String"]=e}else f?(g=a._.arrows.startdx*t||0,h=c.getTotalLength(q.path)-g):(g=0,h=c.getTotalLength(q.path)-(a._.arrows.enddx*t||0)),a._.arrows[n+"Path"]&&u(p,{d:Raphael.getSubpath(q.path,g,h)}),delete a._.arrows[n+"Path"],delete a._.arrows[n+"Marker"],delete a._.arrows[n+"dx"],delete a._.arrows[n+"Type"],delete a._.arrows[n+"String"];for(k in s)if(s[b](k)&&!s[k]){var F=c._g.doc.getElementById(k);F&&F.parentNode.removeChild(F)}}},A={"":[0],none:[0],"-":[3,1],".":[1,1],"-.":[3,1,1,1],"-..":[3,1,1,1,1,1],". ":[1,3],"- ":[4,3],"--":[8,3],"- .":[4,3,1,3],"--.":[8,3,1,3],"--..":[8,3,1,3,1,3]},B=function(a,b,e){var f=A[d(b).toLowerCase()];if(b=f||void 0!==b&&[].concat(b)){var g,h=a.attrs["stroke-width"]||"1",i={round:h,square:h,butt:0}[a.attrs["stroke-linecap"]||e["stroke-linecap"]]||0,j=g=b.length;if(f)for(;g--;)b[g]=b[g]*h+(g%2?1:-1)*i;else for(g=0;j>g;g+=2)b[g]-=i,b[g+1]&&(b[g+1]+=i),b[g]<=0&&(b[g]=.1);c.is(b,"array")&&u(a.node,{"stroke-dasharray":b.join(",")})}},C=c._setFillAndStroke=function(a,e){if(a.paper.canvas){var g=a.node,j=a.attrs,k=a.paper,n=g.style,p=n.visibility;n.visibility="hidden";for(var r in e)if(e[b](r)){if(!c._availableAttrs[b](r))continue;var s=e[r];switch(j[r]=s,r){case"blur":a.blur(s);break;case"href":case"title":case"target":var t=g.parentNode;if("a"!=t.tagName.toLowerCase()){if(s==o)break;var v=u("a");t.insertBefore(v,g),v.appendChild(g),t=v}"target"==r?t.setAttributeNS(q,"show","blank"==s?"new":s):t.setAttributeNS(q,r,s),g.titleNode=t;break;case"cursor":n.cursor=s;break;case"transform":a.transform(s);break;case"rotation":c.is(s,"array")?a.rotate.apply(a,s):a.rotate(s);break;case"arrow-start":z(a,s);break;case"arrow-end":z(a,s,1);break;case"clip-path":var w=!0;case"clip-rect":var A=!w&&d(s).split(l);if(a._.clipispath=!!w,w||4==A.length){a.clip&&a.clip.parentNode.parentNode.removeChild(a.clip.parentNode);var C=u("clipPath"),D=u(w?"path":"rect");C.id=c.createUUID(),u(D,w?{d:s?j["clip-path"]=c._pathToAbsolute(s):c._availableAttrs.path,fill:"none"}:{x:A[0],y:A[1],width:A[2],height:A[3],transform:a.matrix.invert()}),C.appendChild(D),k.defs.appendChild(C),u(g,{"clip-path":"url('"+c._url+"#"+C.id+"')"}),a.clip=D}if(!s){var F=g.getAttribute("clip-path");if(F){var G=c._g.doc.getElementById(F.replace(/(^url\(#|\)$)/g,o));G&&G.parentNode.removeChild(G),u(g,{"clip-path":o}),delete a.clip}}break;case"path":"path"==a.type&&(u(g,{d:s?j.path=c._pathToAbsolute(s):c._availableAttrs.path}),a._.dirty=1,a._.arrows&&("startString"in a._.arrows&&z(a,a._.arrows.startString),"endString"in a._.arrows&&z(a,a._.arrows.endString,1)));break;case"width":if(g.setAttribute(r,s),a._.dirty=1,!j.fx)break;r="x",s=j.x;case"x":j.fx&&(s=-j.x-(j.width||0));case"rx":if("rx"==r&&"rect"==a.type)break;case"cx":g.setAttribute(r,s),a.pattern&&y(a),a._.dirty=1;break;case"height":if(g.setAttribute(r,s),a._.dirty=1,!j.fy)break;r="y",s=j.y;case"y":j.fy&&(s=-j.y-(j.height||0));case"ry":if("ry"==r&&"rect"==a.type)break;case"cy":g.setAttribute(r,s),a.pattern&&y(a),a._.dirty=1;break;case"r":"rect"==a.type?u(g,{rx:s,ry:s}):g.setAttribute(r,s),a._.dirty=1;break;case"src":"image"==a.type&&g.setAttributeNS(q,"href",s);break;case"stroke-width":(1!=a._.sx||1!=a._.sy)&&(s/=h(i(a._.sx),i(a._.sy))||1),k._vbSize&&(s*=k._vbSize),m&&0===s&&(s=1e-6),g.setAttribute(r,s),j["stroke-dasharray"]&&B(a,j["stroke-dasharray"],e),a._.arrows&&("startString"in a._.arrows&&z(a,a._.arrows.startString),"endString"in a._.arrows&&z(a,a._.arrows.endString,1));break;case"stroke-dasharray":B(a,s,e);break;case"fill":var H=d(s).match(c._ISURL);if(H){C=u("pattern");var I=u("image");C.id=c.createUUID(),u(C,{x:0,y:0,patternUnits:"userSpaceOnUse",height:1,width:1}),u(I,{x:0,y:0,"xlink:href":H[1]}),C.appendChild(I),function(a){c._preload(H[1],function(){var b=this.offsetWidth,c=this.offsetHeight;u(a,{width:b,height:c}),u(I,{width:b,height:c}),k.safari()})}(C),k.defs.appendChild(C),u(g,{fill:"url('"+c._url+"#"+C.id+"')"}),a.pattern=C,a.pattern&&y(a);break}var J=c.getRGB(s);if(J.error){if(("circle"==a.type||"ellipse"==a.type||"r"!=d(s).charAt())&&x(a,s)){if("opacity"in j||"fill-opacity"in j){var K=c._g.doc.getElementById(g.getAttribute("fill").replace(/^url\(#|\)$/g,o));if(K){var L=K.getElementsByTagName("stop");u(L[L.length-1],{"stop-opacity":("opacity"in j?j.opacity:1)*("fill-opacity"in j?j["fill-opacity"]:1)})}}j.gradient=s,j.fill="none";break}}else delete e.gradient,delete j.gradient,!c.is(j.opacity,"undefined")&&c.is(e.opacity,"undefined")&&u(g,{opacity:j.opacity}),!c.is(j["fill-opacity"],"undefined")&&c.is(e["fill-opacity"],"undefined")&&u(g,{"fill-opacity":j["fill-opacity"]});J[b]("opacity")?(u(g,{"fill-opacity":n.fillOpacity=J.opacity>1?J.opacity/100:J.opacity}),a._.opacitydirty=!0):a._.opacitydirty&&c.is(j["fill-opacity"],"undefined")&&c.is(e["fill-opacity"],"undefined")&&(g.removeAttribute("fill-opacity"),n.fillOpacity=o,delete a._.opacitydirty);case"stroke":J=c.getRGB(s),g.setAttribute(r,J.hex),"stroke"==r&&J[b]("opacity")&&u(g,{"stroke-opacity":J.opacity>1?J.opacity/100:J.opacity}),"stroke"==r&&a._.arrows&&("startString"in a._.arrows&&z(a,a._.arrows.startString),"endString"in a._.arrows&&z(a,a._.arrows.endString,1));break;case"gradient":("circle"==a.type||"ellipse"==a.type||"r"!=d(s).charAt())&&x(a,s);break;case"line-height":case"vertical-align":break;case"visibility":"hidden"===s?a.hide():a.show();break;case"opacity":j.gradient&&!j[b]("stroke-opacity")&&u(g,{"stroke-opacity":s>1?s/100:s});case"fill-opacity":if(j.gradient){K=c._g.doc.getElementById(g.getAttribute("fill").replace(/^url\(#|\)$/g,o)),K&&(L=K.getElementsByTagName("stop"),u(L[L.length-1],{"stop-opacity":s}));break}default:"font-size"==r&&(s=f(s,10)+"px");var M=r.replace(/(\-.)/g,function(a){return a.substring(1).toUpperCase()});n[M]=s,a._.dirty=1,g.setAttribute(r,s)}}E(a,e),n.visibility=p}},D=1.2,E=function(a,f){if("text"==a.type&&(f[b]("text")||f[b]("font")||f[b]("font-size")||f[b]("x")||f[b]("y")||f[b]("line-height")||f[b]("vertical-align"))){var g=a.attrs,h=a.node,i=h.firstChild&&c._g.doc.defaultView.getComputedStyle(h.firstChild,o),j=i?e(c._g.doc.defaultView.getComputedStyle(h.firstChild,o).getPropertyValue("font-size")):10,k=e(f["line-height"]||g["line-height"])||j*D,l=g[b]("vertical-align")?g["vertical-align"]:"middle";if(isNaN(k)&&(k=j*D),l="top"===l?-.5:"bottom"===l?.5:0,f[b]("text")&&(f.text!==g.text||a._textdirty)){for(g.text=f.text;h.firstChild;)h.removeChild(h.firstChild);for(var m,n=d(f.text).split(/\n| /gi),p=[],q=0,r=n.length;r>q;q++)m=u("tspan"),q?u(m,{dy:k,x:g.x}):u(m,{dy:k*n.length*l,x:g.x}),n[q]||(m.setAttributeNS("http://www.w3.org/XML/1998/namespace","xml:space","preserve"),n[q]=" "),m.appendChild(c._g.doc.createTextNode(n[q])),h.appendChild(m),p[q]=m;a._textdirty=!1}else for(p=h.getElementsByTagName("tspan"),q=0,r=p.length;r>q;q++)q?u(p[q],{dy:k,x:g.x}):u(p[0],{dy:k*p.length*l,x:g.x});u(h,{x:g.x,y:g.y}),a._.dirty=1;var s=a._getBBox(),t=g.y-(s.y+s.height/2);if(s.isCalculated)switch(g["vertical-align"]){case"top":t=.75*s.height;break;case"bottom":t=-(.25*s.height);break;default:t=g.y-(s.y+.25*s.height)}t&&c.is(t,"finite")&&p[0]&&u(p[0],{dy:t})}},F=function(a,b,d){var e=this,f=d||b;e.node=e[0]=a,a.raphael=!0,a.raphaelid=e.id=c._oid++,e.matrix=c.matrix(),e.realPath=null,e.attrs=e.attrs||{},e.styles=e.styles||{},e.followers=e.followers||[],e.paper=b,e.ca=e.customAttributes=e.customAttributes||new b._CustomAttributes,e._={transform:[],sx:1,sy:1,deg:0,dx:0,dy:0,dirty:1},e.parent=f,!f.bottom&&(f.bottom=e),e.prev=f.top,f.top&&(f.top.next=e),f.top=e,e.next=null},G=c.el;F.prototype=G,G.constructor=F,c._engine.getNode=function(a){var b=a.node||a[0].node;return b.titleNode||b},c._engine.getLastNode=function(a){var b=a.node||a[a.length-1].node;return b.titleNode||b},c._engine.path=function(a,b,c){var d=u("path");c&&c.canvas&&c.canvas.appendChild(d)||b.canvas&&b.canvas.appendChild(d);var e=new F(d,b,c);return e.type="path",C(e,{fill:"none",stroke:"#000",path:a}),e},G.rotate=function(a,b,c){var f,g=this;return g.removed?g:(a=d(a).split(l),a.length-1&&(b=e(a[1]),c=e(a[2])),a=e(a[0]),null==c&&(b=c),(null==b||null==c)&&(f=g.getBBox(1),b=f.x+f.width/2,c=f.y+f.height/2),g.transform(g._.transform.concat([["r",a,b,c]])),g)},G.scale=function(a,b,c,f){var g,h=this;return h.removed?h:(a=d(a).split(l),a.length-1&&(b=e(a[1]),c=e(a[2]),f=e(a[3])),a=e(a[0]),null==b&&(b=a),null==f&&(c=f),(null==c||null==f)&&(g=h.getBBox(1)),c=null==c?g.x+g.width/2:c,f=null==f?g.y+g.height/2:f,h.transform(h._.transform.concat([["s",a,b,c,f]])),h)},G.translate=function(a,b){var c=this;return c.removed?c:(a=d(a).split(l),a.length-1&&(b=e(a[1])),a=e(a[0])||0,b=+b||0,c.transform(c._.transform.concat([["t",a,b]])),c)},G.transform=function(a){var d,e=this,f=e._;return null==a?f.transform:(c._extractTransform(e,a),e.clip&&!f.clipispath&&u(e.clip,{transform:e.matrix.invert()}),e.pattern&&y(e),e.node&&u(e.node,{transform:e.matrix}),(1!=f.sx||1!=f.sy)&&(d=e.attrs[b]("stroke-width")?e.attrs["stroke-width"]:1,e.attr({"stroke-width":d})),e)},G.hide=function(){var a=this;return!a.removed&&a.paper.safari(a.node.style.display="none"),a},G.show=function(){var a=this;return!a.removed&&a.paper.safari(a.node.style.display=o),a},G.remove=function(){if(!this.removed&&this.parent.canvas){var a,b=this,d=c._engine.getNode(b),e=b.paper,f=e.defs;for(e.__set__&&e.__set__.exclude(b),n.unbind("raphael.*.*."+b.id),b.gradient&&f&&f.removeChild(b.gradient);a=b.followers.pop();)a.el.remove();b.parent.canvas.removeChild(d),c._tear(b,e);for(a in b)b[a]="function"==typeof b[a]?c._removedFactory(a):null;b.removed=!0}},G._getBBox=function(){var a,b,c=this,d=c.node,e={},f=c.attrs;"none"===d.style.display&&(c.show(),b=!0);try{e=d.getBBox(),"text"==c.type&&(void 0===e.x&&(e.isCalculated=!0,a=f["text-anchor"],e.x=(f.x||0)-e.width*("start"===a?0:"middle"===a?.5:1)),void 0===e.y&&(e.isCalculated=!0,a=f["vertical-align"],e.y=(f.y||0)-e.height*("bottom"===a?1:"middle"===a?.5:0)))}catch(g){}finally{e=e||{}}return b&&c.hide(),e},G.css=function(a,d){if(this.removed)return this;if(null==d&&c.is(a,"string")){for(var e=a.split(l),f={},g=0,h=e.length;h>g;g++)a=e[g],a in this.styles&&(f[a]=this.styles[a]);return h-1?f:f[e[0]]}if(null==d&&c.is(a,"array")){for(f={},g=0,h=a.length;h>g;g++)f[a[g]]=this.styles(a[g]);return f}if(null!=d){var i={};i[a]=d}else null!=a&&c.is(a,"object")&&(i=a);var j,k={};for(var m in i)j=m.replace(/\B([A-Z]{1})/g,"-$1").toLowerCase(),c._availableAttrs[b](j)||"color"===j?("color"===j&&"text"===this.type&&(j="fill"),k[j]=i[m],k.dirty=!0):(n("raphael.css."+j+"."+this.id,this,i[m],j),this.node.style[j]=i[m],this.styles[j]=i[m]);for(g=0,h=this.followers.length;h>g;g++)this.followers[g].el.css(i);return k[b]("dirty")&&(delete k.dirty,this.attr(k)),this},G.attr=function(a,d){if(this.removed)return this;if(null==a){var e={};for(var f in this.attrs)this.attrs[b](f)&&(e[f]=this.attrs[f]);return e.gradient&&"none"==e.fill&&(e.fill=e.gradient)&&delete e.gradient,e.transform=this._.transform,e.visibility="none"===this.node.style.display?"hidden":"visible",e}if(null==d&&c.is(a,"string")){if("fill"==a&&"none"==this.attrs.fill&&this.attrs.gradient)return this.attrs.gradient;if("transform"==a)return this._.transform;if("visibility"==a)return"none"===this.node.style.display?"hidden":"visible";for(var g=a.split(l),h={},i=0,j=g.length;j>i;i++)a=g[i],h[a]=a in this.attrs?this.attrs[a]:c.is(this.ca[a],"function")?this.ca[a].def:c._availableAttrs[a];return j-1?h:h[g[0]]}if(null==d&&c.is(a,"array")){for(h={},i=0,j=a.length;j>i;i++)h[a[i]]=this.attr(a[i]);return h}if(null!=d){var k={};k[a]=d}else null!=a&&c.is(a,"object")&&(k=a);for(var m in k)n("raphael.attr."+m+"."+this.id,this,k[m],m);var o={};for(m in this.ca)if(this.ca[m]&&k[b](m)&&c.is(this.ca[m],"function")&&!this.ca["_invoked"+m]){this.ca["_invoked"+m]=!0;var p=this.ca[m].apply(this,[].concat(k[m]));delete this.ca["_invoked"+m];for(var q in p)p[b](q)&&(k[q]=p[q]);this.attrs[m]=k[m],p===!1&&(o[m]=k[m],delete k[m])}C(this,k);var r;for(i=0,j=this.followers.length;j>i;i++)r=this.followers[i],r.cb&&!r.cb.call(r.el,k,this)||r.el.attr(k);for(q in o)k[q]=o[q];return this},G.blur=function(a){var b=this;if(0!==+a){var d=u("filter"),e=u("feGaussianBlur");b.attrs.blur=a,d.id=c.createUUID(),u(e,{stdDeviation:+a||1.5}),d.appendChild(e),b.paper.defs.appendChild(d),b._blur=d,u(b.node,{filter:"url('"+c._url+"#"+d.id+"')"})}else b._blur&&(b._blur.parentNode.removeChild(b._blur),delete b._blur,delete b.attrs.blur),b.node.removeAttribute("filter")},G.on=function(a,b){if(this.removed)return this;var d=b;return c.supportsTouch&&(a=c._touchMap[a]||"click"===a&&"touchstart"||a,d=function(a){a.preventDefault(),b()}),this.node["on"+a]=d,this},c._engine.group=function(a,b,c){var d=u("g");c&&c.canvas&&c.canvas.appendChild(d)||a.canvas&&a.canvas.appendChild(d);var e=new F(d,a,c);return e.type="group",e.canvas=e.node,e.top=null,e.bottom=null,b&&d.setAttribute("class",["red",b,e.id].join("-")),e},c._engine.circle=function(a,b,c,d,e){var f=u("circle");e&&e.canvas&&e.canvas.appendChild(f)||a.canvas&&a.canvas.appendChild(f);var g=new F(f,a,e);return g.attrs={cx:b,cy:c,r:d,fill:"none",stroke:"#000"},g.type="circle",u(f,g.attrs),g},c._engine.rect=function(a,b,c,d,e,f,g){var h=u("rect");g&&g.canvas&&g.canvas.appendChild(h)||a.canvas&&a.canvas.appendChild(h);var i=new F(h,a,g);return i.attrs={x:b,y:c,width:d,height:e,r:f||0,rx:f||0,ry:f||0,fill:"none",stroke:"#000"},i.type="rect",u(h,i.attrs),i},c._engine.ellipse=function(a,b,c,d,e,f){var g=u("ellipse");f&&f.canvas&&f.canvas.appendChild(g)||a.canvas&&a.canvas.appendChild(g);var h=new F(g,a,f);return h.attrs={cx:b,cy:c,rx:d,ry:e,fill:"none",stroke:"#000"},h.type="ellipse",u(g,h.attrs),h},c._engine.image=function(a,b,c,d,e,f,g){var h=u("image");u(h,{x:c,y:d,width:e,height:f,preserveAspectRatio:"none"}),h.setAttributeNS(q,"href",b),g&&g.canvas&&g.canvas.appendChild(h)||a.canvas&&a.canvas.appendChild(h);var i=new F(h,a,g);return i.attrs={x:c,y:d,width:e,height:f,src:b},i.type="image",i},c._engine.text=function(a,b,c,d,e){var f=u("text");e&&e.canvas&&e.canvas.appendChild(f)||a.canvas&&a.canvas.appendChild(f);var g=new F(f,a,e);return g.attrs={x:b,y:c,"text-anchor":"middle","vertical-align":"middle",text:d,stroke:"none",fill:"#000"},g.type="text",g._textdirty=!0,C(g,g.attrs),g},c._engine.setSize=function(a,b){return this.width=a||this.width,this.height=b||this.height,this.canvas.setAttribute("width",this.width),this.canvas.setAttribute("height",this.height),this._viewBox&&this.setViewBox.apply(this,this._viewBox),this},c._engine.create=function(){var a=c._getContainer.apply(0,arguments),b=a&&a.container,d=a.x,e=a.y,f=a.width,g=a.height;if(!b)throw new Error("SVG container not found.");var h,i=u("svg"),j="overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);-webkit-user-select:none;-moz-user-select:-moz-none;-khtml-user-select:none;-ms-user-select:none;user-select:none;-o-user-select:none;cursor:default;";return d=d||0,e=e||0,f=f||512,g=g||342,u(i,{height:g,version:1.1,width:f,xmlns:"http://www.w3.org/2000/svg"}),1==b?(i.style.cssText=j+"position:absolute;left:"+d+"px;top:"+e+"px",c._g.doc.body.appendChild(i),h=1):(i.style.cssText=j+"position:relative",b.firstChild?b.insertBefore(i,b.firstChild):b.appendChild(i)),b=new c._Paper,b.width=f,b.height=g,b.canvas=i,b.clear(),b._left=b._top=0,h&&(b.renderfix=function(){}),b.renderfix(),b},c._engine.setViewBox=function(a,b,c,d,e){n("raphael.setViewBox",this,this._viewBox,[a,b,c,d,e]);var f,g,i=h(c/this.width,d/this.height),j=this.top,k=e?"meet":"xMinYMin";for(null==a?(this._vbSize&&(i=1),delete this._vbSize,f="0 0 "+this.width+p+this.height):(this._vbSize=i,f=a+p+b+p+c+p+d),u(this.canvas,{viewBox:f,preserveAspectRatio:k});i&&j;)g="stroke-width"in j.attrs?j.attrs["stroke-width"]:1,j.attr({"stroke-width":g}),j._.dirty=1,j._.dirtyT=1,j=j.prev;return this._viewBox=[a,b,c,d,!!e],this},c.prototype.renderfix=function(){var a,b=this.canvas,c=b.style;try{a=b.getScreenCTM()||b.createSVGMatrix()}catch(d){a=b.createSVGMatrix()}var e=-a.e%1,f=-a.f%1;(e||f)&&(e&&(this._left=(this._left+e)%1,c.left=this._left+"px"),f&&(this._top=(this._top+f)%1,c.top=this._top+"px"))},c.prototype.clear=function(){n("raphael.clear",this);for(var a=this.canvas;a.firstChild;)a.removeChild(a.firstChild);this.bottom=this.top=null,(this.desc=u("desc")).appendChild(c._g.doc.createTextNode(c.is(c.desc,"string")&&c.desc||"Created with Red Raphaël "+c.version)),a.appendChild(this.desc),a.appendChild(this.defs=u("defs"))},c.prototype.remove=function(){n("raphael.remove",this),this.canvas.parentNode&&this.canvas.parentNode.removeChild(this.canvas);for(var a in this)this[a]="function"==typeof this[a]?c._removedFactory(a):null;this.removed=!0};var H=c.st;for(var I in G)G[b](I)&&!H[b](I)&&(H[I]=function(a){return function(){var b=arguments;return this.forEach(function(c){c[a].apply(c,b)})}}(I))}}(),function(){if(c.vml){var a="hasOwnProperty",b=String,d=parseFloat,e=Math,f=e.round,g=e.max,h=e.min,i=e.sqrt,j=e.abs,k="fill",l=/[, ]+/,m=c.eve,n=" progid:DXImageTransform.Microsoft",o=" ",p="",q={M:"m",L:"l",C:"c",Z:"x",m:"t",l:"r",c:"v",z:"x"},r=/([clmz]),?([^clmz]*)/gi,s=/ progid:\S+Blur\([^\)]+\)/g,t=/-?[^,\s-]+/g,u="position:absolute;left:0;top:0;width:1px;height:1px",v=21600,w={path:1,rect:1,image:1},x={circle:1,ellipse:1},y=function(a){var d=/[ahqstv]/gi,e=c._pathToAbsolute;if(b(a).match(d)&&(e=c._path2curve),d=/[clmz]/g,e==c._pathToAbsolute&&!b(a).match(d)){var g=b(a).replace(r,function(a,b,c){var d=[],e="m"==b.toLowerCase(),g=q[b];return c.replace(t,function(a){e&&2==d.length&&(g+=d+q["m"==b?"l":"L"],d=[]),d.push(f(a*v))}),g+d});return g||"m0,0"}var h,i,j=e(a);g=[];for(var k=0,l=j.length;l>k;k++){h=j[k],i=j[k][0].toLowerCase(),"z"==i&&(i="x");for(var m=1,n=h.length;n>m;m++)i+=f(h[m]*v)+(m!=n-1?",":p);g.push(i)}return g.length?g.join(o):"m0,0"},z=function(a,b,d){var e=c.matrix();return e.rotate(-a,.5,.5),{dx:e.x(b,d),dy:e.y(b,d)}},A=function(a,b,c,d,e,f){var g=a._,h=a.matrix,i=g.fillpos,l=a.node,m=l.style,n=1,p="",q=v/b,r=v/c;if(m.visibility="hidden",b&&c){if(l.coordsize=j(q)+o+j(r),m.rotation=f*(0>b*c?-1:1),f){var s=z(f,d,e);d=s.dx,e=s.dy}if(0>b&&(p+="x"),0>c&&(p+=" y")&&(n=-1),m.flip=p,l.coordorigin=d*-q+o+e*-r,i||g.fillsize){var t=l.getElementsByTagName(k);t=t&&t[0],l.removeChild(t),i&&(s=z(f,h.x(i[0],i[1]),h.y(i[0],i[1])),t.position=s.dx*n+o+s.dy*n),g.fillsize&&(t.size=g.fillsize[0]*j(b)+o+g.fillsize[1]*j(c)),l.appendChild(t)}m.visibility="visible"}};c._url=p,c.toString=function(){return"Your browser doesn’t support SVG. Falling down to VML.\nYou are running Raphaël "+this.version};var B=function(a,c,d){for(var e=b(c).toLowerCase().split("-"),f=d?"end":"start",g=e.length,h="classic",i="medium",j="medium";g--;)switch(e[g]){case"block":case"classic":case"oval":case"diamond":case"open":case"none":h=e[g];break;case"wide":case"narrow":j=e[g];break;case"long":case"short":i=e[g]}var k=a.node.getElementsByTagName("stroke")[0];k[f+"arrow"]=h,k[f+"arrowlength"]=i,k[f+"arrowwidth"]=j},C=c._setFillAndStroke=function(e,i){if(e.paper.canvas){e.attrs=e.attrs||{};var j=e.node,m=e.attrs,n=j.style,q=w[e.type]&&(i.x!=m.x||i.y!=m.y||i.width!=m.width||i.height!=m.height||i.cx!=m.cx||i.cy!=m.cy||i.rx!=m.rx||i.ry!=m.ry||i.r!=m.r),r=x[e.type]&&(m.cx!=i.cx||m.cy!=i.cy||m.r!=i.r||m.rx!=i.rx||m.ry!=i.ry),s="group"===e.type,t=e;for(var u in i)i[a](u)&&(m[u]=i[u]);if(q&&(m.path=c._getPath[e.type](e),e._.dirty=1),i.href&&(j.href=i.href),i.title&&(j.title=i.title),i.target&&(j.target=i.target),i.cursor&&(n.cursor=i.cursor),"blur"in i&&e.blur(i.blur),(i.path&&"path"==e.type||q)&&(j.path=y(~b(m.path).toLowerCase().indexOf("r")?c._pathToAbsolute(m.path):m.path),"image"==e.type&&(e._.fillpos=[m.x,m.y],e._.fillsize=[m.width,m.height],A(e,1,1,0,0,0))),"transform"in i&&e.transform(i.transform),"rotation"in i){var z=i.rotation;c.is(z,"array")?e.rotate.apply(e,z):e.rotate(z)}if("visibility"in i&&("hidden"===i.visibility?e.hide():e.show()),r){var C=+m.cx,E=+m.cy,F=+m.rx||+m.r||0,H=+m.ry||+m.r||0;j.path=c.format("ar{0},{1},{2},{3},{4},{1},{4},{1}x",f((C-F)*v),f((E-H)*v),f((C+F)*v),f((E+H)*v),f(C*v))}if("clip-rect"in i){var I=b(i["clip-rect"]).split(l);if(4==I.length){I[0]=+I[0],I[1]=+I[1],I[2]=+I[2]+I[0],I[3]=+I[3]+I[1];var J,K=s?j:j.clipRect||c._g.doc.createElement("div"),L=K.style;s?(e.clip=I.slice(),J=e.matrix.offset(),J=[d(J[0]),d(J[1])],I[0]-=J[0],I[1]-=J[1],I[2]-=J[0],I[3]-=J[1],L.width="10800px",L.height="10800px"):j.clipRect||(L.top="0",L.left="0",L.width=e.paper.width+"px",L.height=e.paper.height+"px",j.parentNode.insertBefore(K,j),K.appendChild(j),j.clipRect=K),L.position="absolute",L.clip=c.format("rect({1}px {2}px {3}px {0}px)",I)}i["clip-rect"]||(s&&e.clip?(j.style.clip="rect(auto auto auto auto)",delete e.clip):j.clipRect&&(j.clipRect.style.clip="rect(auto auto auto auto)"))}if(e.textpath){var M=e.textpath.style;i.font&&(M.font=i.font),i["font-family"]&&(M.fontFamily='"'+i["font-family"].split(",")[0].replace(/^['"]+|['"]+$/g,p)+'"'),i["font-size"]&&(M.fontSize=i["font-size"]),i["font-weight"]&&(M.fontWeight=i["font-weight"]),i["font-style"]&&(M.fontStyle=i["font-style"])}if("arrow-start"in i&&B(t,i["arrow-start"]),"arrow-end"in i&&B(t,i["arrow-end"],1),null!=i.opacity||null!=i["stroke-width"]||null!=i.fill||null!=i.src||null!=i.stroke||null!=i["stroke-width"]||null!=i["stroke-opacity"]||null!=i["fill-opacity"]||null!=i["stroke-dasharray"]||null!=i["stroke-miterlimit"]||null!=i["stroke-linejoin"]||null!=i["stroke-linecap"]){var N=j.getElementsByTagName(k),O=!1,P=-1;if(N=N&&N[0],!N&&(O=N=G(k)),"image"==e.type&&i.src&&(N.src=i.src),i.fill&&(N.on=!0),(null==N.on||"none"==i.fill||null===i.fill)&&(N.on=!1),N.on&&i.fill){var Q=b(i.fill).match(c._ISURL);if(Q){N.parentNode==j&&j.removeChild(N),N.rotate=!0,N.src=Q[1],N.type="tile";var R=e.getBBox(1);N.position=R.x+o+R.y,e._.fillpos=[R.x,R.y],c._preload(Q[1],function(){e._.fillsize=[this.offsetWidth,this.offsetHeight]})}else{var S=c.getRGB(i.fill);N.color=S.hex,N.src=p,N.type="solid",S.error&&(t.type in{circle:1,ellipse:1}||"r"!=b(i.fill).charAt())&&D(t,i.fill,N)?(m.fill="none",m.gradient=i.fill,N.rotate=!1):"opacity"in S&&!("fill-opacity"in i)&&(P=S.opacity)}}if(-1!==P||"fill-opacity"in i||"opacity"in i){var T=((+m["fill-opacity"]+1||2)-1)*((+m.opacity+1||2)-1)*((+P+1||2)-1);T=h(g(T,0),1),N.opacity=T,N.src&&(N.color="none")}j.appendChild(N);var U=j.getElementsByTagName("stroke")&&j.getElementsByTagName("stroke")[0],V=!1;!U&&(V=U=G("stroke")),(i.stroke&&"none"!=i.stroke||i["stroke-width"]||null!=i["stroke-opacity"]||i["stroke-dasharray"]||i["stroke-miterlimit"]||i["stroke-linejoin"]||i["stroke-linecap"])&&(U.on=!0),("none"==i.stroke||null===i.stroke||null==U.on||0==i.stroke||0==i["stroke-width"])&&(U.on=!1);var W=c.getRGB("stroke"in i?i.stroke:m.stroke);U.on&&i.stroke&&(U.color=W.hex),T=((+m["stroke-opacity"]+1||2)-1)*((+m.opacity+1||2)-1)*((+W.opacity+1||2)-1);var X=.75*(d(i["stroke-width"])||1);if(T=h(g(T,0),1),null==i["stroke-width"]&&(X=m["stroke-width"]),i["stroke-width"]&&(U.weight=X),X&&1>X&&(T*=X)&&(U.weight=1),U.opacity=T,i["stroke-linejoin"]&&(U.joinstyle=i["stroke-linejoin"])||V&&(V.joinstyle="miter"),U.miterlimit=i["stroke-miterlimit"]||8,i["stroke-linecap"]&&(U.endcap="butt"==i["stroke-linecap"]?"flat":"square"==i["stroke-linecap"]?"square":"round"),i["stroke-dasharray"]){var Y={"-":"shortdash",".":"shortdot","-.":"shortdashdot","-..":"shortdashdotdot",". ":"dot","- ":"dash","--":"longdash","- .":"dashdot","--.":"longdashdot","--..":"longdashdotdot"};U.dashstyle=Y[a](i["stroke-dasharray"])?Y[i["stroke-dasharray"]]:i["stroke-dasharray"].join&&i["stroke-dasharray"].join(" ")||p}V&&j.appendChild(U)}if("text"==t.type){t.paper.canvas.style.display=p;var Z=t.paper.span,$=100,_=m.font&&m.font.match(/\d+(?:\.\d*)?(?=px)/),ab=m["line-height"]&&(m["line-height"]+p).match(/\d+(?:\.\d*)?(?=px)/);n=Z.style,m.font&&(n.font=m.font),m["font-family"]&&(n.fontFamily=m["font-family"]),m["font-weight"]&&(n.fontWeight=m["font-weight"]),m["font-style"]&&(n.fontStyle=m["font-style"]),_=d(m["font-size"]||_&&_[0])||10,n.fontSize=_*$+"px",ab=d(m["line-height"]||ab&&ab[0])||12,m["line-height"]&&(n.lineHeight=ab*$+"px"),t.textpath.string&&(Z.innerHTML=b(t.textpath.string).replace(/"));var bb=Z.getBoundingClientRect();switch(t.W=m.w=(bb.right-bb.left)/$,t.H=m.h=(bb.bottom-bb.top)/$,t.X=m.x,t.Y=m.y,m["vertical-align"]){case"top":t.bby=t.H/2;break;case"bottom":t.bby=-t.H/2;break;default:t.bby=0}("x"in i||"y"in i||void 0!==t.bby)&&(t.path.v=c.format("m{0},{1}l{2},{1}",f(m.x*v),f((m.y+(t.bby||0))*v),f(m.x*v)+1));for(var cb=["x","y","text","font","font-family","font-weight","font-style","font-size","line-height"],db=0,eb=cb.length;eb>db;db++)if(cb[db]in i){t._.dirty=1;break}switch(m["text-anchor"]){case"start":t.textpath.style["v-text-align"]="left",t.bbx=t.W/2;break;case"end":t.textpath.style["v-text-align"]="right",t.bbx=-t.W/2;break;default:t.textpath.style["v-text-align"]="center",t.bbx=0}t.textpath.style["v-text-kern"]=!0}}},D=function(a,e,f){a.attrs=a.attrs||{};var g=(a.attrs,Math.pow),h="linear",j=".5 .5";if(a.attrs.gradient=e,e=b(e).replace(c._radial_gradient,function(a,b){h="radial",b=b&&b.split(",")||[];var c=(b[0],b[1],b[2],b[3]),e=b[4];return b[5],c&&e&&(c=d(c),e=d(e),g(c-.5,2)+g(e-.5,2)>.25&&(e=i(.25-g(c-.5,2))*(2*(e>.5)-1)+.5),j=c+o+e),p}),e=e.split(/\s*\-\s*/),"linear"==h){var k=e.shift();if(k=-d(k),isNaN(k))return null}var l=c._parseDots(e);if(!l)return null;if(a=a.shape||a.node,l.length){a.removeChild(f),f.on=!0,f.method="none",f.color=l[0].color,f.color2=l[l.length-1].color;for(var m=[],n=1,q=void 0===l[0].opacity?1:l[0].opacity,r=0,s=l.length;s>r;r++)l[r].offset&&m.push(l[r].offset+o+l[r].color),void 0!==l[r].opacity&&(n=l[r].opacity);f.colors=m.length?m.join():"0% "+f.color,f.opacity=n,f["o:opacity2"]=q,"radial"==h?(f.type="gradientTitle",f.focus="100%",f.focussize="0 0",f.focusposition=j,f.angle=0):(f.type="gradient",f.angle=(270-k)%360),a.appendChild(f)}return 1},E=function(a,b,d){var e=this,f=d||b;e.node=e[0]=a,a.raphael=!0,a.raphaelid=e.id=c._oid++,e.X=0,e.Y=0,e.attrs=e.attrs||{},e.styles=e.styles||{},e.followers=e.followers||[],e.paper=b,e.ca=e.customAttributes=e.customAttributes||new b._CustomAttributes,e.matrix=c.matrix(),e._={transform:[],sx:1,sy:1,dx:0,dy:0,deg:0,dirty:1,dirtyT:1},e.parent=f,!f.bottom&&(f.bottom=e),e.prev=f.top,f.top&&(f.top.next=e),f.top=e,e.next=null},F=c.el;E.prototype=F,F.constructor=E,F.transform=function(a){if(null==a)return this._.transform;var d,e=this.paper._viewBoxShift,f=e?"s"+[e.scale,e.scale]+"-1-1t"+[e.dx,e.dy]:p;e&&(d=a=b(a).replace(/\.{3}|\u2026/g,this._.transform||p)),c._extractTransform(this,f+a);var g,h=this.matrix.clone(),i=this.skew,j=this.node,k=~b(this.attrs.fill).indexOf("-"),l=!b(this.attrs.fill).indexOf("url(");if(h.translate(-.5,-.5),l||k||"image"==this.type)if(i.matrix="1 0 0 1",i.offset="0 0",g=h.split(),k&&g.noRotation||!g.isSimple){j.style.filter=h.toFilter();var m=this.getBBox(),n=this.getBBox(1),q=m.x2&&n.x2&&"x2"||"x",r=m.y2&&n.y2&&"y2"||"y",s=m[q]-n[q],t=m[r]-n[r];j.coordorigin=s*-v+o+t*-v,A(this,1,1,s,t,0)}else j.style.filter=p,A(this,g.scalex,g.scaley,g.dx,g.dy,g.rotate);else j.style.filter=p,i.matrix=b(h),i.offset=h.offset();return d&&(this._.transform=d),this},F.rotate=function(a,c,e){if(this.removed)return this;if(null!=a){if(a=b(a).split(l),a.length-1&&(c=d(a[1]),e=d(a[2])),a=d(a[0]),null==e&&(c=e),null==c||null==e){var f=this.getBBox(1);c=f.x+f.width/2,e=f.y+f.height/2}return this._.dirtyT=1,this.transform(this._.transform.concat([["r",a,c,e]])),this}},F.translate=function(a,c){return this.removed?this:(a=b(a).split(l),a.length-1&&(c=d(a[1])),a=d(a[0])||0,c=+c||0,this._.bbox&&(this._.bbox.x+=a,this._.bbox.y+=c),this.transform(this._.transform.concat([["t",a,c]])),this)},F.scale=function(a,c,e,f){if(this.removed)return this;if(a=b(a).split(l),a.length-1&&(c=d(a[1]),e=d(a[2]),f=d(a[3]),isNaN(e)&&(e=null),isNaN(f)&&(f=null)),a=d(a[0]),null==c&&(c=a),null==f&&(e=f),null==e||null==f)var g=this.getBBox(1);return e=null==e?g.x+g.width/2:e,f=null==f?g.y+g.height/2:f,this.transform(this._.transform.concat([["s",a,c,e,f]])),this._.dirtyT=1,this},F.hide=function(){var a=this;return!a.removed&&(a.node.style.display="none"),a},F.show=function(){var a=this;return!a.removed&&(a.node.style.display=p),a},F._getBBox=function(){return this.removed?{}:{x:this.X+(this.bbx||0)-this.W/2,y:this.Y+(this.bby||0)-this.H/2,width:this.W,height:this.H}},F.remove=function(){if(!this.removed&&this.parent.canvas){var a,b=c._engine.getNode(this);for(this.paper.__set__&&this.paper.__set__.exclude(this),m.unbind("raphael.*.*."+this.id);a=this.followers.pop();)a.el.remove();this.shape&&this.shape.parentNode.removeChild(this.shape),b.parentNode.removeChild(b),c._tear(this,this.paper);for(var a in this)this[a]="function"==typeof this[a]?c._removedFactory(a):null;this.removed=!0}},F.css=function(b,d){if(this.removed)return this;if(null==d&&c.is(b,"string")){for(var e=b.split(l),f={},g=0,h=e.length;h>g;g++)b=e[g],b in this.styles&&(f[b]=this.styles[b]);return h-1?f:f[e[0]]}if(null==d&&c.is(b,"array")){for(f={},g=0,h=b.length;h>g;g++)f[b[g]]=this.styles(b[g]);return f}if(null!=d){var i={};i[b]=d}else null!=b&&c.is(b,"object")&&(i=b);var j,k={};for(var n in i)j=n.replace(/\B([A-Z]{1})/g,"-$1").toLowerCase(),"color"===j&&"text"===this.type&&(j="fill"),c._availableAttrs[a](j)?(k[j]=i[n],k.dirty=!0):(m("raphael.css."+j+"."+this.id,this,i[n],j),void 0!=i[n]&&(this.node.style[j]=i[n]),this.styles[j]=i[n]);for(g=0,h=this.followers.length;h>g;g++)this.followers[g].el.css(i);return k[a]("dirty")&&(delete k.dirty,this.attr(k)),this},F.attr=function(b,d){if(this.removed)return this;if(null==b){var e={};for(var f in this.attrs)this.attrs[a](f)&&(e[f]=this.attrs[f]);return e.gradient&&"none"==e.fill&&(e.fill=e.gradient)&&delete e.gradient,e.transform=this._.transform,e.visibility="none"===this.node.style.display?"hidden":"visible",e}if(null==d&&c.is(b,"string")){if(b==k&&"none"==this.attrs.fill&&this.attrs.gradient)return this.attrs.gradient;if("visibility"==b)return"none"===this.node.style.display?"hidden":"visible";for(var g=b.split(l),h={},i=0,j=g.length;j>i;i++)b=g[i],h[b]=b in this.attrs?this.attrs[b]:c.is(this.ca[b],"function")?this.ca[b].def:c._availableAttrs[b];return j-1?h:h[g[0]]}if(this.attrs&&null==d&&c.is(b,"array")){for(h={},i=0,j=b.length;j>i;i++)h[b[i]]=this.attr(b[i]);return h}var n;null!=d&&(n={},n[b]=d),null==d&&c.is(b,"object")&&(n=b);for(var o in n)m("raphael.attr."+o+"."+this.id,this,n[o],o);if(n){var p={};for(o in this.ca)if(this.ca[o]&&n[a](o)&&c.is(this.ca[o],"function")&&!this.ca["_invoked"+o]){this.ca["_invoked"+o]=!0;var q=this.ca[o].apply(this,[].concat(n[o]));
+delete this.ca["_invoked"+o];for(var r in q)q[a](r)&&(n[r]=q[r]);this.attrs[o]=n[o],q===!1&&(p[o]=n[o],delete n[o])}"text"in n&&"text"==this.type&&(this.textpath.string=n.text.replace(/ /gi,"\n")),C(this,n);var s;for(i=0,j=this.followers.length;j>i;i++)s=this.followers[i],s.cb&&!s.cb.call(s.el,n,this)||s.el.attr(n);for(var r in p)n[r]=p[r]}return this},F.blur=function(a){var b=this.node.runtimeStyle,d=b.filter;return d=d.replace(s,p),0!==+a?(this.attrs.blur=a,b.filter=d+o+n+".Blur(pixelradius="+(+a||1.5)+")",b.margin=c.format("-{0}px 0 0 -{0}px",f(+a||1.5))):(b.filter=d,b.margin=0,delete this.attrs.blur),this},F.on=function(a,b){return this.removed?this:(this.node["on"+a]=function(){var a=c._g.win.event;a.target=a.srcElement,b(a)},this)},c._engine.getNode=function(a){var b=a.node||a[0].node;return b.clipRect||b},c._engine.getLastNode=function(a){var b=a.node||a[a.length-1].node;return b.clipRect||b},c._engine.group=function(a,b,d){var e=c._g.doc.createElement("div"),f=new E(e,a,d);return e.style.cssText=u,b&&(e.className=["red",b,f.id].join("-")),(d||a).canvas.appendChild(e),f.type="group",f.canvas=f.node,f.transform=c._engine.group.transform,f.top=null,f.bottom=null,f},c._engine.group.transform=function(a){if(null==a)return this._.transform;var e,f,g,h,i,j=this,k=j.node.style,l=j.clip,m=j.paper._viewBoxShift,n=m?"s"+[m.scale,m.scale]+"-1-1t"+[m.dx,m.dy]:p;return m&&(e=a=b(a).replace(/\.{3}|\u2026/g,j._.transform||p)),c._extractTransform(j,n+a),f=j.matrix,g=f.offset(),h=d(g[0])||0,i=d(g[1])||0,k.left=h+"px",k.top=i+"px",k.zoom=(j._.tzoom=f.get(0))+p,l&&(k.clip=c.format("rect({1}px {2}px {3}px {0}px)",[l[0]-h,l[1]-i,l[2]-h,l[3]-i])),j},c._engine.path=function(a,b,c){var d=G("shape");d.style.cssText=u,d.coordsize=v+o+v,d.coordorigin=b.coordorigin;var e=new E(d,b,c),f={fill:"none",stroke:"#000"};a&&(f.path=a),e.type="path",e.path=[],e.Path=p,C(e,f),(c||b).canvas.appendChild(d);var g=G("skew");return g.on=!0,d.appendChild(g),e.skew=g,e},c._engine.rect=function(a,b,d,e,f,g,h){var i=c._rectPath(b,d,e,f,g),j=a.path(i,h),k=j.attrs;return j.X=k.x=b,j.Y=k.y=d,j.W=k.width=e,j.H=k.height=f,k.r=g,k.path=i,j.type="rect",j},c._engine.ellipse=function(a,b,c,d,e,f){var g=a.path(void 0,f);return g.X=b-d,g.Y=c-e,g.W=2*d,g.H=2*e,g.type="ellipse",C(g,{cx:b,cy:c,rx:d,ry:e}),g},c._engine.circle=function(a,b,c,d,e){var f=a.path(void 0,e);return f.X=b-d,f.Y=c-d,f.W=f.H=2*d,f.type="circle",C(f,{cx:b,cy:c,r:d}),f},c._engine.image=function(a,b,d,e,f,g,h){var i=c._rectPath(d,e,f,g),j=a.path(i,h).attr({stroke:"none"}),l=j.attrs,m=j.node,n=m.getElementsByTagName(k)[0];return l.src=b,j.X=l.x=d,j.Y=l.y=e,j.W=l.width=f,j.H=l.height=g,l.path=i,j.type="image",n.parentNode==m&&m.removeChild(n),n.rotate=!0,n.src=b,n.type="tile",j._.fillpos=[d,e],j._.fillsize=[f,g],m.appendChild(n),A(j,1,1,0,0,0),j},c._engine.text=function(a,d,e,g,h){var i=G("shape"),j=G("path"),k=G("textpath");d=d||0,e=e||0,g=g,j.v=c.format("m{0},{1}l{2},{1}",f(d*v),f(e*v),f(d*v)+1),j.textpathok=!0,k.string=b(g).replace(/ /gi,"\n"),k.on=!0,i.style.cssText=u,i.coordsize=v+o+v,i.coordorigin="0 0";var l=new E(i,a,h),m={fill:"#000",stroke:"none",text:g};l.shape=i,l.path=j,l.textpath=k,l.type="text",l.attrs.text=b(g||p),l.attrs.x=d,l.attrs.y=e,l.attrs.w=1,l.attrs.h=1,C(l,m),i.appendChild(k),i.appendChild(j),(h||a).canvas.appendChild(i);var n=G("skew");return n.on=!0,i.appendChild(n),l.skew=n,l},c._engine.setSize=function(a,b){var d=this.canvas.style;return this.width=a,this.height=b,a==+a&&(a+="px"),b==+b&&(b+="px"),d.width=a,d.height=b,d.clip="rect(0 "+a+" "+b+" 0)",this._viewBox&&c._engine.setViewBox.apply(this,this._viewBox),this},c._engine.setViewBox=function(a,b,c,d,e){m("raphael.setViewBox",this,this._viewBox,[a,b,c,d,e]);var f,h,i=this.width,j=this.height,k=1/g(c/i,d/j);return e&&(f=j/d,h=i/c,i>c*f&&(a-=(i-c*f)/2/f),j>d*h&&(b-=(j-d*h)/2/h)),this._viewBox=[a,b,c,d,!!e],this._viewBoxShift={dx:-a,dy:-b,scale:k},this.forEach(function(a){a.transform("...")}),this};var G;c._engine.initWin=function(a){var d=a.document;d.createStyleSheet().addRule(".rvml","behavior:url(#default#VML)");try{!d.namespaces.rvml&&d.namespaces.add("rvml","urn:schemas-microsoft-com:vml"),G=c._createNode=function(a,c){var e,f=d.createElement("');for(e in c)f[e]=b(c[e]);return f}}catch(e){G=c._createNode=function(a,c){var e,f=d.createElement("<"+a+' xmlns="urn:schemas-microsoft.com:vml" class="rvml">');for(e in c)f[e]=b(c[e]);return f}}},c._engine.initWin(c._g.win),c._engine.create=function(){var a=c._getContainer.apply(0,arguments),b=a.container,d=a.height,e=a.width,f=a.x,g=a.y;if(!b)throw new Error("VML container not found.");var h=new c._Paper,i=h.canvas=c._g.doc.createElement("div"),j=i.style;return f=f||0,g=g||0,e=e||512,d=d||342,h.width=e,h.height=d,e==+e&&(e+="px"),d==+d&&(d+="px"),h.coordsize=1e3*v+o+1e3*v,h.coordorigin="0 0",h.span=c._g.doc.createElement("span"),h.span.style.cssText="position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;",i.appendChild(h.span),j.cssText=c.format("top:0;left:0;width:{0};height:{1};display:inline-block;cursor:default;position:relative;clip:rect(0 {0} {1} 0);overflow:hidden",e,d),1==b?(c._g.doc.body.appendChild(i),j.left=f+"px",j.top=g+"px",j.position="absolute"):b.firstChild?b.insertBefore(i,b.firstChild):b.appendChild(i),h.renderfix=function(){},h},c.prototype.clear=function(){m("raphael.clear",this),this.canvas.innerHTML=p,this.span=c._g.doc.createElement("span"),this.span.style.cssText="position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;display:inline;",this.canvas.appendChild(this.span),this.bottom=this.top=null},c.prototype.remove=function(){m("raphael.remove",this),this.canvas.parentNode.removeChild(this.canvas);for(var a in this)this[a]="function"==typeof this[a]?c._removedFactory(a):null;return!0};var H=c.st;for(var I in F)F[a](I)&&!H[a](I)&&(H[I]=function(a){return function(){var b=arguments;return this.forEach(function(c){c[a].apply(c,b)})}}(I))}}(),function(){if(c.canvas){var a,b,d,e=c._g.win,f=c._g.doc,g=c._g,h="string",i="px",j=/[, ]+/,k=e.String,l=e.parseInt,m=e.parseFloat,n=e.Math,o=n.max,p=n.min,q=n.PI,r=(n.floor,c.eve),s=c.fn,t=c.el,u=c.st,v=c.clone,w=q/180,x="hasOwnProperty",y=" ",z="ontouchstart"in e||navigator.msMaxTouchPoints>0,A=("click dblclick mousedown mousemove mouseout mouseover mouseup touchstart touchmove touchend touchcancel".split(y),!1),B=[],C=[],D=function(a){for(var b,c=a.clientX,d=a.clientY,e=g.doc.documentElement.scrollTop||g.doc.body.scrollTop,f=g.doc.documentElement.scrollLeft||g.doc.body.scrollLeft,h=C.length;h--;){if(b=C[h],z){for(var i,j=a.touches.length;j--;)if(i=a.touches[j],i.identifier==b.el._drag.id){c=i.clientX,d=i.clientY,(a.originalEvent?a.originalEvent:a).preventDefault();break}}else a.preventDefault();c+=f,d+=e,r("raphael.drag.move."+b.el.id,b.move_scope||b.el,c-b.el._drag.x,d-b.el._drag.y,c,d,a)}},E=function(a){c.unmousemove(D).unmouseup(E);for(var b,d=C.length;d--;)b=C[d],b.el._drag={},r("raphael.drag.end."+b.el.id,b.end_scope||b.start_scope||b.move_scope||b.el,a);C=[]};if(c.canvas){a=c._createNode=function(b,c){if(c){typeof b===h&&(b=a(b));for(var d in c)c.hasOwnProperty(d)&&b.setAttribute(d,k(c[d]))}else b=f.createElement(b);return b},c._getConnectedNodes=function(){return{above:[],below:[]}},c._getTargetNode=function(a){a[0],a[1]},c._containerEventHandler=function(a){if(a=a||e.event,!A){var b=a.offsetX,d=a.offsetY;a.type,c._getTargetNode([b,d])}},b=function(a){this.type="basic",this.owner=a,this._rElement=null,this.mouseInteractions=!1,this.matrix=null,this.outlinePath=null,this.conf={}},b.prototype={constructor:b,render:function(){var a=this;return a.draw(),a.setBBox(),a},draw:function(){var a,b=this,c=b.context,d=b._rElement,e=d.matrix,f=b.isClipped,g=b.validateAttrs();d.attrs=g,c.save(),c.fillStyle=g.fill,c.strokeStyle=g.stroke,c.lineWidth=g["stroke-width"],(a=g["clip-rect"])&&(a=a.split(" "),c.rect(a[0],a[1],a[2],a[3]),c.clip(),f=b.isClipped=!0),b.applyTransform(e),b.paint(),c.restore()},paint:function(){},redraw:function(){this.COMInstance.redraw(this)},clear:function(){var a=this,b=a.context,c=a._bbox;c&&b.clearRect(c.x,c.y,c.width,c.height)},addMouseInteractivity:function(){var b,c=this,d=c._rElement.attrs,e=c._bbox,f=c.owner.wrapper._map,g="circle"===c.type?"circle":"rect",h="circle"===g?[d.cx,d.cy,d.r].join(","):[e.x,e.y,e.x2,e.y2].join(",");b=a("area",{shape:g,coords:h}),f.firstChild?f.insertBefore(b,f.firstChild):f.appendChild(b),c._mouseArea=b,c.eventListeners={}},updateMapAreaCoords:function(){var a=this,b=a._mouseArea,c=a._bbox;if(b)if(a instanceof K){var d=c.width/2;b.setAttribute("coords",[c.x+d,c.y+d,d].join(","))}else b.setAttribute("coords",[c.x,c.y,c.x2,c.y2].join(","))},applyTransform:function(a){var b,c=this,d=c.context;a&&(b=a.split(),d.translate(b.dx,b.dy),!b.noRotation&&d.rotate(w*b.rotate),d.scale(b.scalex,b.scaley))},setBBox:function(){var a=this,b=a._rElement,d=b.matrix,e=a.owner,f=e.getTransformMatrix&&e.getTransformMatrix();f&&(f=f.clone(),f.add(d),d=f),a.outlinePath?a._bbox=c.pathBBox(c.transformPath(a.outlinePath,d.toTransformString()).toString()):a.setShapeBBox(d),a._mouseArea&&a.updateMapAreaCoords()},getBBox:function(){return this._bbox},drawPath:function(a){var b=this,c=b.context,d=(a&&a.length||0,PathParser);for(d.reset(),d.setTokens(a),null!=c&&c.beginPath();!d.isEnd();)switch(d.nextCommand(),d.command){case"M":case"m":var e=d.getAsCurrentPoint();for(d.addMarker(e),null!=c&&c.moveTo(e.x,e.y),d.start=d.current;!d.isCommandOrEnd();){var e=d.getAsCurrentPoint();d.addMarker(e,d.start),null!=c&&c.lineTo(e.x,e.y)}break;case"L":case"l":for(;!d.isCommandOrEnd();){var f=d.current,e=d.getAsCurrentPoint();d.addMarker(e,f),null!=c&&c.lineTo(e.x,e.y)}break;case"H":case"h":for(;!d.isCommandOrEnd();){var g=new Point((d.isRelativeCommand()?d.current.x:0)+d.getScalar(),d.current.y);d.addMarker(g,d.current),d.current=g,null!=c&&c.lineTo(d.current.x,d.current.y)}break;case"V":case"v":for(;!d.isCommandOrEnd();){var g=new Point(d.current.x,(d.isRelativeCommand()?d.current.y:0)+d.getScalar());d.addMarker(g,d.current),d.current=g,null!=c&&c.lineTo(d.current.x,d.current.y)}break;case"C":case"c":for(;!d.isCommandOrEnd();){var h=d.current,i=d.getPoint(),j=d.getAsControlPoint(),k=d.getAsCurrentPoint();d.addMarker(k,j,i),null!=c&&c.bezierCurveTo(i.x,i.y,j.x,j.y,k.x,k.y)}break;case"S":case"s":for(;!d.isCommandOrEnd();){var h=d.current,i=d.getReflectedControlPoint(),j=d.getAsControlPoint(),k=d.getAsCurrentPoint();d.addMarker(k,j,i),null!=c&&c.bezierCurveTo(i.x,i.y,j.x,j.y,k.x,k.y)}break;case"Q":case"q":for(;!d.isCommandOrEnd();){var h=d.current,j=d.getAsControlPoint(),k=d.getAsCurrentPoint();d.addMarker(k,j,j),null!=c&&c.quadraticCurveTo(j.x,j.y,k.x,k.y)}break;case"T":case"t":for(;!d.isCommandOrEnd();){var h=d.current,j=d.getReflectedControlPoint();d.control=j;var k=d.getAsCurrentPoint();d.addMarker(k,j,j),null!=c&&c.quadraticCurveTo(j.x,j.y,k.x,k.y)}break;case"A":case"a":for(;!d.isCommandOrEnd();){var h=d.current,l=d.getScalar(),m=d.getScalar(),n=d.getScalar()*(Math.PI/180),o=d.getScalar(),p=d.getScalar(),k=d.getAsCurrentPoint(),q=new Point(Math.cos(n)*(h.x-k.x)/2+Math.sin(n)*(h.y-k.y)/2,-Math.sin(n)*(h.x-k.x)/2+Math.cos(n)*(h.y-k.y)/2),r=Math.pow(q.x,2)/Math.pow(l,2)+Math.pow(q.y,2)/Math.pow(m,2);r>1&&(l*=Math.sqrt(r),m*=Math.sqrt(r));var s=(o==p?-1:1)*Math.sqrt((Math.pow(l,2)*Math.pow(m,2)-Math.pow(l,2)*Math.pow(q.y,2)-Math.pow(m,2)*Math.pow(q.x,2))/(Math.pow(l,2)*Math.pow(q.y,2)+Math.pow(m,2)*Math.pow(q.x,2)));isNaN(s)&&(s=0);var t=new Point(s*l*q.y/m,s*-m*q.x/l),u=new Point((h.x+k.x)/2+Math.cos(n)*t.x-Math.sin(n)*t.y,(h.y+k.y)/2+Math.sin(n)*t.x+Math.cos(n)*t.y),v=function(a){return Math.sqrt(Math.pow(a[0],2)+Math.pow(a[1],2))},w=function(a,b){return(a[0]*b[0]+a[1]*b[1])/(v(a)*v(b))},x=function(a,b){return(a[0]*b[1]=1&&(B=0);var C=1-p?1:-1,D=y+C*(B/2),E=new Point(u.x+l*Math.cos(D),u.y+m*Math.sin(D));if(d.addMarkerAngle(E,D-C*Math.PI/2),d.addMarkerAngle(k,D-C*Math.PI),null!=c){var w=l>m?l:m,F=l>m?1:l/m,G=l>m?m/l:1;c.translate(u.x,u.y),c.rotate(n),c.scale(F,G),c.arc(0,0,w,y,y+B,1-p),c.scale(1/F,1/G),c.rotate(-n),c.translate(-u.x,-u.y)}}break;case"Z":case"z":null!=c&&c.closePath(),d.current=d.start}return b.outlinePath=a,b},addEventListener:function(){var a,b,d=this,e=arguments,f=e&&e[0],g=e&&e[1];if(d._mouseArea||d.addMouseInteractivity(),a=d._mouseArea,"string"==typeof f&&"function"==typeof g)if(d._path)if("mouseover"===f||"mouseout"===f||"mousemove"===f){if(!d._mousemoveAdded){var h=function(a){var b=!1,d=!0,e=!1;return function(f){var g=f.layerX,h=f.layerY;e=!1,c.isPointInsidePath(a._transformPath,g,h)?(b=!0,d&&(d=!1,e=!0)):(d=!0,b&&(b=!1,e=!0)),d&&e&&a.eventListeners.mouseout&&a.eventListeners.mouseout.apply(this,arguments),b&&(e&&a.eventListeners.mouseover&&a.eventListeners.mouseover.apply(this,arguments),a.eventListeners.mousemove&&a.eventListeners.mousemove.apply(this,arguments))}}(d);a.addEventListener("mousemove",h,!1),d._mousemoveAdded=!0}d.eventListeners[f]=g}else{var b=function(a,b){return function(d){c.isPointInsidePath(a._path,d.layerX,d.layerY)&&b.apply(this,arguments)}}(d,g);a.addEventListener(f,b,!1)}else a.addEventListener(f,g,!1)},removeEventListener:function(){var a,b=this,c=arguments,d=c&&c[0],e=c&&c[1];b._mouseArea&&(a=b._mouseArea,"string"==typeof d&&"function"==typeof e&&a.removeEventListener(d,e))},attachEvent:function(){},detachEvent:function(){},validateAttrs:function(a){var b,c,d=this,e=v(d._rElement.attrs);if(null===a){if(d._isValid)return e;d._isValid=!0}a=a||e;for(b in a)switch(c=a[b],b){default:continue}return a},attrs:function(){}};var F=function(a){this.node=a,this.next=null,this.prev=null},G=function(){this.top=null,this.bottom=null};G.prototype={constructor:G,add:function(a){a=new F(a),this.bottom||(this.bottom=a),this.top&&(this.top.next=a),a.next=null,a.prev=this.top,this.top=a},addList:function(a){this.bottom||(this.bottom=a.bottom),this.top&&(this.top.next=a.bottom,a.bottom.prev=this.top),this.top=a.top},toFront:function(a){return this.top===a?!1:(this.bottom===a&&(this.bottom=a.next),a.prev&&(a.prev.next=a.next),a.next&&(a.next.prev=a.prev),this.top.next=a,a.prev=this.top,a.next=null,this.top=a,void 0)},toBack:function(a){return this.bottom===a?!1:(this.top===a&&(this.top=a.prev),a.prev&&(a.prev.next=a.next),a.next&&(a.next.prev=a.prev),this.bottom.prev=a,a.prev=null,a.next=this.bottom,this.bottom=a,void 0)},insertBefore:function(){},insertAfter:function(){},each:function(a,b){for(var c=this.bottom;c;)a.apply(c.node,b),c=c.next},iterate:function(a,b){for(var c=this.bottom,d=!0;c&&(d=a.apply(c.node,b),d!==!1);)c=c.next},dispose:function(){this.each(function(){this.node.dispose&&this.node.dispose()}),this.top=null,this.bottom=null}};var H=function(a,b){this.items=new G,this.owner=a,this.element=null,b?this.element=b:this.init()};H.prototype={constructor:H,appendChild:function(){var a=this,b=a.owner.wrapper,c=this.element;b._image?b.insertBefore(c,b._image):b.appendChild(c)},insertBefore:function(){},insertAfter:function(){},init:function(){this.element=a("canvas"),a(this.element,{width:this.owner.wrapper.offsetWidth,height:this.owner.wrapper.offsetHeight}),this.element.style.cssText="position:absolute;left:0;top:0;",this.appendChild()},getCanvas:function(){return this.element},getContext:function(){return this.element.getContext("2d")},addToLayer:function(a){this.items.add(a)},mergeWithLayerOnTop:function(a){this.items.addList(a.items),a.dispose(!0)},mergeWithLayerOnBottom:function(a){a.items.addList(this.items),this.items=a.items,a.dispose(!0)},dispose:function(a){a||this.items.each(function(){this.dispose()}),this.items=null,this.owner=null,this.element.parentNode.removeChild(this.element),this.element=null}};var I=function(a,b,c){this.nodeItems=new G,this.collectionItems=new G,this.layerItems=new G,this.owner=this.parent=a,this.layerOnTop=null,this.currentLayer=null,this.baseLayer=null,b?(this.wrapper=b,this.currentLayer=this.baseLayer=new H(this,c)):this.init()};I.prototype={constructor:I,init:function(){var b=this,c=b.parent,d=c.wrapper._image,e=a("div");e.style.cssText="width:100%;height:100%;position:absolute;left:0;top:0;",e._map=c.wrapper._map,d?c.wrapper.insertBefore(e,d):c.wrapper.appendChild(e),b.wrapper=e,b.currentLayer=b.baseLayer=new H(b)},getCurrentContext:function(){return this.currentLayer.getContext()},setLayerOnTop:function(a){this.layerOnTop=a},getCurrentCanvas:function(){return this.currentLayer.getCanvas()},addNode:function(a){this.nodeItems.add(a),"group"===a.type?this.addCollection(a):this.currentLayer.addToLayer(a)},addCollection:function(a){a=a||new I(this),this.collectionItems.add(a),this.currentLayer=new H(this),this.layerItems.add(this.currentLayer),a.setLayerOnTop(this.currentLayer)},dispose:function(){this.nodeItems.dispose(),this.collectionItems.dispose(),this.layerItems.dispose(),this.owner=this.parent=null,this.ownerLayer=null,this.currentLayer=null,this.baseLayer=null}};var J=function(a,c,d,e){var f=this,g=new I(null,c,a);f.width=d,f.height=e,f.createNode=function(a,c){c=c||g;var d,e=c.nodeItems,f=c.currentLayer,h=f.getCanvas();switch(a){case"rect":d=new RectFauxNode(c);break;case"circle":d=new K(c);break;case"path":d=new L(c);break;case"text":d=new M(c);break;case"group":d=new N(c),c.addCollection(d);break;default:d=new b(h)}return d.COMInstance=this,e.add(d),f.addToLayer(d),d},f.redraw=function(a){var b,c,d;if("group"===a.type)b=a.nodeItems,a.render();else for(d=a.layer,b=d.items,c=b.bottom,d.element.width=d.element.width;c;)fNode=c.node,"group"!==fNode.type&&fNode.render(),c=c.next},f.insertBefore=function(){},f.insertAfter=function(){},f.removeNode=function(){},f.refreshNode=function(){},f.refreshAll=function(){}};Point=function(a,b){this.x=a,this.y=b},Point.prototype.angleTo=function(a){return Math.atan2(a.y-this.y,a.x-this.x)},Point.prototype.applyTransform=function(a){var b=this.x*a[0]+this.y*a[2]+a[4],c=this.x*a[1]+this.y*a[3]+a[5];this.x=b,this.y=c},PathParser=new function(){this.tokens=null,this.setTokens=function(a){this.tokens="string"==typeof a?a.split(" "):a},this.reset=function(){this.i=-1,this.command="",this.previousCommand="",this.start=new Point(0,0),this.control=new Point(0,0),this.current=new Point(0,0),this.points=[],this.angles=[]},this.isEnd=function(){return this.i>=this.tokens.length-1},this.isCommandOrEnd=function(){return this.isEnd()?!0:null!=this.tokens[this.i+1].toString().match(/^[A-Za-z]$/)},this.isRelativeCommand=function(){switch(this.command){case"m":case"l":case"h":case"v":case"c":case"s":case"q":case"t":case"a":case"z":return!0}return!1},this.getToken=function(){return this.i++,this.tokens[this.i]},this.getScalar=function(){return parseFloat(this.getToken())},this.nextCommand=function(){this.previousCommand=this.command,this.command=this.getToken()},this.getPoint=function(){var a=new Point(this.getScalar(),this.getScalar());return this.makeAbsolute(a)},this.getAsControlPoint=function(){var a=this.getPoint();return this.control=a,a},this.getAsCurrentPoint=function(){var a=this.getPoint();return this.current=a,a},this.getReflectedControlPoint=function(){if("c"!=this.previousCommand.toLowerCase()&&"s"!=this.previousCommand.toLowerCase()&&"q"!=this.previousCommand.toLowerCase()&&"t"!=this.previousCommand.toLowerCase())return this.current;var a=new Point(2*this.current.x-this.control.x,2*this.current.y-this.control.y);return a},this.makeAbsolute=function(a){return this.isRelativeCommand()&&(a.x+=this.current.x,a.y+=this.current.y),a},this.addMarker=function(a,b,c){null!=c&&this.angles.length>0&&null==this.angles[this.angles.length-1]&&(this.angles[this.angles.length-1]=this.points[this.points.length-1].angleTo(c)),this.addMarkerAngle(a,null==b?null:b.angleTo(a))},this.addMarkerAngle=function(a,b){this.points.push(a),this.angles.push(b)},this.getMarkerPoints=function(){return this.points},this.getMarkerAngles=function(){for(var a=0;ah&&(a.r=h),0>c&&(a.r=0);break;case"width":case"height":0>c&&(a[b]=0);break;default:continue}return a},setShapeBBox:function(a){var b=this,c=b._rElement,d=c.attrs,e=a.get(0),f=a.get(3),g=a.get(4),h=a.get(5),i=d["stroke-width"];b._bbox={x:d.x*e+g-i,y:d.y*f+h-i,width:d.width*e+2*i,height:d.height*f+2*i},b._bbox.x2=b._bbox.x+b._bbox.width,b._bbox.y2=b._bbox.y+b._bbox.height,b.X=b._bbox.x,b.Y=b._bbox.y,b.W=b._bbox.width,b.H=b._bbox.height}});var K=function(a){this.type="circle",this._isValid=!1,this.parent=this.owner=a,this.context=a.getCurrentContext(),this.layer=a.currentLayer},L=function(a){this.type="path",this._isValid=!1,this.parent=this.owner=a,this.context=a.getCurrentContext(),this.layer=a.currentLayer},M=function(a){this.type="text",this._isValid=!1,this.parent=this.owner=a,this.context=a.getCurrentContext(),this.layer=a.currentLayer},N=function(a){this.type="group",this.nodeItems=new G,this.collectionItems=new G,this.layerItems=new G,this.owner=this.parent=a,this.layerOnTop=a.currentLayer,this.currentLayer=null,this.baseLayer=null,this.init()};K.prototype=c.extend(new b,{constructor:K,paint:function(){var a=this,b=a.context,c=a.validateAttrs(),d=c.cx,e=c.cy,f=c.r,g=f||c.rx,h=f||c.ry;if(c.r){if(a.drawPath(["M",d+f,e,"A",g,h,0,1,0,d-f,e,"A",g,h,0,1,0,d+f,e,"Z"]),c["stroke-width"]){var i=void 0===c["stroke-opacity"]?c.opacity:c["stroke-opacity"];void 0!==i&&(b.globalAlpha=i),b.stroke()}var j=void 0===c["fill-opacity"]?c.opacity:c["fill-opacity"];void 0!==j&&(b.globalAlpha=j),b.fill()}},setShapeBBox:function(a){var b=this,c=b._rElement,d=c.attrs,e=a.get(0),f=a.get(3),g=a.get(4),h=a.get(5),i=d["stroke-width"];b._bbox={x:g+(d.cx-d.r)*e-i,y:h+(d.cy-d.r)*f-i,width:2*(i+d.r*e),height:2*(d.r*f+i)},b._bbox.x2=b._bbox.x+b._bbox.width,b._bbox.y2=b._bbox.y+b._bbox.height,b.X=b._bbox.x,b.Y=b._bbox.y,b.W=b._bbox.width,b.H=b._bbox.height}}),L.prototype=c.extend(new b,{constructor:L,paint:function(){var a=this,b=a._rElement,d=b.attrs,e=b.attr("path"),f=b.matrix,g=a.context;a.drawPath(e),a._transformPath=c.transformPath(e,f.toTransformString());var h=void 0===d["stroke-opacity"]?d.opacity:d["stroke-opacity"];void 0!==h&&(g.globalAlpha=h),g.stroke();var i=void 0===d["fill-opacity"]?d.opacity:d["fill-opacity"];void 0!==i&&(g.globalAlpha=i),g.fill()}}),M.prototype=c.extend(new b,{constructor:M,paint:function(){var a=this,b=a._rElement,c=b.attr(),d=c.text,e=c.stroke,f=c["vertical-align"],g=c["text-anchor"],h=c.x,i=c.y,j=(b.matrix,a.context),m=c["font-size"]||10,n=c["line-height"]||1.2*l(m,10),q=["normal",m,c.font];if(j.fillStyle=e,j.font=q.join(" "),d){var r,s,t,u,v=k(d).split(/\n| /gi),w=v.length*n,x=-1/0,y=1/0;r="top"===f?i+n:"middle"===f?i-w/2+n/2:i-w+n;for(var z=0,A=v.length;A>z;z+=1)d=v[z],u=r+n*z,s=j.measureText(d).width,t="start"===g?h:"middle"===g?h-s/2:h-s,x=o(x,s),y=p(y,t),j.fillText(d,t,u);b._textdirty=!1}a.outlinePath=["M",y,r-n/1.4,"H",y+x,"V",r-n+w,"H",y,"V",r-n/1.4]}}),N.prototype=c.extend(c.extend(new b,I.prototype),{constructor:N,draw:function(){this.layerItems.each(function(){this.element.width=this.element.width}),b.prototype.draw.apply(this,arguments)},render:function(){var a=this;return a.draw(),a.setBBox(),a},paint:function(){var a=this,b=a.nodeList,c=a._rElement,d=(a.canvas,c.attrs),e=b.bottom;for(void 0!==d.opacity&&this.layerItems.each(function(){this.getContext().globalAlpha=d.opacity});e;)e.render(),e=e.next},setBBox:function(){},addMouseInteractivity:function(){},applyTransform:function(a){var c=this,d=c.parent,e=d.getTransformMatrix&&d.getTransformMatrix();e?(c.matrixApplied=e.clone(),c.matrixApplied.add(a.a,a.b,a.c,a.d,a.e,a.f)):c.matrixApplied=a,this.layerItems.each(function(){b.prototype.applyTransform.apply(this,[c.matrixApplied])})},getTransformMatrix:function(){return this.matrixApplied}}),d=function(a,b,d){var e=this,f=d||b;e.node=e[0]=a,a.raphael=!0,a.raphaelid=e.id=c._oid++,a._rElement=e,e.X=0,e.Y=0,e.attrs=e.attrs||{},e.styles=e.styles||{},e.followers=e.followers||[],e.paper=b,e.com=f.com,e.ca=e.customAttributes=e.customAttributes||new b._CustomAttributes,e.matrix=c.matrix(),e._={transform:[],sx:1,sy:1,dx:0,dy:0,deg:0},e.parent=f,!f.bottom&&(f.bottom=e),e.prev=f.top||null,f.top&&(f.top.next=e),f.top=e,e.next=null},d.prototype=t,t.constructor=d;var O=function(a,b){var d,e=S(a),f=c._getConnectedNodes(e),g=a.attrs;f.above,f.below;for(d in b)g[d]=b[d];e.redraw()},P=function(a,b){var d,e,f=(a.attrs,a.node),g={},h=!1,i=!1,j=!1;for(d in b)if(b[x](d)){if(!c._availableAttrs[x](d))continue;switch(e=b[d],d){case"fill-opacity":case"opacity":case"stroke-opcaity":case"stroke":case"fill":g[d]=e,h=!0;break;case"stroke-width":case"cx":case"cy":case"x":case"y":g[d]=e,i=!0;break;case"width":case"height":g[d]=e,j=!0;break;case"clip-rect":g[d]=e,h=!0;break;case"font-size":case"font":case"vertical-align":case"text-anchor":g[d]=e,h=!0;default:continue}}R(a,b,g),g=f.validateAttrs(g),(h||i||j)&&O(a,g,i,j)},Q=1.2,R=function(a,b,c){if("text"==a.type&&(b[x]("text")||b[x]("font")||b[x]("font-size")||b[x]("x")||b[x]("y")||b[x]("line-height")||b[x]("vertical-align"))){var d=a.attr(),e=b["font-size"]||d["font-size"]||10,f=m(b["line-height"]||d["line-height"])||l(e,10)*Q,g=b["vertical-align"]||d["vertical-align"]||"middle";isNaN(f)&&(f=e*Q),c["font-size"]=l(e,10)+"px",c.font=b.font||d.font||"Verdana",c["vertical-align"]=g,c.x=b.x||d.x||0,c.y=b.y||d.y||0,c["line-height"]=l(f,10),c["text-anchor"]=b["text-anchor"]||d["text-anchor"]||"middle"}};c._engine.initWin=function(a){a=a,f=a.document},c._engine.setSize=function(a,b){var c=this,d=c.canvas.style;return d.width=(c.width=+a||c.width)+i,d.height=(c.height=+b||c.height)+i,c},c._engine.create=function(){var b,d,e,g,h,i,j=c._getContainer.apply(0,arguments)||{},k=j.container,l=j.x,m=j.y,n=j.width,o=j.height;if(!k)throw new Error("Canvas container not found.");return h=new c._Paper,h.canvas=b=a("div"),l=l||0,m=m||0,h.width=n=n||512,h.height=o=o||342,h.left=h.top=0,1==k?(b.style.cssText=d+c.format(";width:100%;height:100%;position:absolute;left:{0}px;top:{1}px;",[l,m]),f.body.appendChild(b)):(b.style.cssText=d+";width:100%;height:100%;position:absolute",k.firstChild?k.insertBefore(b,k.firstChild):k.appendChild(b)),d="overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);-webkit-user-select:none;-moz-user-select:-moz-none;-khtml-user-select:none;-ms-user-select:none;user-select:none;-o-user-select:none;cursor:default;"+c.format("width:{0}px;height:{1}px;",[n,o]),i=a("canvas"),i.style.cssText="position:absolute;left:0;top:0",i.setAttribute("width",h.width),i.setAttribute("height",h.height),h.com=new J(i,b,h.width,h.height),b.appendChild(i),e=a("img"),e.src="image1.png",e.style.cssText="opacity: 0;z-index: 100;background: transparent;position: absolute;left: 0;top: 0;width: "+n+"px;height: "+o+"px",b.appendChild(e),g=a("map"),g.setAttribute("name","mousemap"),g.setAttribute("id","mousemap"),b.appendChild(g),e.setAttribute("usemap","#mousemap"),b._image=e,b._map=g,h};var S=c._engine.getNode=function(a){return a.node||a[0].node};c._engine.getLastNode=function(a){return a.node||a[a.length-1].node},c._engine.rect=function(a,b,c,e,f,g,h){var i=a.com.createNode("rect",h&&h.node),j=new d(i,a,h),k=j.attrs;return k.x=b,k.y=c,k.width=e,k.height=f,k.fill="#fff",k.stroke="#000",k["stroke-width"]=1,k.r=g||0,k.rx=g||0,k.ry=g||0,j.type="rect",i.render(),j},c._engine.circle=function(a,b,c,e,f){var g=a.com.createNode("circle",f&&f.node),h=new d(g,a,f),i=h.attrs;return i.cx=b,i.cy=c,i.r=e,i.fill="none",i.stroke="#000",i["stroke-width"]=1,h.type="circle",g.render(),h},c._engine.ellipse=function(a,c,e,f,g,h){var i=new b,j=new d(i,a,h);return j.type="ellipse",j},c._engine.image=function(a,c,e,f,g,h,i){var j=new b,k=new d(j,a,i);return k.type="image",k},c._engine.text=function(a,b,c,e,f){var g=a.com.createNode("text",f&&f.node),h=new d(g,a,f),i=h.attrs;return i.x=b,i.y=c,i.text=e,i.fill="none",i.stroke="#000",i.font="Verdana",i["font-size"]="12px",i["vertical-align"]="middle",i["text-anchor"]="middle",h.type="text",g.render(),h},c._engine.path=function(a,b,c){var e=b.com.createNode("path",c&&c.node),f=new d(e,b,c),g=f.attrs;return g.path=a,g.fill="#fff",g.stroke="#000",g["stroke-width"]=1,f.type="path",e.render(),f},c._engine.group=function(a,b,c){var e=a.com.createNode("group",c&&c.node),f=new d(e,a,c),g=e.wrapper;return b&&g.setAttribute("class",["red",b].join("-")),f.canvas=g,f.type="group",f},t._getBBox=function(){return this.removed?{}:{x:this.X+(this.bbx||0)-this.W/2,y:this.Y+(this.bby||0)-this.H/2,width:this.W,height:this.H}},t.toFront=function(){if(this.removed)return this;var a,b,d,e=this,f=e.node,g=e.parent,h=f.owner,i=e.followers;for(c._tofront(e,g)&&("group"===t.type?g.canvas.appendChild(f):h.nodeList.tofront(f)),b=0,d=i.length;d>b;b++)(a=i[b]).stalk&&a.el[a.stalk](e);return e},t.toBack=function(){if(this.removed)return this;var a,b,d,e=this,f=e.node,g=e.parent,h=f.owner,i=e.followers;for(c._toback(e,g)&&("group"===t.type?g.canvas.appendChild(f):h.nodeList.toback(f)),b=0,d=i.length;d>b;b++)(a=i[b]).stalk&&a.el[a.stalk](e);return e},t.insertAfter=function(a){if(this.removed)return this;var b,d,e,f=this,g=f.node,h=a.node,i=h.owner,j=f.followers;for(h.next?i.nodeList.insertBefore(g,h.next):i.appendChild(g),c._insertafter(f,a,f.parent,a.parent),d=0,e=j.length;e>d;d++)(b=j[d]).stalk&&b.el[b.stalk](a);return f},t.insertBefore=function(a){if(this.removed)return this;var b,d,e,f=this,g=f.node,h=a.node,i=h.owner,j=f.followers;for(h?i.nodeList.insertBefore(g,h):i.appendChild(g),c._insertafter(f,a,f.parent,a.parent),d=0,e=j.length;e>d;d++)(b=j[d]).stalk&&b.el[b.stalk](a);return f},t.appendChild=function(){return this},t.removeChild=function(){return this},t.attr=function(a,b){if(this.removed)return this;var d,e,f,g,h,i,k,l,m,n,o,p=this,q=p.attrs,s=p.ca;if(null==a){g={};for(h in q)q.hasOwnProperty(h)&&(g[h]=q[h]);return g.gradient&&"none"==g.fill&&(g.fill=g.gradient)&&delete g.gradient,g.transform=p._.transform,g}if(null==b&&c.is(a,"string")){if("fill"==a&&"none"==q.fill&&q.gradient)return q.gradient;if("transform"==a)return p._.transform;for(d=a.split(j),i={},o=0,n=d.length;n>o;o++)a=d[o],i[a]=a in q?q[a]:c.is(s[a],"function")?s[a].def:c._availableAttrs[a];return n-1?i:i[d[0]]}if(null==b&&c.is(a,"array")){for(i={},o=0,n=a.length;n>o;o++)i[a[o]]=p.attr(a[o]);return i}null!=b?(e={},e[a]=b):null!=a&&c.is(a,"object")&&(e=a);for(h in e)r("raphael.attr."+h+"."+p.id,p,e[h],h);l={};for(h in s)if(s[h]&&e.hasOwnProperty(h)&&c.is(s[h],"function")&&!s["_invoked"+h]){s["_invoked"+h]=!0,f=s[h].apply(p,[].concat(e[h])),delete s["_invoked"+h];for(k in f)f.hasOwnProperty(k)&&(e[k]=f[k]);q[h]=e[h],f===!1&&(l[h]=e[h],delete e[h])}for(P(this,e),o=0,n=p.followers.length;n>o;o++)m=p.followers[o],m.cb&&!m.cb.call(m.el,e,p)||m.el.attr(e);
+for(k in l)e[k]=l[k];return this},t.css=function(){return this},t.drag=function(a,b,d,e,f,h){function i(i){(i.originalEvent||i).preventDefault();var j=g.doc.documentElement.scrollTop||g.doc.body.scrollTop,k=g.doc.documentElement.scrollLeft||g.doc.body.scrollLeft;this._drag.x=i.clientX+k,this._drag.y=i.clientY+j,this._drag.id=i.identifier,!C.length&&c.mousemove(D).mouseup(E),C.push({el:this,move_scope:e,start_scope:f,end_scope:h}),b&&r.on("raphael.drag.start."+this.id,b),a&&r.on("raphael.drag.move."+this.id,a),d&&r.on("raphael.drag.end."+this.id,d),r("raphael.drag.start."+this.id,f||e||this,i.clientX+k,i.clientY+j,i)}return this._drag={},B.push({el:this,start:i}),this.mousedown(i),this},t.undrag=function(){for(var a=B.length;a--;)B[a].el==this&&(this.unmousedown(B[a].start),B.splice(a,1),r.unbind("raphael.drag.*."+this.id));!B.length&&c.unmousemove(D).unmouseup(E)},t.rotate=function(a,b,c){var d,e=this;return e.removed?e:(a=k(a).split(j),a.length-1&&(b=m(a[1]),c=m(a[2])),a=m(a[0]),null==c&&(b=c),(null==b||null==c)&&(d=e.getBBox(1),b=d.x+d.width/2,c=d.y+d.height/2),e.transform(e._.transform.concat([["r",a,b,c]])),e)},t.scale=function(a,b,c,d){var e,f=this;return f.removed?f:(a=k(a).split(j),a.length-1&&(b=m(a[1]),c=m(a[2]),d=m(a[3])),a=m(a[0]),null==b&&(b=a),null==d&&(c=d),(null==c||null==d)&&(e=f.getBBox(1)),c=null==c?e.x+e.width/2:c,d=null==d?e.y+e.height/2:d,f.transform(f._.transform.concat([["s",a,b,c,d]])),f)},t.translate=function(a,b){var c=this;return c.removed?c:(a=k(a).split(j),a.length-1&&(b=m(a[1])),a=m(a[0])||0,b=+b||0,c.transform(c._.transform.concat([["t",a,b]])),c)},t.transform=function(a){var b,d=this,e=d._;return null===a?e.transform:(c._extractTransform(d,a),(1!=e.sx||1!=e.sy)&&(b=d.attrs[x]("stroke-width")?d.attrs["stroke-width"]:1,d.attr({"stroke-width":b})),d.node&&d.node.redraw(),d)},t.hide=function(){return this},t.show=function(){return this},t.blur=function(){return this},t.on=function(a,b){var c=this,d=c.listeners;d||(d=c.listeners={}),d[a]||(d[a]=[]),d[a].push(b)},t.remove=function(){return this},s.clear=function(){return r("raphael.clear",this),this},s.remove=function(){if(!this.removed){var a,b=this,d=b.canvas,e=d.parentNode;r("raphael.remove",b),e.removeChild(d);for(a in b)b[a]="function"==typeof b[a]?c._removedFactory(a):null;this.removed=!0}},c.toString=function(){return"Your browser supports canvas.\nYou are running RedRaphael "+c.version};for(var T in t)t.hasOwnProperty(T)&&!u.hasOwnProperty(T)&&(u[T]=function(a){return function(){var b=arguments;return this.forEach(function(c){c[a].apply(c,b)})}}(T))}}}(),M.was?L.win.Raphael=c:Raphael=c,c});
\ No newline at end of file
diff --git a/package/raphael.js b/package/raphael.js
new file mode 100644
index 0000000..fd21694
--- /dev/null
+++ b/package/raphael.js
@@ -0,0 +1,12310 @@
+/**!
+ * RedRaphael 1.0.0 - JavaScript Vector Library
+ * Copyright (c) 2012-2013 FusionCharts Technologies
+ *
+ * Raphael 2.1.0
+ * Copyright (c) 2008-2012 Dmitry Baranovskiy
+ * Copyright © 2008-2012 Sencha Labs
+ *
+ * Licensed under the MIT license.
+ */
+// Copyright (c) 2013 Adobe Systems Incorporated. All rights reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+// ┌────────────────────────────────────────────────────────────┐ \\
+// │ Eve 0.4.2 - JavaScript Events Library │ \\
+// ├────────────────────────────────────────────────────────────┤ \\
+// │ Author Dmitry Baranovskiy (http://dmitry.baranovskiy.com/) │ \\
+// └────────────────────────────────────────────────────────────┘ \\
+
+(function (glob) {
+ var version = "0.4.2",
+ has = "hasOwnProperty",
+ separator = /[\.\/]/,
+ wildcard = "*",
+ fun = function () {},
+ numsort = function (a, b) {
+ return a - b;
+ },
+ current_event,
+ stop,
+ events = {n: {}},
+ /*\
+ * eve
+ [ method ]
+
+ * Fires event with given `name`, given scope and other parameters.
+
+ > Arguments
+
+ - name (string) name of the *event*, dot (`.`) or slash (`/`) separated
+ - scope (object) context for the event handlers
+ - varargs (...) the rest of arguments will be sent to event handlers
+
+ = (object) array of returned values from the listeners
+ \*/
+ eve = function (name, scope) {
+ name = String(name);
+ var e = events,
+ oldstop = stop,
+ args = Array.prototype.slice.call(arguments, 2),
+ listeners = eve.listeners(name),
+ z = 0,
+ f = false,
+ l,
+ indexed = [],
+ queue = {},
+ out = [],
+ ce = current_event,
+ errors = [];
+ current_event = name;
+ stop = 0;
+ for (var i = 0, ii = listeners.length; i < ii; i++) if ("zIndex" in listeners[i]) {
+ indexed.push(listeners[i].zIndex);
+ if (listeners[i].zIndex < 0) {
+ queue[listeners[i].zIndex] = listeners[i];
+ }
+ }
+ indexed.sort(numsort);
+ while (indexed[z] < 0) {
+ l = queue[indexed[z++]];
+ out.push(l.apply(scope, args));
+ if (stop) {
+ stop = oldstop;
+ return out;
+ }
+ }
+ for (i = 0; i < ii; i++) {
+ l = listeners[i];
+ if ("zIndex" in l) {
+ if (l.zIndex == indexed[z]) {
+ out.push(l.apply(scope, args));
+ if (stop) {
+ break;
+ }
+ do {
+ z++;
+ l = queue[indexed[z]];
+ l && out.push(l.apply(scope, args));
+ if (stop) {
+ break;
+ }
+ } while (l)
+ } else {
+ queue[l.zIndex] = l;
+ }
+ } else {
+ out.push(l.apply(scope, args));
+ if (stop) {
+ break;
+ }
+ }
+ }
+ stop = oldstop;
+ current_event = ce;
+ return out.length ? out : null;
+ };
+ // Undocumented. Debug only.
+ eve._events = events;
+ /*\
+ * eve.listeners
+ [ method ]
+
+ * Internal method which gives you array of all event handlers that will be triggered by the given `name`.
+
+ > Arguments
+
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated
+
+ = (array) array of event handlers
+ \*/
+ eve.listeners = function (name) {
+ var names = name.split(separator),
+ e = events,
+ item,
+ items,
+ k,
+ i,
+ ii,
+ j,
+ jj,
+ nes,
+ es = [e],
+ out = [];
+ for (i = 0, ii = names.length; i < ii; i++) {
+ nes = [];
+ for (j = 0, jj = es.length; j < jj; j++) {
+ e = es[j].n;
+ items = [e[names[i]], e[wildcard]];
+ k = 2;
+ while (k--) {
+ item = items[k];
+ if (item) {
+ nes.push(item);
+ out = out.concat(item.f || []);
+ }
+ }
+ }
+ es = nes;
+ }
+ return out;
+ };
+
+ /*\
+ * eve.on
+ [ method ]
+ **
+ * Binds given event handler with a given name. You can use wildcards “`*`” for the names:
+ | eve.on("*.under.*", f);
+ | eve("mouse.under.floor"); // triggers f
+ * Use @eve to trigger the listener.
+ **
+ > Arguments
+ **
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated, with optional wildcards
+ - f (function) event handler function
+ **
+ = (function) returned function accepts a single numeric parameter that represents z-index of the handler. It is an optional feature and only used when you need to ensure that some subset of handlers will be invoked in a given order, despite of the order of assignment.
+ > Example:
+ | eve.on("mouse", eatIt)(2);
+ | eve.on("mouse", scream);
+ | eve.on("mouse", catchIt)(1);
+ * This will ensure that `catchIt()` function will be called before `eatIt()`.
+ *
+ * If you want to put your handler before non-indexed handlers, specify a negative value.
+ * Note: I assume most of the time you don’t need to worry about z-index, but it’s nice to have this feature “just in case”.
+ \*/
+ eve.on = function (name, f) {
+ name = String(name);
+ if (typeof f != "function") {
+ return function () {};
+ }
+ var names = name.split(separator),
+ e = events;
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ e = e.n;
+ e = e.hasOwnProperty(names[i]) && e[names[i]] || (e[names[i]] = {n: {}});
+ }
+ e.f = e.f || [];
+ for (i = 0, ii = e.f.length; i < ii; i++) if (e.f[i] == f) {
+ return fun;
+ }
+ e.f.push(f);
+ return function (zIndex) {
+ if (+zIndex == +zIndex) {
+ f.zIndex = +zIndex;
+ }
+ };
+ };
+ /*\
+ * eve.f
+ [ method ]
+ **
+ * Returns function that will fire given event with optional arguments.
+ * Arguments that will be passed to the result function will be also
+ * concated to the list of final arguments.
+ | el.onclick = eve.f("click", 1, 2);
+ | eve.on("click", function (a, b, c) {
+ | console.log(a, b, c); // 1, 2, [event object]
+ | });
+ > Arguments
+ - event (string) event name
+ - varargs (…) and any other arguments
+ = (function) possible event handler function
+ \*/
+ eve.f = function (event) {
+ var attrs = [].slice.call(arguments, 1);
+ return function () {
+ eve.apply(null, [event, null].concat(attrs).concat([].slice.call(arguments, 0)));
+ };
+ };
+ /*\
+ * eve.stop
+ [ method ]
+ **
+ * Is used inside an event handler to stop the event, preventing any subsequent listeners from firing.
+ \*/
+ eve.stop = function () {
+ stop = 1;
+ };
+ /*\
+ * eve.nt
+ [ method ]
+ **
+ * Could be used inside event handler to figure out actual name of the event.
+ **
+ > Arguments
+ **
+ - subname (string) #optional subname of the event
+ **
+ = (string) name of the event, if `subname` is not specified
+ * or
+ = (boolean) `true`, if current event’s name contains `subname`
+ \*/
+ eve.nt = function (subname) {
+ if (subname) {
+ return new RegExp("(?:\\.|\\/|^)" + subname + "(?:\\.|\\/|$)").test(current_event);
+ }
+ return current_event;
+ };
+ /*\
+ * eve.nts
+ [ method ]
+ **
+ * Could be used inside event handler to figure out actual name of the event.
+ **
+ **
+ = (array) names of the event
+ \*/
+ eve.nts = function () {
+ return current_event.split(separator);
+ };
+ /*\
+ * eve.off
+ [ method ]
+ **
+ * Removes given function from the list of event listeners assigned to given name.
+ * If no arguments specified all the events will be cleared.
+ **
+ > Arguments
+ **
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated, with optional wildcards
+ - f (function) event handler function
+ \*/
+ /*\
+ * eve.unbind
+ [ method ]
+ **
+ * See @eve.off
+ \*/
+ eve.off = eve.unbind = function (name, f) {
+ if (!name) {
+ eve._events = events = {n: {}};
+ return;
+ }
+ var names = name.split(separator),
+ e,
+ key,
+ splice,
+ i, ii, j, jj,
+ cur = [events];
+ for (i = 0, ii = names.length; i < ii; i++) {
+ for (j = 0; j < cur.length; j += splice.length - 2) {
+ splice = [j, 1];
+ e = cur[j].n;
+ if (names[i] != wildcard) {
+ if (e[names[i]]) {
+ splice.push(e[names[i]]);
+ }
+ } else {
+ for (key in e) if (e[has](key)) {
+ splice.push(e[key]);
+ }
+ }
+ cur.splice.apply(cur, splice);
+ }
+ }
+ for (i = 0, ii = cur.length; i < ii; i++) {
+ e = cur[i];
+ while (e.n) {
+ if (f) {
+ if (e.f) {
+ for (j = 0, jj = e.f.length; j < jj; j++) if (e.f[j] == f) {
+ e.f.splice(j, 1);
+ break;
+ }
+ !e.f.length && delete e.f;
+ }
+ for (key in e.n) if (e.n[has](key) && e.n[key].f) {
+ var funcs = e.n[key].f;
+ for (j = 0, jj = funcs.length; j < jj; j++) if (funcs[j] == f) {
+ funcs.splice(j, 1);
+ break;
+ }
+ !funcs.length && delete e.n[key].f;
+ }
+ } else {
+ delete e.f;
+ for (key in e.n) if (e.n[has](key) && e.n[key].f) {
+ delete e.n[key].f;
+ }
+ }
+ e = e.n;
+ }
+ }
+ };
+ /*\
+ * eve.once
+ [ method ]
+ **
+ * Binds given event handler with a given name to only run once then unbind itself.
+ | eve.once("login", f);
+ | eve("login"); // triggers f
+ | eve("login"); // no listeners
+ * Use @eve to trigger the listener.
+ **
+ > Arguments
+ **
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated, with optional wildcards
+ - f (function) event handler function
+ **
+ = (function) same return function as @eve.on
+ \*/
+ eve.once = function (name, f) {
+ var f2 = function () {
+ eve.unbind(name, f2);
+ return f.apply(this, arguments);
+ };
+ return eve.on(name, f2);
+ };
+ /*\
+ * eve.version
+ [ property (string) ]
+ **
+ * Current version of the library.
+ \*/
+ eve.version = version;
+ eve.toString = function () {
+ return "You are running Eve " + version;
+ };
+ (typeof module != "undefined" && module.exports) ? (module.exports = eve) : (typeof define != "undefined" ? (define("eve", [], function() { return eve; })) : (glob.eve = eve));
+})(this);
+/**!
+ * RedRaphael 1.0.0 - JavaScript Vector Library
+ * Copyright (c) 2012-2013 FusionCharts Technologies
+ *
+ * Raphael 2.1.0
+ * Copyright (c) 2008-2012 Dmitry Baranovskiy
+ * Copyright © 2008-2012 Sencha Labs
+ *
+ * Licensed under the MIT license.
+ */
+(function (glob, factory) {
+ // AMD support
+ if (typeof define === "function" && define.amd) {
+ // Define as an anonymous module
+ define(["eve"], function( eve ) {
+ return factory(glob, eve);
+ });
+ } else {
+ // Browser globals (glob is window)
+ // Raphael adds itself to window
+ factory(glob, glob.eve);
+ }
+}(this, function (window, eve) {
+ /*\
+ * Raphael
+ [ method ]
+ **
+ * Creates a canvas object on which to draw.
+ * You must do this first, as all future calls to drawing methods
+ * from this instance will be bound to this canvas.
+ > Parameters
+ **
+ - container (HTMLElement|string) DOM element or its ID which is going to be a parent for drawing surface
+ - width (number)
+ - height (number)
+ - callback (function) #optional callback function which is going to be executed in the context of newly created paper
+ * or
+ - x (number)
+ - y (number)
+ - width (number)
+ - height (number)
+ - callback (function) #optional callback function which is going to be executed in the context of newly created paper
+ * or
+ - all (array) (first 3 or 4 elements in the array are equal to [containerID, width, height] or [x, y, width, height]. The rest are element descriptions in format {type: type, }). See @Paper.add.
+ - callback (function) #optional callback function which is going to be executed in the context of newly created paper
+ * or
+ - onReadyCallback (function) function that is going to be called on DOM ready event. You can also subscribe to this event via Eve’s “DOMLoad” event. In this case method returns `undefined`.
+ = (object) @Paper
+ > Usage
+ | // Each of the following examples create a canvas
+ | // that is 320px wide by 200px high.
+ | // Canvas is created at the viewport’s 10,50 coordinate.
+ | var paper = Raphael(10, 50, 320, 200);
+ | // Canvas is created at the top left corner of the #notepad element
+ | // (or its top right corner in dir="rtl" elements)
+ | var paper = Raphael(document.getElementById("notepad"), 320, 200);
+ | // Same as above
+ | var paper = Raphael("notepad", 320, 200);
+ | // Image dump
+ | var set = Raphael(["notepad", 320, 200, {
+ | type: "rect",
+ | x: 10,
+ | y: 10,
+ | width: 25,
+ | height: 25,
+ | stroke: "#f00"
+ | }, {
+ | type: "text",
+ | x: 30,
+ | y: 40,
+ | text: "Dump"
+ | }]);
+ \*/
+ function R(first) {
+ var args,
+ f;
+
+
+ if (R._url) { // reinitialize URL to be safe from popstate event
+ R._url = (R._g && R._g.win || window).location.href.replace(/#.*?$/, "");
+ }
+ if (R.is(first, "function")) {
+ return loaded ? first() : eve.on("raphael.DOMload", first);
+ }
+ else if (R.is(first, array)) {
+ return R._engine.create[apply](R, first.splice(0, 3 + R.is(first[0], nu))).add(first);
+ }
+ else {
+ args = Array.prototype.slice.call(arguments, 0);
+ if (R.is(args[args.length - 1], "function")) {
+ f = args.pop();
+ return loaded ? f.call(R._engine.create[apply](R, args)) : eve.on("raphael.DOMload", function() {
+ f.call(R._engine.create[apply](R, args));
+ });
+ } else {
+ return R._engine.create[apply](R, arguments);
+ }
+ }
+ }
+
+ R.upgrade = "1.0.0";
+ R.version = "2.1.0";
+ R.eve = eve;
+
+ var loaded,
+
+ undef,
+ E = "",
+ S = " ",
+ proto = "prototype",
+ has = "hasOwnProperty",
+ appendChild = "appendChild",
+ apply = "apply",
+ concat = "concat",
+ nu = "number",
+ string = "string",
+ array = "array",
+ object = "object",
+ finite = "finite",
+ toString = "toString",
+ fillString = "fill",
+ push = "push",
+ setAttribute = "setAttribute",
+ split = "split",
+ none = "none",
+ OBJECTSTRING = "object",
+ arrayToStr = "[object Array]",
+ objectToStr = "[object Object]",
+ arraySlice = Array.prototype.slice,
+ arraySplice = Array.prototype.splice,
+ g = {
+ doc: document,
+ win: window
+ },
+ oldRaphael = {
+ was: Object.prototype[has].call(g.win, "Raphael"),
+ is: g.win.Raphael
+ },
+ doc = g.doc,
+ win = g.win,
+
+ supportsTouch = R.supportsTouch = "createTouch" in doc,
+
+ CustomAttributes = function () {
+ /*\
+ * Raphael.ca
+ [ property (object) ]
+ **
+ * Shortcut for @Raphael.customAttributes
+ \*/
+ /*\
+ * Raphael.customAttributes
+ [ property (object) ]
+ **
+ * If you have a set of attributes that you would like to represent
+ * as a function of some number across all papers you can do it
+ * easily with custom attributes:
+ > Usage
+ | Raphael.customAttributes.hue = function (num) {
+ | num = num % 1;
+ | return {fill: "hsb(" + num + ", 0.75, 1)"};
+ | };
+ | // Custom attribute “hue” will change fill
+ | // to be given hue with fixed saturation and brightness.
+ | // Now you can use it like this:
+ | var c = paper.circle(10, 10, 10).attr({hue: .45});
+ | // or even like this:
+ | c.animate({hue: 1}, 1e3);
+ |
+ | // You could also create custom attribute
+ | // with multiple parameters:
+ | Raphael.customAttributes.hsb = function (h, s, b) {
+ | return {fill: "hsb(" + [h, s, b].join(",") + ")"};
+ | };
+ | c.attr({hsb: "0.5 .8 1"});
+ | c.animate({hsb: [1, 0, 0.5]}, 1e3);
+ \*/
+ },
+ caproto = R.ca = R.customAttributes = CustomAttributes.prototype,
+
+ Paper = function () {
+ /*\
+ * Paper.ca
+ [ property (object) ]
+ **
+ * Shortcut for @Paper.customAttributes
+ \*/
+ /*\
+ * Paper.customAttributes
+ [ property (object) ]
+ **
+ * If you have a set of attributes that you would like to represent
+ * as a function of some number you can do it easily with custom attributes:
+ > Usage
+ | paper.customAttributes.hue = function (num) {
+ | num = num % 1;
+ | return {fill: "hsb(" + num + ", 0.75, 1)"};
+ | };
+ | // Custom attribute “hue” will change fill
+ | // to be given hue with fixed saturation and brightness.
+ | // Now you can use it like this:
+ | var c = paper.circle(10, 10, 10).attr({hue: .45});
+ | // or even like this:
+ | c.animate({hue: 1}, 1e3);
+ |
+ | // You could also create custom attribute
+ | // with multiple parameters:
+ | paper.customAttributes.hsb = function (h, s, b) {
+ | return {fill: "hsb(" + [h, s, b].join(",") + ")"};
+ | };
+ | c.attr({hsb: "0.5 .8 1"});
+ | c.animate({hsb: [1, 0, 0.5]}, 1e3);
+ \*/
+ this.ca = this.customAttributes = new CustomAttributes();
+ this._CustomAttributes = function () {};
+ this._CustomAttributes.prototype = this.ca;
+ },
+
+ /*\
+ * Raphael.fn
+ [ property (object) ]
+ **
+ * You can add your own method to the canvas. For example if you want to draw a pie chart,
+ * you can create your own pie chart function and ship it as a Raphaël plugin. To do this
+ * you need to extend the `Raphael.fn` object. You should modify the `fn` object before a
+ * Raphaël instance is created, otherwise it will take no effect. Please note that the
+ * ability for namespaced plugins was removed in Raphael 2.0. It is up to the plugin to
+ * ensure any namespacing ensures proper context.
+ > Usage
+ | Raphael.fn.arrow = function (x1, y1, x2, y2, size) {
+ | return this.path( ... );
+ | };
+ | // or create namespace
+ | Raphael.fn.mystuff = {
+ | arrow: function () {…},
+ | star: function () {…},
+ | // etc…
+ | };
+ | var paper = Raphael(10, 10, 630, 480);
+ | // then use it
+ | paper.arrow(10, 10, 30, 30, 5).attr({fill: "#f00"});
+ | paper.mystuff.arrow();
+ | paper.mystuff.star();
+ \*/
+ paperproto = R.fn = Paper.prototype = R.prototype,
+
+ elements = {
+ circle: 1,
+ rect: 1,
+ path: 1,
+ ellipse: 1,
+ text: 1,
+ image: 1,
+ group: 1
+ },
+ events = "click dblclick mousedown mousemove mouseout mouseover mouseup touchstart touchmove touchend touchcancel"[split](S),
+ touchMap = R._touchMap = {
+ mousedown: "touchstart",
+ mousemove: "touchmove",
+ mouseup: "touchend"
+ },
+
+ Str = win.String,
+ toFloat = win.parseFloat,
+ toInt = win.parseInt,
+ math = win.Math,
+ mmax = math.max,
+ mmin = math.min,
+ abs = math.abs,
+ pow = math.pow,
+ mathCos = math.cos,
+ mathSin = math.sin,
+ mathSqrt = math.sqrt,
+ round = math.round,
+ PI = math.PI,
+ deg2rad = PI / 180,
+ rad2deg = 180 / PI,
+
+ lowerCase = Str.prototype.toLowerCase,
+ upperCase = Str.prototype.toUpperCase,
+ objectToString = win.Object.prototype.toString,
+ paper = {},
+
+ separator = /[, ]+/,
+ formatrg = /\{(\d+)\}/g,
+ ISURL = R._ISURL = /^url\(['"]?([^\)]+?)['"]?\)$/i,
+ colourRegExp = /^\s*((#[a-f\d]{6})|(#[a-f\d]{3})|rgba?\(\s*([\d\.]+%?\s*,\s*[\d\.]+%?\s*,\s*[\d\.]+%?(?:\s*,\s*[\d\.]+%?)?)\s*\)|hsba?\(\s*([\d\.]+(?:deg|\xb0|%)?\s*,\s*[\d\.]+%?\s*,\s*[\d\.]+(?:%?\s*,\s*[\d\.]+)?)%?\s*\)|hsla?\(\s*([\d\.]+(?:deg|\xb0|%)?\s*,\s*[\d\.]+%?\s*,\s*[\d\.]+(?:%?\s*,\s*[\d\.]+)?)%?\s*\))\s*$/i,
+ bezierrg = /^(?:cubic-)?bezier\(([^,]+),([^,]+),([^,]+),([^\)]+)\)/,
+ whitespace = /[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]/g,
+ commaSpaces = /[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*/,
+ p2s = /,?([achlmqrstvxz]),?/gi,
+ pathCommand = /([achlmrqstvz])[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029,]*((-?\d*\.?\d*(?:e[\-+]?\d+)?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*)+)/ig,
+ tCommand = /([rstm])[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029,]*((-?\d*\.?\d*(?:e[\-+]?\d+)?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*)+)/ig,
+ pathValues = /(-?\d*\.?\d*(?:e[\-+]?\d+)?)[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*/ig,
+ radial_gradient = R._radial_gradient = /^x?r(?:\(([^\)]*?)\))?/,
+
+ isnan = {
+ "NaN": 1,
+ "Infinity": 1,
+ "-Infinity": 1
+ },
+ hsrg = {
+ hs: 1,
+ rg: 1
+ },
+ availableAttrs = R._availableAttrs = {
+ "arrow-end": none,
+ "arrow-start": none,
+ blur: 0,
+ "clip-rect": "0 0 1e9 1e9",
+ "clip-path": E,
+ cursor: "default",
+ cx: 0,
+ cy: 0,
+ fill: "#fff",
+ "fill-opacity": 1,
+ font: '10px "Arial"',
+ "font-family": '"Arial"',
+ "font-size": "10",
+ "font-style": "normal",
+ "font-weight": 400,
+ gradient: 0,
+ height: 0,
+ href: "about:blank",
+ "letter-spacing": 0,
+ "line-height": 12,
+ "vertical-align": "middle",
+ opacity: 1,
+ path: "M0,0",
+ r: 0,
+ rx: 0,
+ ry: 0,
+ src: E,
+ stroke: "#000",
+ "stroke-dasharray": E,
+ "stroke-linecap": "butt",
+ "stroke-linejoin": "butt",
+ "stroke-miterlimit": 0,
+ "stroke-opacity": 1,
+ "stroke-width": 1,
+ target: "_blank",
+ "text-anchor": "middle",
+ "visibility": E,
+ title: E,
+ transform: E,
+ rotation: 0,
+ width: 0,
+ x: 0,
+ y: 0
+ },
+ availableAnimAttrs = R._availableAnimAttrs = {
+ blur: nu,
+ "clip-rect": "csv",
+ "clip-path": "path",
+ cx: nu,
+ cy: nu,
+ fill: "colour",
+ "fill-opacity": nu,
+ "font-size": nu,
+ height: nu,
+ opacity: nu,
+ path: "path",
+ r: nu,
+ rx: nu,
+ ry: nu,
+ stroke: "colour",
+ "stroke-opacity": nu,
+ "stroke-width": nu,
+ transform: "transform",
+ width: nu,
+ x: nu,
+ y: nu
+ },
+ eldata = {},
+
+ sortByKey = function(a, b) {
+ return a.key - b.key;
+ },
+ sortByNumber = function(a, b) {
+ return toFloat(a) - toFloat(b);
+ },
+ fun = function() {
+ },
+ pipe = function(x) {
+ return x;
+ },
+
+ rectPath = R._rectPath = function(x, y, w, h, r) {
+ if (r) {
+ return [["M", x + r, y], ["l", w - r * 2, 0], ["a", r, r, 0, 0, 1, r, r], ["l", 0, h - r * 2], ["a", r, r, 0, 0, 1, -r, r], ["l", r * 2 - w, 0], ["a", r, r, 0, 0, 1, -r, -r], ["l", 0, r * 2 - h], ["a", r, r, 0, 0, 1, r, -r], ["z"]];
+ }
+ return [["M", x, y], ["l", w, 0], ["l", 0, h], ["l", -w, 0], ["z"]];
+ },
+
+ ellipsePath = function(x, y, rx, ry) {
+ if (ry == null) {
+ ry = rx;
+ }
+ return [["M", x, y], ["m", 0, -ry], ["a", rx, ry, 0, 1, 1, 0, 2 * ry], ["a", rx, ry, 0, 1, 1, 0, -2 * ry], ["z"]];
+ },
+
+ getPath = R._getPath = {
+ group: function() {
+ return false;
+ },
+ path: function(el) {
+ return el.attr("path");
+ },
+ circle: function(el) {
+ var a = el.attrs;
+ return ellipsePath(a.cx, a.cy, a.r);
+ },
+ ellipse: function(el) {
+ var a = el.attrs;
+ return ellipsePath(a.cx, a.cy, a.rx, a.ry);
+ },
+ rect: function(el) {
+ var a = el.attrs;
+ return rectPath(a.x, a.y, a.width, a.height, a.r);
+ },
+ image: function(el) {
+ var a = el.attrs;
+ return rectPath(a.x, a.y, a.width, a.height);
+ },
+ text: function(el) {
+ var bbox = el._getBBox();
+ return rectPath(bbox.x, bbox.y, bbox.width, bbox.height);
+ }
+ },
+
+ /*\
+ * Raphael.mapPath
+ [ method ]
+ **
+ * Transform the path string with given matrix.
+ > Parameters
+ - path (string) path string
+ - matrix (object) see @Matrix
+ = (string) transformed path string
+ \*/
+ mapPath = R.mapPath = function(path, matrix) {
+ if (!matrix) {
+ return path;
+ }
+ var x,
+ y,
+ i,
+ j,
+ ii,
+ jj,
+ pathi;
+
+ path = path2curve(path);
+ for (i = 0, ii = path.length; i < ii; i++) {
+ pathi = path[i];
+ for (j = 1, jj = pathi.length; j < jj; j += 2) {
+ x = matrix.x(pathi[j], pathi[j + 1]);
+ y = matrix.y(pathi[j], pathi[j + 1]);
+ pathi[j] = x;
+ pathi[j + 1] = y;
+ }
+ }
+ return path;
+ },
+
+ /*\
+ * Raphael.pick
+ [ method ]
+ **
+ * Returns the first truthy argument.
+ \*/
+ pick = R.pick = function() {
+ for (var arg, i = 0, ii = arguments.length; i < ii; i += 1) {
+ arg = arguments[i];
+ if (!arg && arg !== false && arg !== 0) {
+ continue;
+ }
+ return arg;
+ }
+ return undef;
+ },
+
+ lastArgIfGroup = R._lastArgIfGroup = function (args, clear) {
+ var last = args.length - 1,
+ arg = args[last];
+
+ if (arg && (arg.constructor === R.el.constructor) && arg.type === 'group') {
+ if (clear) {
+ arraySplice.call(args, last, 1);
+ }
+ return arg;
+ }
+ },
+
+ merge = R.merge = function (obj1, obj2, skipUndef, tgtArr, srcArr) {
+ var item,
+ srcVal,
+ tgtVal,
+ str,
+ cRef;
+ //check whether obj2 is an array
+ //if array then iterate through it's index
+ //**** MOOTOOLS precution
+
+ if (!srcArr) {
+ tgtArr = [obj1];
+ srcArr = [obj2];
+ }
+ else {
+ tgtArr.push(obj1);
+ srcArr.push(obj2);
+ }
+
+ if (obj2 instanceof Array) {
+ for (item = 0; item < obj2.length; item += 1) {
+ try {
+ srcVal = obj1[item];
+ tgtVal = obj2[item];
+ }
+ catch (e) {
+ continue;
+ }
+
+ if (typeof tgtVal !== OBJECTSTRING) {
+ if (!(skipUndef && tgtVal === undefined)) {
+ obj1[item] = tgtVal;
+ }
+ }
+ else {
+ if (srcVal === null || typeof srcVal !== OBJECTSTRING) {
+ srcVal = obj1[item] = tgtVal instanceof Array ? [] : {};
+ }
+ cRef = checkCyclicRef(tgtVal, srcArr);
+ if (cRef !== -1) {
+ srcVal = obj1[item] = tgtArr[cRef];
+ }
+ else {
+ merge(srcVal, tgtVal, skipUndef, tgtArr, srcArr);
+ }
+ }
+ }
+ }
+ else {
+ for (item in obj2) {
+ try {
+ srcVal = obj1[item];
+ tgtVal = obj2[item];
+ }
+ catch (e) {
+ continue;
+ }
+
+ if (tgtVal !== null && typeof tgtVal === OBJECTSTRING) {
+ // Fix for issue BUG: FWXT-602
+ // IE < 9 Object.prototype.toString.call(null) gives
+ // "[object Object]" instead of "[object Null]"
+ // that's why null value becomes Object in IE < 9
+ str = objectToString.call(tgtVal);
+ if (str === objectToStr) {
+ if (srcVal === null || typeof srcVal !== OBJECTSTRING) {
+ srcVal = obj1[item] = {};
+ }
+ cRef = checkCyclicRef(tgtVal, srcArr);
+ if (cRef !== -1) {
+ srcVal = obj1[item] = tgtArr[cRef];
+ }
+ else {
+ merge(srcVal, tgtVal, skipUndef, tgtArr, srcArr);
+ }
+ }
+ else if (str === arrayToStr) {
+ if (srcVal === null || !(srcVal instanceof Array)) {
+ srcVal = obj1[item] = [];
+ }
+ cRef = checkCyclicRef(tgtVal, srcArr);
+ if (cRef !== -1) {
+ srcVal = obj1[item] = tgtArr[cRef];
+ }
+ else {
+ merge(srcVal, tgtVal, skipUndef, tgtArr, srcArr);
+ }
+ }
+ else {
+ obj1[item] = tgtVal;
+ }
+ }
+ else {
+ obj1[item] = tgtVal;
+ }
+ }
+ }
+ return obj1;
+ },
+
+ extend = R.extend = function (obj1, obj2, skipUndef) {
+ if (typeof obj1 !== OBJECTSTRING && typeof obj2 !== OBJECTSTRING) {//if none of the arguments are object then return back
+ return null;
+ }
+
+ if (typeof obj2 !== OBJECTSTRING || obj2 === null) {
+ return obj1;
+ }
+
+ if (typeof obj1 !== OBJECTSTRING) {
+ obj1 = obj2 instanceof Array ? [] : {};
+ }
+ merge(obj1, obj2, skipUndef);
+ return obj1;
+
+ },
+
+ /*\
+ * Raphael.is
+ [ method ]
+ **
+ * Handfull replacement for `typeof` operator.
+ > Parameters
+ - o (…) any object or primitive
+ - type (string) name of the type, i.e. “string”, “function”, “number”, etc.
+ = (boolean) is given value is of given type
+ \*/
+ is = R.is = function(o, type) {
+ type = lowerCase.call(type);
+
+ if (type == finite) {
+ return !isnan[has](+o);
+ }
+ if (type == array) {
+ return o instanceof Array;
+ }
+ if (type === 'object' && (o === undef || o === null)) {
+ return false;
+ }
+ return (type == "null" && o === null) ||
+ (type == typeof o && o !== null) ||
+ (type == object && o === Object(o)) ||
+ (type == "array" && Array.isArray && Array.isArray(o)) ||
+ objectToString.call(o).slice(8, -1).toLowerCase() == type;
+ },
+
+ /*\
+ * Raphael.clone
+ [ method ]
+ **
+ * Returns a recursively cloned version of an object.
+ \*/
+ clone = R.clone = function (obj) {
+ if (Object(obj) !== obj) {
+ return obj;
+ }
+ var res = new obj.constructor;
+ for (var key in obj)
+ if (obj[has](key)) {
+ res[key] = clone(obj[key]);
+ }
+ return res;
+ },
+
+ /*\
+ * Raphael.createUUID
+ [ method ]
+ **
+ * Returns RFC4122, version 4 ID
+ \*/
+ createUUID = R.createUUID = (function(uuidRegEx, uuidReplacer) {
+ return function() {
+ return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(uuidRegEx, uuidReplacer).toUpperCase();
+ };
+ })(/[xy]/g, function(c) {
+ var r = math.random() * 16 | 0,
+ v = c == "x" ? r : (r & 3 | 8);
+ return v.toString(16);
+ });
+
+ R._g = g;
+
+ /*\
+ * Raphael.type
+ [ property (string) ]
+ **
+ * Can be “SVG”, “VML” or empty, depending on browser support.
+ \*/
+ R.type = (win.ENABLE_RED_CANVAS && (win.CanvasRenderingContext2D || doc.createElement('canvas').getContext)) ? "CANVAS" :
+ (win.SVGAngle || doc.implementation.hasFeature("http://www.w3.org/TR/SVG11/feature#BasicStructure", "1.1") ? "SVG" : "VML");
+
+ if (R.type == "VML") {
+ var d = doc.createElement("div"),
+ b;
+
+ d.innerHTML = ' ';
+ b = d.firstChild;
+ b.style.behavior = "url(#default#VML)";
+ if (!(b && typeof b.adj == object)) {
+ return (R.type = E);
+ }
+ d = null;
+ }
+
+ /*\
+ * Raphael.svg
+ [ property (boolean) ]
+ **
+ * `true` if browser supports SVG.
+ \*/
+ /*\
+ * Raphael.vml
+ [ property (boolean) ]
+ **
+ * `true` if browser supports VML.
+ \*/
+ R.svg = !((R.vml = R.type == "VML") || (R.canvas = R.type == "CANVAS"));
+
+ R._Paper = Paper;
+ R._id = 0;
+ R._oid = 0;
+
+ /*\
+ * Raphael.angle
+ [ method ]
+ **
+ * Returns angle between two or three points
+ > Parameters
+ - x1 (number) x coord of first point
+ - y1 (number) y coord of first point
+ - x2 (number) x coord of second point
+ - y2 (number) y coord of second point
+ - x3 (number) #optional x coord of third point
+ - y3 (number) #optional y coord of third point
+ = (number) angle in degrees.
+ \*/
+ R.angle = function (x1, y1, x2, y2, x3, y3) {
+ if (x3 == null) {
+ var x = x1 - x2,
+ y = y1 - y2;
+ if (!x && !y) {
+ return 0;
+ }
+ return (180 + math.atan2(-y, -x) * rad2deg + 360) % 360;
+ }
+ else {
+ return R.angle(x1, y1, x3, y3) - R.angle(x2, y2, x3, y3);
+ }
+ };
+
+ /*\
+ * Raphael.rad
+ [ method ]
+ **
+ * Transform angle to radians
+ > Parameters
+ - deg (number) angle in degrees
+ = (number) angle in radians.
+ \*/
+ R.rad = function (deg) {
+ return deg % 360 * deg2rad;
+ };
+
+ /*\
+ * Raphael.deg
+ [ method ]
+ **
+ * Transform angle to degrees
+ > Parameters
+ - deg (number) angle in radians
+ = (number) angle in degrees.
+ \*/
+ R.deg = function (rad) {
+ return rad * rad2deg % 360;
+ };
+
+ /*\
+ * Raphael.snapTo
+ [ method ]
+ **
+ * Snaps given value to given grid.
+ > Parameters
+ - values (array|number) given array of values or step of the grid
+ - value (number) value to adjust
+ - tolerance (number) #optional tolerance for snapping. Default is `10`.
+ = (number) adjusted value.
+ \*/
+ R.snapTo = function (values, value, tolerance) {
+ var rem,
+ i;
+
+ if (!is(tolerance, finite)) {
+ tolerance = 10;
+ }
+
+ if (is(values, array)) {
+ i = values.length;
+ while (i--) {
+ if (abs(values[i] - value) <= tolerance) {
+ return values[i];
+ }
+ }
+ }
+ else {
+ values = +values;
+ rem = value % values;
+
+ if (rem < tolerance) {
+ return value - rem;
+ }
+ if (rem > values - tolerance) {
+ return value - rem + values;
+ }
+ }
+ return value;
+ };
+
+ /*\
+ * Raphael.setWindow
+ [ method ]
+ **
+ * Used when you need to draw in `<iframe>`. Switched window to the iframe one.
+ > Parameters
+ - newwin (window) new window object
+ \*/
+ R.setWindow = function (newwin) {
+ eve("raphael.setWindow", R, g.win, newwin);
+ win = g.win = newwin;
+ doc = g.doc = g.win.document;
+ if (R._engine.initWin) {
+ R._engine.initWin(g.win);
+ }
+ };
+
+ var toHex = function (color) {
+ if (R.vml) {
+ // http://dean.edwards.name/weblog/2009/10/convert-any-colour-value-to-hex-in-msie/
+ var trim = /^\s+|\s+$/g;
+ var bod;
+ try {
+ var docum = new ActiveXObject("htmlfile");
+ docum.write("");
+ docum.close();
+ bod = docum.body;
+ } catch (e) {
+ bod = createPopup().document.body;
+ }
+ var range = bod.createTextRange();
+ toHex = cacher(function(color) {
+ try {
+ bod.style.color = Str(color).replace(trim, E);
+ var value = range.queryCommandValue("ForeColor");
+ value = ((value & 255) << 16) | (value & 65280) | ((value & 16711680) >>> 16);
+ return "#" + ("000000" + value.toString(16)).slice(-6);
+ } catch (e) {
+ return none;
+ }
+ });
+ } else {
+ var i = g.doc.createElement("i");
+ i.title = "Rapha\xebl Colour Picker";
+ i.style.display = none;
+ g.doc.body.appendChild(i);
+ toHex = cacher(function(color) {
+ i.style.color = color;
+ return g.doc.defaultView.getComputedStyle(i, E).getPropertyValue("color");
+ });
+ }
+ return toHex(color);
+ },
+ hsbtoString = function() {
+ return "hsb(" + [this.h, this.s, this.b] + ")";
+ },
+ hsltoString = function() {
+ return "hsl(" + [this.h, this.s, this.l] + ")";
+ },
+ rgbtoString = function() {
+ return this.hex;
+ },
+ prepareRGB = function(r, g, b) {
+ if (g == null && is(r, object) && "r" in r && "g" in r && "b" in r) {
+ b = r.b;
+ g = r.g;
+ r = r.r;
+ }
+ if (g == null && is(r, string)) {
+ var clr = R.getRGB(r);
+ r = clr.r;
+ g = clr.g;
+ b = clr.b;
+ }
+ if (r > 1 || g > 1 || b > 1) {
+ r /= 255;
+ g /= 255;
+ b /= 255;
+ }
+
+ return [r, g, b];
+ },
+ packageRGB = function(r, g, b, o) {
+ var rgb = {
+ r: (r *= 255),
+ g: (g *= 255),
+ b: (b *= 255),
+ hex: R.rgb(r, g, b),
+ toString: rgbtoString
+ };
+ is(o, "finite") && (rgb.opacity = o);
+ return rgb;
+ };
+
+ /*\
+ * Raphael.color
+ [ method ]
+ **
+ * Parses the color string and returns object with all values for the given color.
+ > Parameters
+ - clr (string) color string in one of the supported formats (see @Raphael.getRGB)
+ = (object) Combined RGB & HSB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue,
+ o hex (string) color in HTML/CSS format: #••••••,
+ o error (boolean) `true` if string can’t be parsed,
+ o h (number) hue,
+ o s (number) saturation,
+ o v (number) value (brightness),
+ o l (number) lightness
+ o }
+ \*/
+ R.color = function(clr) {
+ var rgb;
+ if (R.is(clr, object) && "h" in clr && "s" in clr && "b" in clr) {
+ rgb = R.hsb2rgb(clr);
+ clr.r = rgb.r;
+ clr.g = rgb.g;
+ clr.b = rgb.b;
+ clr.hex = rgb.hex;
+ } else if (R.is(clr, object) && "h" in clr && "s" in clr && "l" in clr) {
+ rgb = R.hsl2rgb(clr);
+ clr.r = rgb.r;
+ clr.g = rgb.g;
+ clr.b = rgb.b;
+ clr.hex = rgb.hex;
+ } else {
+ if (R.is(clr, "string")) {
+ clr = R.getRGB(clr);
+ }
+ if (R.is(clr, object) && "r" in clr && "g" in clr && "b" in clr) {
+ rgb = R.rgb2hsl(clr);
+ clr.h = rgb.h;
+ clr.s = rgb.s;
+ clr.l = rgb.l;
+ rgb = R.rgb2hsb(clr);
+ clr.v = rgb.b;
+ } else {
+ clr = {
+ hex: none
+ };
+ clr.r = clr.g = clr.b = clr.h = clr.s = clr.v = clr.l = -1;
+ }
+ }
+ clr.toString = rgbtoString;
+ return clr;
+ };
+
+ /*\
+ * Raphael.hsb2rgb
+ [ method ]
+ **
+ * Converts HSB values to RGB object.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - v (number) value or brightness
+ = (object) RGB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue,
+ o hex (string) color in HTML/CSS format: #••••••
+ o }
+ \*/
+ R.hsb2rgb = function(h, s, v, o) {
+ if (this.is(h, object) && "h" in h && "s" in h && "b" in h) {
+ v = h.b;
+ s = h.s;
+ h = h.h;
+ o = h.o;
+ }
+ h *= 360;
+ var R, G, B, X, C;
+ h = (h % 360) / 60;
+ C = v * s;
+ X = C * (1 - abs(h % 2 - 1));
+ R = G = B = v - C;
+
+ h = ~~h;
+ R += [C, X, 0, 0, X, C][h];
+ G += [X, C, C, X, 0, 0][h];
+ B += [0, 0, X, C, C, X][h];
+ return packageRGB(R, G, B, o);
+ };
+
+ /*\
+ * Raphael.hsl2rgb
+ [ method ]
+ **
+ * Converts HSL values to RGB object.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - l (number) luminosity
+ = (object) RGB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue,
+ o hex (string) color in HTML/CSS format: #••••••
+ o }
+ \*/
+ R.hsl2rgb = function(h, s, l, o) {
+ if (this.is(h, object) && "h" in h && "s" in h && "l" in h) {
+ l = h.l;
+ s = h.s;
+ h = h.h;
+ }
+ if (h > 1 || s > 1 || l > 1) {
+ h /= 360;
+ s /= 100;
+ l /= 100;
+ }
+ h *= 360;
+ var R, G, B, X, C;
+ h = (h % 360) / 60;
+ C = 2 * s * (l < .5 ? l : 1 - l);
+ X = C * (1 - abs(h % 2 - 1));
+ R = G = B = l - C / 2;
+
+ h = ~~h;
+ R += [C, X, 0, 0, X, C][h];
+ G += [X, C, C, X, 0, 0][h];
+ B += [0, 0, X, C, C, X][h];
+ return packageRGB(R, G, B, o);
+ };
+
+ /*\
+ * Raphael.rgb2hsb
+ [ method ]
+ **
+ * Converts RGB values to HSB object.
+ > Parameters
+ - r (number) red
+ - g (number) green
+ - b (number) blue
+ = (object) HSB object in format:
+ o {
+ o h (number) hue
+ o s (number) saturation
+ o b (number) brightness
+ o }
+ \*/
+ R.rgb2hsb = function(r, g, b) {
+ b = prepareRGB(r, g, b);
+ r = b[0];
+ g = b[1];
+ b = b[2];
+
+ var H, S, V, C;
+ V = mmax(r, g, b);
+ C = V - mmin(r, g, b);
+ H = (C == 0 ? null :
+ V == r ? (g - b) / C :
+ V == g ? (b - r) / C + 2 :
+ (r - g) / C + 4
+ );
+ H = ((H + 360) % 6) * 60 / 360;
+ S = C == 0 ? 0 : C / V;
+ return {
+ h: H,
+ s: S,
+ b: V,
+ toString: hsbtoString
+ };
+ };
+
+ /*\
+ * Raphael.rgb2hsl
+ [ method ]
+ **
+ * Converts RGB values to HSL object.
+ > Parameters
+ - r (number) red
+ - g (number) green
+ - b (number) blue
+ = (object) HSL object in format:
+ o {
+ o h (number) hue
+ o s (number) saturation
+ o l (number) luminosity
+ o }
+ \*/
+ R.rgb2hsl = function(r, g, b) {
+ b = prepareRGB(r, g, b);
+ r = b[0];
+ g = b[1];
+ b = b[2];
+
+ var H, S, L, M, m, C;
+ M = mmax(r, g, b);
+ m = mmin(r, g, b);
+ C = M - m;
+ H = (C == 0 ? null :
+ M == r ? (g - b) / C :
+ M == g ? (b - r) / C + 2 :
+ (r - g) / C + 4);
+ H = ((H + 360) % 6) * 60 / 360;
+ L = (M + m) / 2;
+ S = (C == 0 ? 0 :
+ L < .5 ? C / (2 * L) :
+ C / (2 - 2 * L));
+ return {
+ h: H,
+ s: S,
+ l: L,
+ toString: hsltoString
+ };
+ };
+
+ R._path2string = function() {
+ return this.join(",").replace(p2s, "$1");
+ };
+
+ function repush(array, item) {
+ for (var i = 0, ii = array.length; i < ii; i++) {
+ if (array[i] === item) {
+ return array.push(array.splice(i, 1)[0]);
+ }
+ }
+ }
+
+ var cacher = R._cacher = function (f, scope, postprocessor) {
+ function cachedfunction() {
+ var arg = arraySlice.call(arguments, 0),
+ args = arg.join("\u2400"),
+ cache = cachedfunction.cache = cachedfunction.cache || {},
+ count = cachedfunction.count = cachedfunction.count || [];
+ if (cache[has](args)) {
+ repush(count, args);
+ return postprocessor ? postprocessor(cache[args]) : cache[args];
+ }
+ count.length >= 1e3 && delete cache[count.shift()];
+ count.push(args);
+ cache[args] = f[apply](scope, arg);
+ return postprocessor ? postprocessor(cache[args]) : cache[args];
+ }
+ return cachedfunction;
+ };
+
+ var preload = R._preload = function(src, f) {
+ var img = doc.createElement("img");
+ img.style.cssText = "position:absolute;left:-9999em;top:-9999em";
+ img.onload = function() {
+ f.call(this);
+ this.onload = null;
+ doc.body.removeChild(this);
+ };
+ img.onerror = function() {
+ doc.body.removeChild(this);
+ };
+ doc.body.appendChild(img);
+ img.src = src;
+ };
+
+ function clrToString() {
+ return this.hex;
+ }
+
+ /*\
+ * Raphael.getRGB
+ [ method ]
+ **
+ * Parses colour string as RGB object
+ > Parameters
+ - colour (string) colour string in one of formats:
+ #
+ # Colour name (“red
”, “green
”, “cornflowerblue
”, etc)
+ # #••• — shortened HTML colour: (“#000
”, “#fc0
”, etc)
+ # #•••••• — full length HTML colour: (“#000000
”, “#bd2300
”)
+ # rgb(•••, •••, •••) — red, green and blue channels’ values: (“rgb(200, 100, 0)
”)
+ # rgb(•••%, •••%, •••%) — same as above, but in %: (“rgb(100%, 175%, 0%)
”)
+ # hsb(•••, •••, •••) — hue, saturation and brightness values: (“hsb(0.5, 0.25, 1)
”)
+ # hsb(•••%, •••%, •••%) — same as above, but in %
+ # hsl(•••, •••, •••) — same as hsb
+ # hsl(•••%, •••%, •••%) — same as hsb
+ #
+ = (object) RGB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue
+ o hex (string) color in HTML/CSS format: #••••••,
+ o error (boolean) true if string can’t be parsed
+ o }
+ \*/
+ R.getRGB = cacher(function(colour) {
+ var opacity,
+ res,
+ red,
+ green,
+ blue,
+ t,
+ values,
+ rgb;
+
+ colour && is(colour, 'object') && "opacity" in colour &&
+ (opacity = colour.opacity);
+ if (!colour || !!((colour = Str(colour)).indexOf("-") + 1)) {
+ return {
+ r: -1,
+ g: -1,
+ b: -1,
+ hex: none,
+ error: 1,
+ toString: clrToString
+ };
+ }
+ if (colour == none) {
+ return {
+ r: -1,
+ g: -1,
+ b: -1,
+ hex: none,
+ toString: clrToString
+ };
+ }
+ !(hsrg[has](colour.toLowerCase().substring(0, 2)) ||
+ colour.charAt() === "#") && (colour = toHex(colour));
+
+
+ if ((rgb = colour.match(colourRegExp))) {
+ if (rgb[2]) {
+ blue = toInt(rgb[2].substring(5), 16);
+ green = toInt(rgb[2].substring(3, 5), 16);
+ red = toInt(rgb[2].substring(1, 3), 16);
+ }
+ if (rgb[3]) {
+ blue = toInt((t = rgb[3].charAt(3)) + t, 16);
+ green = toInt((t = rgb[3].charAt(2)) + t, 16);
+ red = toInt((t = rgb[3].charAt(1)) + t, 16);
+ }
+ if (rgb[4]) {
+ values = rgb[4][split](commaSpaces);
+ red = toFloat(values[0]);
+ values[0].slice(-1) == "%" && (red *= 2.55);
+ green = toFloat(values[1]);
+ values[1].slice(-1) == "%" && (green *= 2.55);
+ blue = toFloat(values[2]);
+ values[2].slice(-1) == "%" && (blue *= 2.55);
+ rgb[1].toLowerCase().slice(0, 4) == "rgba" && (opacity = toFloat(values[3]));
+ values[3] && values[3].slice(-1) == "%" && (opacity /= 100);
+ }
+ if (rgb[5]) {
+ values = rgb[5][split](commaSpaces);
+ red = toFloat(values[0]);
+ values[0].slice(-1) == "%" && (red *= 2.55);
+ green = toFloat(values[1]);
+ values[1].slice(-1) == "%" && (green *= 2.55);
+ blue = toFloat(values[2]);
+ values[2].slice(-1) == "%" && (blue *= 2.55);
+ (values[0].slice(-3) == "deg" || values[0].slice(-1) == "\xb0") && (red /= 360);
+ rgb[1].toLowerCase().slice(0, 4) == "hsba" && (opacity = toFloat(values[3]));
+ values[3] && values[3].slice(-1) == "%" && (opacity /= 100);
+ return R.hsb2rgb(red, green, blue, opacity);
+ }
+ if (rgb[6]) {
+ values = rgb[6][split](commaSpaces);
+ red = toFloat(values[0]);
+ values[0].slice(-1) == "%" && (red *= 2.55);
+ green = toFloat(values[1]);
+ values[1].slice(-1) == "%" && (green *= 2.55);
+ blue = toFloat(values[2]);
+ values[2].slice(-1) == "%" && (blue *= 2.55);
+ (values[0].slice(-3) == "deg" || values[0].slice(-1) == "\xb0") && (red /= 360);
+ rgb[1].toLowerCase().slice(0, 4) == "hsla" && (opacity = toFloat(values[3]));
+ values[3] && values[3].slice(-1) == "%" && (opacity /= 100);
+ return R.hsl2rgb(red, green, blue, opacity);
+ }
+ rgb = {
+ r: red,
+ g: green,
+ b: blue,
+ toString: clrToString
+ };
+ rgb.hex = "#" + (16777216 | blue | (green << 8) | (red << 16)).toString(16).slice(1);
+ R.is(opacity, "finite") && (rgb.opacity = opacity);
+ return rgb;
+ }
+ return {
+ r: -1,
+ g: -1,
+ b: -1,
+ hex: none,
+ error: 1,
+ toString: clrToString
+ };
+ }, R);
+
+ R.tintshade = cacher(function(colour, percent) {
+ var rgb = R.getRGB(colour),
+ tint,
+ offset = 255;
+
+ (percent < 0) && (percent *= -1, offset = 0);
+ (percent > 1) && (percent = 1);
+
+ tint = percent === 0 ? rgb : {
+ r: offset - (offset - rgb.r) * percent,
+ g: offset - (offset - rgb.g) * percent,
+ b: offset - (offset - rgb.b) * percent,
+ toString: clrToString
+ };
+ tint.hex = R.rgb(tint.r, tint.g, tint.b);
+ rgb.error && (tint.error = rgb.error);
+
+ if ("opacity" in rgb) {
+ tint.rgba = 'rgba(' + [tint.r, tint.g, tint.b, rgb.opacity].join(',') + ')';
+ tint.opacity = rgb.opacity;
+ }
+ else {
+ tint.rgba = 'rgb(' + [tint.r, tint.g, tint.b].join(',') + ')';
+ }
+ return tint;
+ }, R);
+
+ /*\
+ * Raphael.hsb
+ [ method ]
+ **
+ * Converts HSB values to hex representation of the colour.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - b (number) value or brightness
+ = (string) hex representation of the colour.
+ \*/
+ R.hsb = cacher(function(h, s, b) {
+ return R.hsb2rgb(h, s, b).hex;
+ });
+
+ /*\
+ * Raphael.hsl
+ [ method ]
+ **
+ * Converts HSL values to hex representation of the colour.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - l (number) luminosity
+ = (string) hex representation of the colour.
+ \*/
+ R.hsl = cacher(function(h, s, l) {
+ return R.hsl2rgb(h, s, l).hex;
+ });
+
+ /*\
+ * Raphael.rgb
+ [ method ]
+ **
+ * Converts RGB values to hex representation of the colour.
+ > Parameters
+ - r (number) red
+ - g (number) green
+ - b (number) blue
+ = (string) hex representation of the colour.
+ \*/
+ R.rgb = cacher(function(r, g, b) {
+ return "#" + (16777216 | b | (g << 8) | (r << 16)).toString(16).slice(1);
+ });
+
+ /*\
+ * Raphael.getColor
+ [ method ]
+ **
+ * On each call returns next colour in the spectrum. To reset it back to red call @Raphael.getColor.reset
+ > Parameters
+ - value (number) #optional brightness, default is `0.75`
+ = (string) hex representation of the colour.
+ \*/
+ R.getColor = function(value) {
+ var start = this.getColor.start = this.getColor.start || {
+ h: 0,
+ s: 1,
+ b: value || .75
+ },
+ rgb = this.hsb2rgb(start.h, start.s, start.b);
+ start.h += .075;
+ if (start.h > 1) {
+ start.h = 0;
+ start.s -= .2;
+ start.s <= 0 && (this.getColor.start = {
+ h: 0,
+ s: 1,
+ b: start.b
+ });
+ }
+ return rgb.hex;
+ };
+
+ /*\
+ * Raphael.getColor.reset
+ [ method ]
+ **
+ * Resets spectrum position for @Raphael.getColor back to red.
+ \*/
+ R.getColor.reset = function() {
+ delete this.start;
+ };
+
+ // http://schepers.cc/getting-to-the-point
+ function catmullRom2bezier(crp, z) {
+ var d = [];
+ for (var i = 0, iLen = crp.length; iLen - 2 * !z > i; i += 2) {
+ var p = [
+ {
+ x: +crp[i - 2],
+ y: +crp[i - 1]
+ },
+ {
+ x: +crp[i],
+ y: +crp[i + 1]
+ },
+ {
+ x: +crp[i + 2],
+ y: +crp[i + 3]
+ },
+ {
+ x: +crp[i + 4],
+ y: +crp[i + 5]
+ }
+ ];
+ if (z) {
+ if (!i) {
+ p[0] = {
+ x: +crp[iLen - 2],
+ y: +crp[iLen - 1]
+ };
+ } else if (iLen - 4 == i) {
+ p[3] = {
+ x: +crp[0],
+ y: +crp[1]
+ };
+ } else if (iLen - 2 == i) {
+ p[2] = {
+ x: +crp[0],
+ y: +crp[1]
+ };
+ p[3] = {
+ x: +crp[2],
+ y: +crp[3]
+ };
+ }
+ } else {
+ if (iLen - 4 == i) {
+ p[3] = p[2];
+ } else if (!i) {
+ p[0] = {
+ x: +crp[i],
+ y: +crp[i + 1]
+ };
+ }
+ }
+ d.push(["C",
+ (-p[0].x + 6 * p[1].x + p[2].x) / 6,
+ (-p[0].y + 6 * p[1].y + p[2].y) / 6,
+ (p[1].x + 6 * p[2].x - p[3].x) / 6,
+ (p[1].y + 6 * p[2].y - p[3].y) / 6,
+ p[2].x,
+ p[2].y
+ ]);
+ }
+
+ return d;
+ }
+
+ /*\
+ * Raphael.parsePathString
+ [ method ]
+ **
+ * Utility method
+ **
+ * Parses given path string into an array of arrays of path segments.
+ > Parameters
+ - pathString (string|array) path string or array of segments (in the last case it will be returned straight away)
+ = (array) array of segments.
+ \*/
+ R.parsePathString = function(pathString) {
+ if (!pathString) {
+ return null;
+ }
+ var pth = paths(pathString);
+ if (pth.arr) {
+ return pathClone(pth.arr);
+ }
+
+ var paramCounts = {
+ a: 7,
+ c: 6,
+ h: 1,
+ l: 2,
+ m: 2,
+ r: 4,
+ q: 4,
+ s: 4,
+ t: 2,
+ v: 1,
+ z: 0
+ },
+ data = [];
+ if (R.is(pathString, array) && R.is(pathString[0], array)) { // rough assumption
+ data = pathClone(pathString);
+ }
+ if (!data.length) {
+ Str(pathString).replace(pathCommand, function(a, b, c) {
+ var params = [],
+ name = b.toLowerCase();
+ c.replace(pathValues, function(a, b) {
+ b && params.push(+b);
+ });
+ if (name == "m" && params.length > 2) {
+ data.push([b][concat](params.splice(0, 2)));
+ name = "l";
+ b = b == "m" ? "l" : "L";
+ }
+ if (name == "r") {
+ data.push([b][concat](params));
+ } else
+ while (params.length >= paramCounts[name]) {
+ data.push([b][concat](params.splice(0, paramCounts[name])));
+ if (!paramCounts[name]) {
+ break;
+ }
+ }
+ });
+ }
+ data.toString = R._path2string;
+ pth.arr = pathClone(data);
+ return data;
+ };
+
+ /*\
+ * Raphael.parseTransformString
+ [ method ]
+ **
+ * Utility method
+ **
+ * Parses given path string into an array of transformations.
+ > Parameters
+ - TString (string|array) transform string or array of transformations (in the last case it will be returned straight away)
+ = (array) array of transformations.
+ \*/
+ R.parseTransformString = cacher(function(TString) {
+ if (!TString) {
+ return null;
+ }
+ var paramCounts = {
+ r: 3,
+ s: 4,
+ t: 2,
+ m: 6
+ },
+ data = [];
+ if (R.is(TString, array) && R.is(TString[0], array)) { // rough assumption
+ data = pathClone(TString);
+ }
+ if (!data.length) {
+ Str(TString).replace(tCommand, function(a, b, c) {
+ var params = [],
+ name = lowerCase.call(b);
+ c.replace(pathValues, function(a, b) {
+ b && params.push(+b);
+ });
+ data.push([b][concat](params));
+ });
+ }
+ data.toString = R._path2string;
+ return data;
+ });
+ // PATHS
+ var paths = function(ps) {
+ var p = paths.ps = paths.ps || {};
+ if (p[ps]) {
+ p[ps].sleep = 100;
+ } else {
+ p[ps] = {
+ sleep: 100
+ };
+ }
+ setTimeout(function() {
+ for (var key in p)
+ if (p[has](key) && key != ps) {
+ p[key].sleep--;
+ !p[key].sleep && delete p[key];
+ }
+ });
+ return p[ps];
+ };
+
+ /*\
+ * Raphael.findDotsAtSegment
+ [ method ]
+ **
+ * Utility method
+ **
+ * Find dot coordinates on the given cubic bezier curve at the given t.
+ > Parameters
+ - p1x (number) x of the first point of the curve
+ - p1y (number) y of the first point of the curve
+ - c1x (number) x of the first anchor of the curve
+ - c1y (number) y of the first anchor of the curve
+ - c2x (number) x of the second anchor of the curve
+ - c2y (number) y of the second anchor of the curve
+ - p2x (number) x of the second point of the curve
+ - p2y (number) y of the second point of the curve
+ - t (number) position on the curve (0..1)
+ = (object) point information in format:
+ o {
+ o x: (number) x coordinate of the point
+ o y: (number) y coordinate of the point
+ o m: {
+ o x: (number) x coordinate of the left anchor
+ o y: (number) y coordinate of the left anchor
+ o }
+ o n: {
+ o x: (number) x coordinate of the right anchor
+ o y: (number) y coordinate of the right anchor
+ o }
+ o start: {
+ o x: (number) x coordinate of the start of the curve
+ o y: (number) y coordinate of the start of the curve
+ o }
+ o end: {
+ o x: (number) x coordinate of the end of the curve
+ o y: (number) y coordinate of the end of the curve
+ o }
+ o alpha: (number) angle of the curve derivative at the point
+ o }
+ \*/
+ R.findDotsAtSegment = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t) {
+ var t1 = 1 - t,
+ t13 = pow(t1, 3),
+ t12 = pow(t1, 2),
+ t2 = t * t,
+ t3 = t2 * t,
+ x = t13 * p1x + t12 * 3 * t * c1x + t1 * 3 * t * t * c2x + t3 * p2x,
+ y = t13 * p1y + t12 * 3 * t * c1y + t1 * 3 * t * t * c2y + t3 * p2y,
+ mx = p1x + 2 * t * (c1x - p1x) + t2 * (c2x - 2 * c1x + p1x),
+ my = p1y + 2 * t * (c1y - p1y) + t2 * (c2y - 2 * c1y + p1y),
+ nx = c1x + 2 * t * (c2x - c1x) + t2 * (p2x - 2 * c2x + c1x),
+ ny = c1y + 2 * t * (c2y - c1y) + t2 * (p2y - 2 * c2y + c1y),
+ ax = t1 * p1x + t * c1x,
+ ay = t1 * p1y + t * c1y,
+ cx = t1 * c2x + t * p2x,
+ cy = t1 * c2y + t * p2y,
+ alpha = (90 - math.atan2(mx - nx, my - ny) * 180 / PI);
+ (mx > nx || my < ny) && (alpha += 180);
+ return {
+ x: x,
+ y: y,
+ m: {
+ x: mx,
+ y: my
+ },
+ n: {
+ x: nx,
+ y: ny
+ },
+ start: {
+ x: ax,
+ y: ay
+ },
+ end: {
+ x: cx,
+ y: cy
+ },
+ alpha: alpha
+ };
+ };
+
+ /*\
+ * Raphael.bezierBBox
+ [ method ]
+ **
+ * Utility method
+ **
+ * Return bounding box of a given cubic bezier curve
+ > Parameters
+ - p1x (number) x of the first point of the curve
+ - p1y (number) y of the first point of the curve
+ - c1x (number) x of the first anchor of the curve
+ - c1y (number) y of the first anchor of the curve
+ - c2x (number) x of the second anchor of the curve
+ - c2y (number) y of the second anchor of the curve
+ - p2x (number) x of the second point of the curve
+ - p2y (number) y of the second point of the curve
+ * or
+ - bez (array) array of six points for bezier curve
+ = (object) point information in format:
+ o {
+ o min: {
+ o x: (number) x coordinate of the left point
+ o y: (number) y coordinate of the top point
+ o }
+ o max: {
+ o x: (number) x coordinate of the right point
+ o y: (number) y coordinate of the bottom point
+ o }
+ o }
+ \*/
+ R.bezierBBox = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y) {
+ if (!R.is(p1x, "array")) {
+ p1x = [p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y];
+ }
+ var bbox = curveDim.apply(null, p1x);
+ return {
+ x: bbox.min.x,
+ y: bbox.min.y,
+ x2: bbox.max.x,
+ y2: bbox.max.y,
+ width: bbox.max.x - bbox.min.x,
+ height: bbox.max.y - bbox.min.y
+ };
+ };
+
+ /*\
+ * Raphael.isPointInsideBBox
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns `true` if given point is inside bounding boxes.
+ > Parameters
+ - bbox (string) bounding box
+ - x (string) x coordinate of the point
+ - y (string) y coordinate of the point
+ = (boolean) `true` if point inside
+ \*/
+ R.isPointInsideBBox = function(bbox, x, y) {
+ return x >= bbox.x && x <= bbox.x2 && y >= bbox.y && y <= bbox.y2;
+ };
+
+ /*\
+ * Raphael.isBBoxIntersect
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns `true` if two bounding boxes intersect
+ > Parameters
+ - bbox1 (string) first bounding box
+ - bbox2 (string) second bounding box
+ = (boolean) `true` if they intersect
+ \*/
+ R.isBBoxIntersect = function(bbox1, bbox2) {
+ var i = R.isPointInsideBBox;
+ return i(bbox2, bbox1.x, bbox1.y) ||
+ i(bbox2, bbox1.x2, bbox1.y) ||
+ i(bbox2, bbox1.x, bbox1.y2) ||
+ i(bbox2, bbox1.x2, bbox1.y2) ||
+ i(bbox1, bbox2.x, bbox2.y) ||
+ i(bbox1, bbox2.x2, bbox2.y) ||
+ i(bbox1, bbox2.x, bbox2.y2) ||
+ i(bbox1, bbox2.x2, bbox2.y2) ||
+ (bbox1.x < bbox2.x2 && bbox1.x > bbox2.x ||
+ bbox2.x < bbox1.x2 && bbox2.x > bbox1.x) &&
+ (bbox1.y < bbox2.y2 && bbox1.y > bbox2.y || bbox2.y < bbox1.y2 && bbox2.y > bbox1.y);
+ };
+
+ function base3(t, p1, p2, p3, p4) {
+ var t1 = -3 * p1 + 9 * p2 - 9 * p3 + 3 * p4,
+ t2 = t * t1 + 6 * p1 - 12 * p2 + 6 * p3;
+ return t * t2 - 3 * p1 + 3 * p2;
+ }
+
+ function bezlen(x1, y1, x2, y2, x3, y3, x4, y4, z) {
+ if (z == null) {
+ z = 1;
+ }
+ z = z > 1 ? 1 : z < 0 ? 0 : z;
+ var z2 = z / 2,
+ n = 12,
+ Tvalues = [-0.1252, 0.1252, -0.3678, 0.3678, -0.5873, 0.5873, -0.7699, 0.7699, -0.9041, 0.9041, -0.9816, 0.9816],
+ Cvalues = [0.2491, 0.2491, 0.2335, 0.2335, 0.2032, 0.2032, 0.1601, 0.1601, 0.1069, 0.1069, 0.0472, 0.0472],
+ sum = 0;
+ for (var i = 0; i < n; i++) {
+ var ct = z2 * Tvalues[i] + z2,
+ xbase = base3(ct, x1, x2, x3, x4),
+ ybase = base3(ct, y1, y2, y3, y4),
+ comb = xbase * xbase + ybase * ybase;
+ sum += Cvalues[i] * mathSqrt(comb);
+ }
+ return z2 * sum;
+ }
+
+ function getTatLen(x1, y1, x2, y2, x3, y3, x4, y4, ll) {
+ if (ll < 0 || bezlen(x1, y1, x2, y2, x3, y3, x4, y4) < ll) {
+ return;
+ }
+ var t = 1,
+ step = t / 2,
+ t2 = t - step,
+ l,
+ e = .01;
+ l = bezlen(x1, y1, x2, y2, x3, y3, x4, y4, t2);
+ while (abs(l - ll) > e) {
+ step /= 2;
+ t2 += (l < ll ? 1 : -1) * step;
+ l = bezlen(x1, y1, x2, y2, x3, y3, x4, y4, t2);
+ }
+ return t2;
+ }
+
+ function intersect(x1, y1, x2, y2, x3, y3, x4, y4) {
+ if (
+ mmax(x1, x2) < mmin(x3, x4) ||
+ mmin(x1, x2) > mmax(x3, x4) ||
+ mmax(y1, y2) < mmin(y3, y4) ||
+ mmin(y1, y2) > mmax(y3, y4)
+ ) {
+ return;
+ }
+ var nx = (x1 * y2 - y1 * x2) * (x3 - x4) - (x1 - x2) * (x3 * y4 - y3 * x4),
+ ny = (x1 * y2 - y1 * x2) * (y3 - y4) - (y1 - y2) * (x3 * y4 - y3 * x4),
+ denominator = (x1 - x2) * (y3 - y4) - (y1 - y2) * (x3 - x4);
+
+ if (!denominator) {
+ return;
+ }
+ var px = nx / denominator,
+ py = ny / denominator,
+ px2 = +px.toFixed(2),
+ py2 = + py.toFixed(2);
+ if (
+ px2 < +mmin(x1, x2).toFixed(2) ||
+ px2 > +mmax(x1, x2).toFixed(2) ||
+ px2 < +mmin(x3, x4).toFixed(2) ||
+ px2 > +mmax(x3, x4).toFixed(2) ||
+ py2 < +mmin(y1, y2).toFixed(2) ||
+ py2 > +mmax(y1, y2).toFixed(2) ||
+ py2 < +mmin(y3, y4).toFixed(2) ||
+ py2 > +mmax(y3, y4).toFixed(2)
+ ) {
+ return;
+ }
+ return {
+ x: px,
+ y: py
+ };
+ }
+
+ function inter(bez1, bez2) {
+ return interHelper(bez1, bez2);
+ }
+
+ function interCount(bez1, bez2) {
+ return interHelper(bez1, bez2, 1);
+ }
+
+ function interHelper(bez1, bez2, justCount) {
+ var bbox1 = R.bezierBBox(bez1),
+ bbox2 = R.bezierBBox(bez2);
+
+ if (!R.isBBoxIntersect(bbox1, bbox2)) {
+ return justCount ? 0 : [];
+ }
+ var l1 = bezlen.apply(0, bez1),
+ l2 = bezlen.apply(0, bez2),
+ n1 = mmax(~~(l1 / 5), 1),
+ n2 = mmax(~~(l2 / 5), 1),
+ dots1 = [],
+ dots2 = [],
+ xy = {},
+ res = justCount ? 0 : [];
+
+ for (var i = 0; i < n1 + 1; i++) {
+ var p = R.findDotsAtSegment.apply(R, bez1.concat(i / n1));
+ dots1.push({
+ x: p.x,
+ y: p.y,
+ t: i / n1
+ });
+ }
+ for (i = 0; i < n2 + 1; i++) {
+ p = R.findDotsAtSegment.apply(R, bez2.concat(i / n2));
+ dots2.push({
+ x: p.x,
+ y: p.y,
+ t: i / n2
+ });
+ }
+ for (i = 0; i < n1; i++) {
+ for (var j = 0; j < n2; j++) {
+ var di = dots1[i],
+ di1 = dots1[i + 1],
+ dj = dots2[j],
+ dj1 = dots2[j + 1],
+ ci = abs(di1.x - di.x) < .001 ? "y" : "x",
+ cj = abs(dj1.x - dj.x) < .001 ? "y" : "x",
+ is = intersect(di.x, di.y, di1.x, di1.y, dj.x, dj.y, dj1.x, dj1.y);
+ if (is) {
+ if (xy[is.x.toFixed(4)] == is.y.toFixed(4)) {
+ continue;
+ }
+ xy[is.x.toFixed(4)] = is.y.toFixed(4);
+ var t1 = di.t + abs((is[ci] - di[ci]) / (di1[ci] - di[ci])) * (di1.t - di.t),
+ t2 = dj.t + abs((is[cj] - dj[cj]) / (dj1[cj] - dj[cj])) * (dj1.t - dj.t);
+ if (t1 >= 0 && t1 <= 1.001 && t2 >= 0 && t2 <= 1.001) {
+ if (justCount) {
+ res++;
+ } else {
+ res.push({
+ x: is.x,
+ y: is.y,
+ t1: mmin(t1, 1),
+ t2: mmin(t2, 1)
+ });
+ }
+ }
+ }
+ }
+ }
+ return res;
+ }
+
+ /*\
+ * Raphael.pathIntersection
+ [ method ]
+ **
+ * Utility method
+ **
+ * Finds intersections of two paths
+ > Parameters
+ - path1 (string) path string
+ - path2 (string) path string
+ = (array) dots of intersection
+ o [
+ o {
+ o x: (number) x coordinate of the point
+ o y: (number) y coordinate of the point
+ o t1: (number) t value for segment of path1
+ o t2: (number) t value for segment of path2
+ o segment1: (number) order number for segment of path1
+ o segment2: (number) order number for segment of path2
+ o bez1: (array) eight coordinates representing beziér curve for the segment of path1
+ o bez2: (array) eight coordinates representing beziér curve for the segment of path2
+ o }
+ o ]
+ \*/
+ R.pathIntersection = function(path1, path2) {
+ return interPathHelper(path1, path2);
+ };
+ R.pathIntersectionNumber = function(path1, path2) {
+ return interPathHelper(path1, path2, 1);
+ };
+ function interPathHelper(path1, path2, justCount) {
+ path1 = R._path2curve(path1);
+ path2 = R._path2curve(path2);
+ var x1, y1, x2, y2, x1m, y1m, x2m, y2m, bez1, bez2,
+ res = justCount ? 0 : [];
+ for (var i = 0, ii = path1.length; i < ii; i++) {
+ var pi = path1[i];
+ if (pi[0] == "M") {
+ x1 = x1m = pi[1];
+ y1 = y1m = pi[2];
+ } else {
+ if (pi[0] == "C") {
+ bez1 = [x1, y1].concat(pi.slice(1));
+ x1 = bez1[6];
+ y1 = bez1[7];
+ } else {
+ bez1 = [x1, y1, x1, y1, x1m, y1m, x1m, y1m];
+ x1 = x1m;
+ y1 = y1m;
+ }
+ for (var j = 0, jj = path2.length; j < jj; j++) {
+ var pj = path2[j];
+ if (pj[0] == "M") {
+ x2 = x2m = pj[1];
+ y2 = y2m = pj[2];
+ } else {
+ if (pj[0] == "C") {
+ bez2 = [x2, y2].concat(pj.slice(1));
+ x2 = bez2[6];
+ y2 = bez2[7];
+ } else {
+ bez2 = [x2, y2, x2, y2, x2m, y2m, x2m, y2m];
+ x2 = x2m;
+ y2 = y2m;
+ }
+ var intr = interHelper(bez1, bez2, justCount);
+ if (justCount) {
+ res += intr;
+ } else {
+ for (var k = 0, kk = intr.length; k < kk; k++) {
+ intr[k].segment1 = i;
+ intr[k].segment2 = j;
+ intr[k].bez1 = bez1;
+ intr[k].bez2 = bez2;
+ }
+ res = res.concat(intr);
+ }
+ }
+ }
+ }
+ }
+ return res;
+ }
+
+ /*\
+ * Raphael.isPointInsidePath
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns `true` if given point is inside a given closed path.
+ > Parameters
+ - path (string) path string
+ - x (number) x of the point
+ - y (number) y of the point
+ = (boolean) true, if point is inside the path
+ \*/
+ R.isPointInsidePath = function(path, x, y) {
+ var bbox = R.pathBBox(path);
+ return R.isPointInsideBBox(bbox, x, y) &&
+ ((interPathHelper(path, [["M", x, y], ["H", bbox.x2 + 10]], 1) % 2 == 1) ||
+ (interPathHelper(path, [["M", x, y], ["V", bbox.y2 + 10]], 1) % 2 == 1))
+ };
+ R._removedFactory = function(methodname) {
+ return function() {
+ eve("raphael.log", null, "Rapha\xebl: you are calling to method \u201c" + methodname + "\u201d of removed object", methodname);
+ };
+ };
+
+ /*\
+ * Raphael.pathBBox
+ [ method ]
+ **
+ * Utility method
+ **
+ * Return bounding box of a given path
+ > Parameters
+ - path (string) path string
+ = (object) bounding box
+ o {
+ o x: (number) x coordinate of the left top point of the box
+ o y: (number) y coordinate of the left top point of the box
+ o x2: (number) x coordinate of the right bottom point of the box
+ o y2: (number) y coordinate of the right bottom point of the box
+ o width: (number) width of the box
+ o height: (number) height of the box
+ o cx: (number) x coordinate of the center of the box
+ o cy: (number) y coordinate of the center of the box
+ o }
+ \*/
+ var pathDimensions = R.pathBBox = function(path) {
+ var pth = paths(path);
+ if (pth.bbox) {
+ return pth.bbox;
+ }
+ if (!path) {
+ return {
+ x: 0,
+ y: 0,
+ width: 0,
+ height: 0,
+ x2: 0,
+ y2: 0
+ };
+ }
+ path = path2curve(path);
+ var x = 0,
+ y = 0,
+ X = [],
+ Y = [],
+ p;
+ for (var i = 0, ii = path.length; i < ii; i++) {
+ p = path[i];
+ if (p[0] == "M") {
+ x = p[1];
+ y = p[2];
+ X.push(x);
+ Y.push(y);
+ } else {
+ var dim = curveDim(x, y, p[1], p[2], p[3], p[4], p[5], p[6]);
+ X = X[concat](dim.min.x, dim.max.x);
+ Y = Y[concat](dim.min.y, dim.max.y);
+ x = p[5];
+ y = p[6];
+ }
+ }
+ var xmin = mmin[apply](0, X),
+ ymin = mmin[apply](0, Y),
+ xmax = mmax[apply](0, X),
+ ymax = mmax[apply](0, Y),
+ bb = {
+ x: xmin,
+ y: ymin,
+ x2: xmax,
+ y2: ymax,
+ width: xmax - xmin,
+ height: ymax - ymin
+ };
+ pth.bbox = clone(bb);
+ return bb;
+ },
+ pathClone = function(pathArray) {
+ var res = clone(pathArray);
+ res.toString = R._path2string;
+ return res;
+ },
+ pathToRelative = R._pathToRelative = function(pathArray) {
+ var pth = paths(pathArray);
+ if (pth.rel) {
+ return pathClone(pth.rel);
+ }
+ if (!R.is(pathArray, array) || !R.is(pathArray && pathArray[0], array)) { // rough assumption
+ pathArray = R.parsePathString(pathArray);
+ }
+ var res = [],
+ x = 0,
+ y = 0,
+ mx = 0,
+ my = 0,
+ start = 0;
+ if (pathArray[0][0] == "M") {
+ x = pathArray[0][1];
+ y = pathArray[0][2];
+ mx = x;
+ my = y;
+ start++;
+ res.push(["M", x, y]);
+ }
+ for (var i = start, ii = pathArray.length; i < ii; i++) {
+ var r = res[i] = [],
+ pa = pathArray[i];
+ if (pa[0] != lowerCase.call(pa[0])) {
+ r[0] = lowerCase.call(pa[0]);
+ switch (r[0]) {
+ case "a":
+ r[1] = pa[1];
+ r[2] = pa[2];
+ r[3] = pa[3];
+ r[4] = pa[4];
+ r[5] = pa[5];
+ r[6] = +(pa[6] - x).toFixed(3);
+ r[7] = +(pa[7] - y).toFixed(3);
+ break;
+ case "v":
+ r[1] = +(pa[1] - y).toFixed(3);
+ break;
+ case "m":
+ mx = pa[1];
+ my = pa[2];
+ default:
+ for (var j = 1, jj = pa.length; j < jj; j++) {
+ r[j] = +(pa[j] - ((j % 2) ? x : y)).toFixed(3);
+ }
+ }
+ } else {
+ r = res[i] = [];
+ if (pa[0] == "m") {
+ mx = pa[1] + x;
+ my = pa[2] + y;
+ }
+ for (var k = 0, kk = pa.length; k < kk; k++) {
+ res[i][k] = pa[k];
+ }
+ }
+ var len = res[i].length;
+ switch (res[i][0]) {
+ case "z":
+ x = mx;
+ y = my;
+ break;
+ case "h":
+ x += +res[i][len - 1];
+ break;
+ case "v":
+ y += +res[i][len - 1];
+ break;
+ default:
+ x += +res[i][len - 2];
+ y += +res[i][len - 1];
+ }
+ }
+ res.toString = R._path2string;
+ pth.rel = pathClone(res);
+ return res;
+ },
+ pathToAbsolute = R._pathToAbsolute = function(pathArray) {
+ var pth = paths(pathArray), res;
+ if (pth.abs) {
+ return pathClone(pth.abs);
+ }
+ if (!R.is(pathArray, array) || !R.is(pathArray && pathArray[0], array)) { // rough assumption
+ pathArray = R.parsePathString(pathArray);
+ }
+ if (!pathArray || !pathArray.length) {
+ res = ["M", 0, 0];
+ res.toString = R._path2string;
+ return res;
+ }
+ var x = 0,
+ y = 0,
+ mx = 0,
+ my = 0,
+ start = 0;
+ res = [];
+ if (pathArray[0][0] == "M") {
+ x = +pathArray[0][1];
+ y = +pathArray[0][2];
+ mx = x;
+ my = y;
+ start++;
+ res[0] = ["M", x, y];
+ }
+ var crz = pathArray.length == 3 && pathArray[0][0] == "M" && pathArray[1][0].toUpperCase() == "R" && pathArray[2][0].toUpperCase() == "Z";
+ for (var r, pa, i = start, ii = pathArray.length; i < ii; i++) {
+ res.push(r = []);
+ pa = pathArray[i];
+ if (pa[0] != upperCase.call(pa[0])) {
+ r[0] = upperCase.call(pa[0]);
+ switch (r[0]) {
+ case "A":
+ r[1] = pa[1];
+ r[2] = pa[2];
+ r[3] = pa[3];
+ r[4] = pa[4];
+ r[5] = pa[5];
+ r[6] = +(pa[6] + x);
+ r[7] = +(pa[7] + y);
+ break;
+ case "V":
+ r[1] = +pa[1] + y;
+ break;
+ case "H":
+ r[1] = +pa[1] + x;
+ break;
+ case "R":
+ var dots = [x, y][concat](pa.slice(1));
+ for (var j = 2, jj = dots.length; j < jj; j++) {
+ dots[j] = +dots[j] + x;
+ dots[++j] = +dots[j] + y;
+ }
+ res.pop();
+ res = res[concat](catmullRom2bezier(dots, crz));
+ break;
+ case "M":
+ mx = +pa[1] + x;
+ my = +pa[2] + y;
+ default:
+ for (j = 1, jj = pa.length; j < jj; j++) {
+ r[j] = +pa[j] + ((j % 2) ? x : y);
+ }
+ }
+ } else if (pa[0] == "R") {
+ dots = [x, y][concat](pa.slice(1));
+ res.pop();
+ res = res[concat](catmullRom2bezier(dots, crz));
+ r = ["R"][concat](pa.slice(-2));
+ } else {
+ for (var k = 0, kk = pa.length; k < kk; k++) {
+ r[k] = pa[k];
+ }
+ }
+ switch (r[0]) {
+ case "Z":
+ x = mx;
+ y = my;
+ break;
+ case "H":
+ x = r[1];
+ break;
+ case "V":
+ y = r[1];
+ break;
+ case "M":
+ mx = r[r.length - 2];
+ my = r[r.length - 1];
+ default:
+ x = r[r.length - 2];
+ y = r[r.length - 1];
+ }
+ }
+ res.toString = R._path2string;
+ pth.abs = pathClone(res);
+ return res;
+ },
+ l2c = function(x1, y1, x2, y2) {
+ return [x1, y1, x2, y2, x2, y2];
+ },
+ q2c = function(x1, y1, ax, ay, x2, y2) {
+ var _13 = 1 / 3,
+ _23 = 2 / 3;
+ return [
+ _13 * x1 + _23 * ax,
+ _13 * y1 + _23 * ay,
+ _13 * x2 + _23 * ax,
+ _13 * y2 + _23 * ay,
+ x2,
+ y2
+ ];
+ },
+ a2c = function(x1, y1, rx, ry, angle, large_arc_flag, sweep_flag, x2, y2, recursive) {
+ // for more information of where this math came from visit:
+ // http://www.w3.org/TR/SVG11/implnote.html#ArcImplementationNotes
+ var _120 = PI * 120 / 180,
+ rad = deg2rad * (+angle || 0),
+ res = [],
+ xy,
+ rotate = cacher(function(x, y, rad) {
+ var X = x * mathCos(rad) - y * mathSin(rad),
+ Y = x * mathSin(rad) + y * mathCos(rad);
+ return {
+ x: X,
+ y: Y
+ };
+ });
+ if (!recursive) {
+ xy = rotate(x1, y1, -rad);
+ x1 = xy.x;
+ y1 = xy.y;
+ xy = rotate(x2, y2, -rad);
+ x2 = xy.x;
+ y2 = xy.y;
+ var cos = mathCos(deg2rad * angle),
+ sin = mathSin(deg2rad * angle),
+ x = (x1 - x2) / 2,
+ y = (y1 - y2) / 2;
+ var h = (x * x) / (rx * rx) + (y * y) / (ry * ry);
+ if (h > 1) {
+ h = mathSqrt(h);
+ rx = h * rx;
+ ry = h * ry;
+ }
+ var rx2 = rx * rx,
+ ry2 = ry * ry,
+ k = (large_arc_flag == sweep_flag ? -1 : 1) *
+ mathSqrt(abs((rx2 * ry2 - rx2 * y * y - ry2 * x * x) / (rx2 * y * y + ry2 * x * x))),
+ cx = k * rx * y / ry + (x1 + x2) / 2,
+ cy = k * -ry * x / rx + (y1 + y2) / 2,
+ f1 = math.asin(((y1 - cy) / ry).toFixed(9)),
+ f2 = math.asin(((y2 - cy) / ry).toFixed(9));
+
+ f1 = x1 < cx ? PI - f1 : f1;
+ f2 = x2 < cx ? PI - f2 : f2;
+ f1 < 0 && (f1 = PI * 2 + f1);
+ f2 < 0 && (f2 = PI * 2 + f2);
+ if (sweep_flag && f1 > f2) {
+ f1 = f1 - PI * 2;
+ }
+ if (!sweep_flag && f2 > f1) {
+ f2 = f2 - PI * 2;
+ }
+ } else {
+ f1 = recursive[0];
+ f2 = recursive[1];
+ cx = recursive[2];
+ cy = recursive[3];
+ }
+ var df = f2 - f1;
+ if (abs(df) > _120) {
+ var f2old = f2,
+ x2old = x2,
+ y2old = y2;
+ f2 = f1 + _120 * (sweep_flag && f2 > f1 ? 1 : -1);
+ x2 = cx + rx * mathCos(f2);
+ y2 = cy + ry * mathSin(f2);
+ res = a2c(x2, y2, rx, ry, angle, 0, sweep_flag, x2old, y2old, [f2, f2old, cx, cy]);
+ }
+ df = f2 - f1;
+ var c1 = mathCos(f1),
+ s1 = mathSin(f1),
+ c2 = mathCos(f2),
+ s2 = mathSin(f2),
+ t = math.tan(df / 4),
+ hx = 4 / 3 * rx * t,
+ hy = 4 / 3 * ry * t,
+ m1 = [x1, y1],
+ m2 = [x1 + hx * s1, y1 - hy * c1],
+ m3 = [x2 + hx * s2, y2 - hy * c2],
+ m4 = [x2, y2];
+ m2[0] = 2 * m1[0] - m2[0];
+ m2[1] = 2 * m1[1] - m2[1];
+ if (recursive) {
+ return [m2, m3, m4][concat](res);
+ } else {
+ res = [m2, m3, m4][concat](res).join()[split](",");
+ var newres = [];
+ for (var i = 0, ii = res.length; i < ii; i++) {
+ newres[i] = i % 2 ? rotate(res[i - 1], res[i], rad).y : rotate(res[i], res[i + 1], rad).x;
+ }
+ return newres;
+ }
+ },
+ findDotAtSegment = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t) {
+ var t1 = 1 - t;
+ return {
+ x: pow(t1, 3) * p1x + pow(t1, 2) * 3 * t * c1x + t1 * 3 * t * t * c2x + pow(t, 3) * p2x,
+ y: pow(t1, 3) * p1y + pow(t1, 2) * 3 * t * c1y + t1 * 3 * t * t * c2y + pow(t, 3) * p2y
+ };
+ },
+ curveDim = cacher(function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y) {
+ var a = (c2x - 2 * c1x + p1x) - (p2x - 2 * c2x + c1x),
+ b = 2 * (c1x - p1x) - 2 * (c2x - c1x),
+ c = p1x - c1x,
+ t1 = (-b + mathSqrt(b * b - 4 * a * c)) / 2 / a,
+ t2 = (-b - mathSqrt(b * b - 4 * a * c)) / 2 / a,
+ y = [p1y, p2y],
+ x = [p1x, p2x],
+ dot;
+ abs(t1) > "1e12" && (t1 = .5);
+ abs(t2) > "1e12" && (t2 = .5);
+ if (t1 > 0 && t1 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t1);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ if (t2 > 0 && t2 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t2);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ a = (c2y - 2 * c1y + p1y) - (p2y - 2 * c2y + c1y);
+ b = 2 * (c1y - p1y) - 2 * (c2y - c1y);
+ c = p1y - c1y;
+ t1 = (-b + mathSqrt(b * b - 4 * a * c)) / 2 / a;
+ t2 = (-b - mathSqrt(b * b - 4 * a * c)) / 2 / a;
+ abs(t1) > "1e12" && (t1 = .5);
+ abs(t2) > "1e12" && (t2 = .5);
+ if (t1 > 0 && t1 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t1);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ if (t2 > 0 && t2 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t2);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ return {
+ min: {
+ x: mmin[apply](0, x),
+ y: mmin[apply](0, y)
+ },
+ max: {
+ x: mmax[apply](0, x),
+ y: mmax[apply](0, y)
+ }
+ };
+ }),
+ path2curve = R._path2curve = cacher(function(path, path2) {
+ var pth = !path2 && paths(path);
+ if (!path2 && pth.curve) {
+ return pathClone(pth.curve);
+ }
+ var p = pathToAbsolute(path),
+ p2 = path2 && pathToAbsolute(path2),
+ attrs = {
+ x: 0,
+ y: 0,
+ bx: 0,
+ by: 0,
+ X: 0,
+ Y: 0,
+ qx: null,
+ qy: null
+ },
+ attrs2 = {
+ x: 0,
+ y: 0,
+ bx: 0,
+ by: 0,
+ X: 0,
+ Y: 0,
+ qx: null,
+ qy: null
+ },
+ processPath = function(path, d) {
+ var nx, ny;
+ if (!path) {
+ return ["C", d.x, d.y, d.x, d.y, d.x, d.y];
+ }
+ !(path[0] in {
+ T: 1,
+ Q: 1
+ }) && (d.qx = d.qy = null);
+ switch (path[0]) {
+ case "M":
+ d.X = path[1];
+ d.Y = path[2];
+ break;
+ case "A":
+ path = ["C"][concat](a2c[apply](0, [d.x, d.y][concat](path.slice(1))));
+ break;
+ case "S":
+ nx = d.x + (d.x - (d.bx || d.x));
+ ny = d.y + (d.y - (d.by || d.y));
+ path = ["C", nx, ny][concat](path.slice(1));
+ break;
+ case "T":
+ d.qx = d.x + (d.x - (d.qx || d.x));
+ d.qy = d.y + (d.y - (d.qy || d.y));
+ path = ["C"][concat](q2c(d.x, d.y, d.qx, d.qy, path[1], path[2]));
+ break;
+ case "Q":
+ d.qx = path[1];
+ d.qy = path[2];
+ path = ["C"][concat](q2c(d.x, d.y, path[1], path[2], path[3], path[4]));
+ break;
+ case "L":
+ path = ["C"][concat](l2c(d.x, d.y, path[1], path[2]));
+ break;
+ case "H":
+ path = ["C"][concat](l2c(d.x, d.y, path[1], d.y));
+ break;
+ case "V":
+ path = ["C"][concat](l2c(d.x, d.y, d.x, path[1]));
+ break;
+ case "Z":
+ path = ["C"][concat](l2c(d.x, d.y, d.X, d.Y));
+ break;
+ }
+ return path;
+ },
+ fixArc = function(pp, i) {
+ if (pp[i].length > 7) {
+ pp[i].shift();
+ var pi = pp[i];
+ while (pi.length) {
+ pp.splice(i++, 0, ["C"][concat](pi.splice(0, 6)));
+ }
+ pp.splice(i, 1);
+ ii = mmax(p.length, p2 && p2.length || 0);
+ }
+ },
+ fixM = function(path1, path2, a1, a2, i) {
+ if (path1 && path2 && path1[i][0] == "M" && path2[i][0] != "M") {
+ path2.splice(i, 0, ["M", a2.x, a2.y]);
+ a1.bx = 0;
+ a1.by = 0;
+ a1.x = path1[i][1];
+ a1.y = path1[i][2];
+ ii = mmax(p.length, p2 && p2.length || 0);
+ }
+ };
+ for (var i = 0, ii = mmax(p.length, p2 && p2.length || 0); i < ii; i++) {
+ p[i] = processPath(p[i], attrs);
+ fixArc(p, i);
+ p2 && (p2[i] = processPath(p2[i], attrs2));
+ p2 && fixArc(p2, i);
+ fixM(p, p2, attrs, attrs2, i);
+ fixM(p2, p, attrs2, attrs, i);
+ var seg = p[i],
+ seg2 = p2 && p2[i],
+ seglen = seg.length,
+ seg2len = p2 && seg2.length;
+ attrs.x = seg[seglen - 2];
+ attrs.y = seg[seglen - 1];
+ attrs.bx = toFloat(seg[seglen - 4]) || attrs.x;
+ attrs.by = toFloat(seg[seglen - 3]) || attrs.y;
+ attrs2.bx = p2 && (toFloat(seg2[seg2len - 4]) || attrs2.x);
+ attrs2.by = p2 && (toFloat(seg2[seg2len - 3]) || attrs2.y);
+ attrs2.x = p2 && seg2[seg2len - 2];
+ attrs2.y = p2 && seg2[seg2len - 1];
+ }
+ if (!p2) {
+ pth.curve = pathClone(p);
+ }
+ return p2 ? [p, p2] : p;
+ }, null, pathClone),
+ parseDots = R._parseDots = cacher(function(gradient) {
+ var dots = [];
+ for (var i = 0, ii = gradient.length; i < ii; i++) {
+ var dot = {},
+ par = gradient[i].match(/^([^:]*):?([\d\.]*)/);
+ dot.color = R.getRGB(par[1]);
+ if (dot.color.error) {
+ return null;
+ }
+ //store opacity information
+ dot.opacity = dot.color.opacity;
+ dot.color = dot.color.hex;
+ par[2] && (dot.offset = par[2] + "%");
+ dots.push(dot);
+ }
+ for (i = 1, ii = dots.length - 1; i < ii; i++) {
+ if (!dots[i].offset) {
+ var start = toFloat(dots[i - 1].offset || 0),
+ end = 0;
+ for (var j = i + 1; j < ii; j++) {
+ if (dots[j].offset) {
+ end = dots[j].offset;
+ break;
+ }
+ }
+ if (!end) {
+ end = 100;
+ j = ii;
+ }
+ end = toFloat(end);
+ var d = (end - start) / (j - i + 1);
+ for (; i < j; i++) {
+ start += d;
+ dots[i].offset = start + "%";
+ }
+ }
+ }
+ return dots;
+ }),
+ tear = R._tear = function(el, paper) {
+ el == paper.top && (paper.top = el.prev);
+ el == paper.bottom && (paper.bottom = el.next);
+ el.next && (el.next.prev = el.prev);
+ el.prev && (el.prev.next = el.next);
+ },
+ tofront = R._tofront = function(el, paper) {
+ if (paper.top === el) {
+ return false;
+ }
+ tear(el, paper);
+ el.next = null;
+ el.prev = paper.top;
+ paper.top.next = el;
+ paper.top = el;
+ return true;
+ },
+ toback = R._toback = function(el, paper) {
+ if (paper.bottom === el) {
+ return false;
+ }
+ tear(el, paper);
+ el.next = paper.bottom;
+ el.prev = null;
+ paper.bottom.prev = el;
+ paper.bottom = el;
+ return true;
+ },
+ insertafter = R._insertafter = function(el, el2, paper, paper2) {
+ tear(el, paper);
+ el.parent = paper2;
+ el2 === paper2.top && (paper2.top = el);
+ el2.next && (el2.next.prev = el);
+ el.next = el2.next;
+ el.prev = el2;
+ el2.next = el;
+ },
+ insertbefore = R._insertbefore = function(el, el2, paper, paper2) {
+ tear(el, paper);
+ el.parent = paper2;
+ el2 === paper2.bottom && (paper2.bottom = el);
+ el2.prev && (el2.prev.next = el);
+ el.prev = el2.prev;
+ el2.prev = el;
+ el.next = el2;
+ },
+
+ /*\
+ * Raphael.toMatrix
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns matrix of transformations applied to a given path
+ > Parameters
+ - path (string) path string
+ - transform (string|array) transformation string
+ = (object) @Matrix
+ \*/
+ toMatrix = R.toMatrix = function(path, transform) {
+ var bb = pathDimensions(path),
+ el = {
+ _: {
+ transform: E
+ },
+ getBBox: function() {
+ return bb;
+ }
+ };
+ extractTransform(el, transform);
+ return el.matrix;
+ },
+
+ /*\
+ * Raphael.transformPath
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns path transformed by a given transformation
+ > Parameters
+ - path (string) path string
+ - transform (string|array) transformation string
+ = (string) path
+ \*/
+ transformPath = R.transformPath = function(path, transform) {
+ return mapPath(path, toMatrix(path, transform));
+ },
+ extractTransform = R._extractTransform = function(el, tstr) {
+ if (tstr == null) {
+ return el._.transform;
+ }
+ tstr = Str(tstr).replace(/\.{3}|\u2026/g, el._.transform || E);
+ var tdata = R.parseTransformString(tstr),
+ deg = 0,
+ dx = 0,
+ dy = 0,
+ sx = 1,
+ sy = 1,
+ _ = el._,
+ m = new Matrix;
+ _.transform = tdata || [];
+ if (tdata) {
+ for (var i = 0, ii = tdata.length; i < ii; i++) {
+ var t = tdata[i],
+ tlen = t.length,
+ command = Str(t[0]).toLowerCase(),
+ absolute = t[0] != command,
+ inver = absolute ? m.invert() : 0,
+ x1,
+ y1,
+ x2,
+ y2,
+ bb;
+ if (command == "t" && tlen == 3) {
+ if (absolute) {
+ x1 = inver.x(0, 0);
+ y1 = inver.y(0, 0);
+ x2 = inver.x(t[1], t[2]);
+ y2 = inver.y(t[1], t[2]);
+ m.translate(x2 - x1, y2 - y1);
+ } else {
+ m.translate(t[1], t[2]);
+ }
+ } else if (command == "r") {
+ if (tlen == 2) {
+ bb = bb || el.getBBox(1);
+ m.rotate(t[1], bb.x + bb.width / 2, bb.y + bb.height / 2);
+ deg += t[1];
+ } else if (tlen == 4) {
+ if (absolute) {
+ x2 = inver.x(t[2], t[3]);
+ y2 = inver.y(t[2], t[3]);
+ m.rotate(t[1], x2, y2);
+ } else {
+ m.rotate(t[1], t[2], t[3]);
+ }
+ deg += t[1];
+ }
+ } else if (command == "s") {
+ if (tlen == 2 || tlen == 3) {
+ bb = bb || el.getBBox(1);
+ m.scale(t[1], t[tlen - 1], bb.x + bb.width / 2, bb.y + bb.height / 2);
+ sx *= t[1];
+ sy *= t[tlen - 1];
+ } else if (tlen == 5) {
+ if (absolute) {
+ x2 = inver.x(t[3], t[4]);
+ y2 = inver.y(t[3], t[4]);
+ m.scale(t[1], t[2], x2, y2);
+ } else {
+ m.scale(t[1], t[2], t[3], t[4]);
+ }
+ sx *= t[1];
+ sy *= t[2];
+ }
+ } else if (command == "m" && tlen == 7) {
+ m.add(t[1], t[2], t[3], t[4], t[5], t[6]);
+ }
+ _.dirtyT = 1;
+ el.matrix = m;
+ }
+ }
+
+ /*\
+ * Element.matrix
+ [ property (object) ]
+ **
+ * Keeps @Matrix object, which represents element transformation
+ \*/
+ el.matrix = m;
+
+ _.sx = sx;
+ _.sy = sy;
+ _.deg = deg;
+ _.dx = dx = m.e;
+ _.dy = dy = m.f;
+
+ if (sx == 1 && sy == 1 && !deg && _.bbox) {
+ _.bbox.x += +dx;
+ _.bbox.y += +dy;
+ } else {
+ _.dirtyT = 1;
+ }
+ },
+ getEmpty = function(item) {
+ var l = item[0];
+ switch (l.toLowerCase()) {
+ case "t":
+ return [l, 0, 0];
+ case "m":
+ return [l, 1, 0, 0, 1, 0, 0];
+ case "r":
+ if (item.length == 4) {
+ return [l, 0, item[2], item[3]];
+ } else {
+ return [l, 0];
+ }
+ case "s":
+ if (item.length == 5) {
+ return [l, 1, 1, item[3], item[4]];
+ } else if (item.length == 3) {
+ return [l, 1, 1];
+ } else {
+ return [l, 1];
+ }
+ }
+ },
+ equaliseTransform = R._equaliseTransform = function(t1, t2) {
+ t2 = Str(t2).replace(/\.{3}|\u2026/g, t1);
+ t1 = R.parseTransformString(t1) || [];
+ t2 = R.parseTransformString(t2) || [];
+ var maxlength = mmax(t1.length, t2.length),
+ from = [],
+ to = [],
+ i = 0, j, jj,
+ tt1, tt2;
+ for (; i < maxlength; i++) {
+ tt1 = t1[i] || getEmpty(t2[i]);
+ tt2 = t2[i] || getEmpty(tt1);
+ if ((tt1[0] != tt2[0]) ||
+ (tt1[0].toLowerCase() == "r" && (tt1[2] != tt2[2] || tt1[3] != tt2[3])) ||
+ (tt1[0].toLowerCase() == "s" && (tt1[3] != tt2[3] || tt1[4] != tt2[4]))
+ ) {
+ return;
+ }
+ from[i] = [];
+ to[i] = [];
+ for (j = 0, jj = mmax(tt1.length, tt2.length); j < jj; j++) {
+ j in tt1 && (from[i][j] = tt1[j]);
+ j in tt2 && (to[i][j] = tt2[j]);
+ }
+ }
+ return {
+ from: from,
+ to: to
+ };
+ };
+ R._getContainer = function(x, y, w, h) {
+ var container;
+ container = h == null && !R.is(x, object) ? g.doc.getElementById(x) : x;
+ if (container == null) {
+ return;
+ }
+ if (container.tagName) {
+ if (y == null) {
+ return {
+ container: container,
+ width: container.style.pixelWidth || container.offsetWidth,
+ height: container.style.pixelHeight || container.offsetHeight
+ };
+ } else {
+ return {
+ container: container,
+ width: y,
+ height: w
+ };
+ }
+ }
+ return {
+ container: 1,
+ x: x,
+ y: y,
+ width: w,
+ height: h
+ };
+ };
+
+ /*\
+ * Raphael.pathToRelative
+ [ method ]
+ **
+ * Utility method
+ **
+ * Converts path to relative form
+ > Parameters
+ - pathString (string|array) path string or array of segments
+ = (array) array of segments.
+ \*/
+ R.pathToRelative = pathToRelative;
+ R._engine = {};
+
+ /*\
+ * Raphael.path2curve
+ [ method ]
+ **
+ * Utility method
+ **
+ * Converts path to a new path where all segments are cubic bezier curves.
+ > Parameters
+ - pathString (string|array) path string or array of segments
+ = (array) array of segments.
+ \*/
+ R.path2curve = path2curve;
+
+ /*\
+ * Raphael.matrix
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns matrix based on given parameters.
+ > Parameters
+ - a (number)
+ - b (number)
+ - c (number)
+ - d (number)
+ - e (number)
+ - f (number)
+ = (object) @Matrix
+ \*/
+ R.matrix = function(a, b, c, d, e, f) {
+ return new Matrix(a, b, c, d, e, f);
+ };
+
+ function Matrix(a, b, c, d, e, f) {
+ if (a != null) {
+ this.a = +a;
+ this.b = +b;
+ this.c = +c;
+ this.d = +d;
+ this.e = +e;
+ this.f = +f;
+ } else {
+ this.a = 1;
+ this.b = 0;
+ this.c = 0;
+ this.d = 1;
+ this.e = 0;
+ this.f = 0;
+ }
+ }
+ (function(matrixproto) {
+
+ /*\
+ * Matrix.add
+ [ method ]
+ **
+ * Adds given matrix to existing one.
+ > Parameters
+ - a (number)
+ - b (number)
+ - c (number)
+ - d (number)
+ - e (number)
+ - f (number)
+ or
+ - matrix (object) @Matrix
+ \*/
+ matrixproto.add = function(a, b, c, d, e, f) {
+ var out = [[], [], []],
+ m = [[this.a, this.c, this.e], [this.b, this.d, this.f], [0, 0, 1]],
+ matrix = [[a, c, e], [b, d, f], [0, 0, 1]],
+ x, y, z, res;
+
+ if (a && a instanceof Matrix) {
+ matrix = [[a.a, a.c, a.e], [a.b, a.d, a.f], [0, 0, 1]];
+ }
+
+ for (x = 0; x < 3; x++) {
+ for (y = 0; y < 3; y++) {
+ res = 0;
+ for (z = 0; z < 3; z++) {
+ res += m[x][z] * matrix[z][y];
+ }
+ out[x][y] = res;
+ }
+ }
+ this.a = out[0][0];
+ this.b = out[1][0];
+ this.c = out[0][1];
+ this.d = out[1][1];
+ this.e = out[0][2];
+ this.f = out[1][2];
+ };
+
+ /*\
+ * Matrix.invert
+ [ method ]
+ **
+ * Returns inverted version of the matrix
+ = (object) @Matrix
+ \*/
+ matrixproto.invert = function() {
+ var me = this,
+ x = me.a * me.d - me.b * me.c;
+ return new Matrix(me.d / x, -me.b / x, -me.c / x, me.a / x, (me.c * me.f - me.d * me.e) / x, (me.b * me.e - me.a * me.f) / x);
+ };
+
+ /*\
+ * Matrix.clone
+ [ method ]
+ **
+ * Returns copy of the matrix
+ = (object) @Matrix
+ \*/
+ matrixproto.clone = function() {
+ return new Matrix(this.a, this.b, this.c, this.d, this.e, this.f);
+ };
+
+ /*\
+ * Matrix.translate
+ [ method ]
+ **
+ * Translate the matrix
+ > Parameters
+ - x (number)
+ - y (number)
+ \*/
+ matrixproto.translate = function(x, y) {
+ this.add(1, 0, 0, 1, x, y);
+ };
+
+ /*\
+ * Matrix.scale
+ [ method ]
+ **
+ * Scales the matrix
+ > Parameters
+ - x (number)
+ - y (number) #optional
+ - cx (number) #optional
+ - cy (number) #optional
+ \*/
+ matrixproto.scale = function(x, y, cx, cy) {
+ y == null && (y = x);
+ (cx || cy) && this.add(1, 0, 0, 1, cx, cy);
+ this.add(x, 0, 0, y, 0, 0);
+ (cx || cy) && this.add(1, 0, 0, 1, -cx, -cy);
+ };
+
+ /*\
+ * Matrix.rotate
+ [ method ]
+ **
+ * Rotates the matrix
+ > Parameters
+ - a (number)
+ - x (number)
+ - y (number)
+ \*/
+ matrixproto.rotate = function(a, x, y) {
+ a = R.rad(a);
+ x = x || 0;
+ y = y || 0;
+ var cos = +mathCos(a).toFixed(9),
+ sin = + mathSin(a).toFixed(9);
+ this.add(cos, sin, -sin, cos, x, y);
+ this.add(1, 0, 0, 1, -x, -y);
+ };
+
+ /*\
+ * Matrix.x
+ [ method ]
+ **
+ * Return x coordinate for given point after transformation described by the matrix. See also @Matrix.y
+ > Parameters
+ - x (number)
+ - y (number)
+ = (number) x
+ \*/
+ matrixproto.x = function(x, y) {
+ return x * this.a + y * this.c + this.e;
+ };
+
+ /*\
+ * Matrix.y
+ [ method ]
+ **
+ * Return y coordinate for given point after transformation described by the matrix. See also @Matrix.x
+ > Parameters
+ - x (number)
+ - y (number)
+ = (number) y
+ \*/
+ matrixproto.y = function(x, y) {
+ return x * this.b + y * this.d + this.f;
+ };
+ matrixproto.get = function(i) {
+ return +this[Str.fromCharCode(97 + i)].toFixed(4);
+ };
+ matrixproto.toString = function() {
+ return R.svg ?
+ "matrix(" + [this.get(0), this.get(1), this.get(2), this.get(3), this.get(4), this.get(5)].join() + ")" :
+ [this.get(0), this.get(2), this.get(1), this.get(3), 0, 0].join();
+ };
+ matrixproto.toMatrixString = function() {
+ return "matrix(" + [this.get(0), this.get(1), this.get(2), this.get(3), this.get(4), this.get(5)].join() + ")";
+ };
+ matrixproto.toFilter = function() {
+ return "progid:DXImageTransform.Microsoft.Matrix(M11=" + this.get(0) +
+ ", M12=" + this.get(2) + ", M21=" + this.get(1) + ", M22=" + this.get(3) +
+ ", Dx=" + this.get(4) + ", Dy=" + this.get(5) + ", sizingmethod='auto expand')";
+ };
+ matrixproto.offset = function() {
+ return [this.e.toFixed(4), this.f.toFixed(4)];
+ };
+ function norm(a) {
+ return a[0] * a[0] + a[1] * a[1];
+ }
+ function normalize(a) {
+ var mag = mathSqrt(norm(a));
+ a[0] && (a[0] /= mag);
+ a[1] && (a[1] /= mag);
+ }
+
+ /*\
+ * Matrix.split
+ [ method ]
+ **
+ * Splits matrix into primitive transformations
+ = (object) in format:
+ o dx (number) translation by x
+ o dy (number) translation by y
+ o scalex (number) scale by x
+ o scaley (number) scale by y
+ o shear (number) shear
+ o rotate (number) rotation in deg
+ o isSimple (boolean) could it be represented via simple transformations
+ \*/
+ matrixproto.split = function() {
+ var out = {};
+ // translation
+ out.dx = this.e;
+ out.dy = this.f;
+
+ // scale and shear
+ var row = [[this.a, this.c], [this.b, this.d]];
+ out.scalex = mathSqrt(norm(row[0]));
+ normalize(row[0]);
+
+ out.shear = row[0][0] * row[1][0] + row[0][1] * row[1][1];
+ row[1] = [row[1][0] - row[0][0] * out.shear, row[1][1] - row[0][1] * out.shear];
+
+ out.scaley = mathSqrt(norm(row[1]));
+ normalize(row[1]);
+ out.shear /= out.scaley;
+
+ // rotation
+ var sin = -row[0][1],
+ cos = row[1][1];
+ if (cos < 0) {
+ out.rotate = R.deg(math.acos(cos));
+ if (sin < 0) {
+ out.rotate = 360 - out.rotate;
+ }
+ } else {
+ out.rotate = R.deg(math.asin(sin));
+ }
+
+ out.isSimple = !+out.shear.toFixed(9) && (out.scalex.toFixed(9) == out.scaley.toFixed(9) || !out.rotate);
+ out.isSuperSimple = !+out.shear.toFixed(9) && out.scalex.toFixed(9) == out.scaley.toFixed(9) && !out.rotate;
+ out.noRotation = !+out.shear.toFixed(9) && !out.rotate;
+ return out;
+ };
+
+ /*\
+ * Matrix.toTransformString
+ [ method ]
+ **
+ * Return transform string that represents given matrix
+ = (string) transform string
+ \*/
+ matrixproto.toTransformString = function(shorter) {
+ var s = shorter || this[split]();
+ if (s.isSimple) {
+ s.scalex = +s.scalex.toFixed(4);
+ s.scaley = +s.scaley.toFixed(4);
+ s.rotate = +s.rotate.toFixed(4);
+ return (s.dx || s.dy ? "t" + [s.dx, s.dy] : E) +
+ (s.scalex != 1 || s.scaley != 1 ? "s" + [s.scalex, s.scaley, 0, 0] : E) +
+ (s.rotate ? "r" + [s.rotate, 0, 0] : E);
+ } else {
+ return "m" + [this.get(0), this.get(1), this.get(2), this.get(3), this.get(4), this.get(5)];
+ }
+ };
+ })(Matrix.prototype);
+
+ // WebKit rendering bug workaround method
+ var version = navigator.userAgent.match(/Version\/(.*?)\s/) || navigator.userAgent.match(/Chrome\/(\d+)/);
+ if ((navigator.vendor == "Apple Computer, Inc.") && (version && version[1] < 4 || navigator.platform.slice(0, 2) == "iP") ||
+ (navigator.vendor == "Google Inc." && version && version[1] < 8)) {
+
+ /*\
+ * Paper.safari
+ [ method ]
+ **
+ * There is an inconvenient rendering bug in Safari (WebKit):
+ * sometimes the rendering should be forced.
+ * This method should help with dealing with this bug.
+ \*/
+ paperproto.safari = function() {
+ var rect = this.rect(-99, -99, this.width + 99, this.height + 99).attr({
+ stroke: "none"
+ });
+ setTimeout(function() {
+ rect.remove();
+ });
+ return true;
+ };
+ } else {
+ paperproto.safari = fun;
+ }
+
+ var preventDefault = function() {
+ this.returnValue = false;
+ },
+ preventTouch = function() {
+ return this.originalEvent.preventDefault();
+ },
+ stopPropagation = function() {
+ this.cancelBubble = true;
+ },
+ stopTouch = function() {
+ return this.originalEvent.stopPropagation();
+ },
+ addEvent = R.addEvent = (function() {
+ if (g.doc.addEventListener) {
+ return function(obj, type, fn, element) {
+ var realName = supportsTouch && touchMap[type] ? touchMap[type] : type,
+ f = function(e) {
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft;
+ if (supportsTouch && touchMap[has](type)) {
+ for (var i = 0, ii = e.targetTouches && e.targetTouches.length; i < ii; i++) {
+ if (e.targetTouches[i].target == obj) {
+ var olde = e;
+ e = e.targetTouches[i];
+ e.originalEvent = olde;
+ e.preventDefault = preventTouch;
+ e.stopPropagation = stopTouch;
+ break;
+ }
+ }
+ }
+ return fn.call(element, e, e.clientX + scrollX, e.clientY + scrollY);
+ };
+ obj.addEventListener(realName, f, false);
+ return function() {
+ obj.removeEventListener(realName, f, false);
+ return true;
+ };
+ };
+ } else if (g.doc.attachEvent) {
+ return function(obj, type, fn, element) {
+ var f = function(e) {
+ e = e || g.win.event;
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft,
+ x = e.clientX + scrollX,
+ y = e.clientY + scrollY;
+ e.preventDefault = e.preventDefault || preventDefault;
+ e.stopPropagation = e.stopPropagation || stopPropagation;
+ return fn.call(element, e, x, y);
+ };
+ obj.attachEvent("on" + type, f);
+ var detacher = function() {
+ obj.detachEvent("on" + type, f);
+ return true;
+ };
+ return detacher;
+ };
+ }
+ })(),
+ drag = [],
+ dragMove = function(e) {
+ var x = e.clientX,
+ y = e.clientY,
+ scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft,
+ dragi,
+ j = drag.length;
+ while (j--) {
+ dragi = drag[j];
+ if (supportsTouch) {
+ var i = e.touches.length,
+ touch;
+ while (i--) {
+ touch = e.touches[i];
+ if (touch.identifier == dragi.el._drag.id) {
+ x = touch.clientX;
+ y = touch.clientY;
+ (e.originalEvent ? e.originalEvent : e).preventDefault();
+ break;
+ }
+ }
+ } else {
+ e.preventDefault();
+ }
+ var node = dragi.el.node,
+ o,
+ next = node.nextSibling,
+ parent = node.parentNode,
+ display = node.style.display;
+ g.win.opera && parent.removeChild(node);
+ node.style.display = "none";
+ o = dragi.el.paper.getElementByPoint(x, y);
+ node.style.display = display;
+ g.win.opera && (next ? parent.insertBefore(node, next) : parent.appendChild(node));
+ o && eve("raphael.drag.over." + dragi.el.id, dragi.el, o);
+ x += scrollX;
+ y += scrollY;
+ eve("raphael.drag.move." + dragi.el.id, dragi.move_scope || dragi.el, x - dragi.el._drag.x, y - dragi.el._drag.y, x, y, e);
+ }
+ },
+ dragUp = function(e) {
+ R.unmousemove(dragMove).unmouseup(dragUp);
+ var i = drag.length,
+ dragi;
+ while (i--) {
+ dragi = drag[i];
+ dragi.el._drag = {};
+ eve("raphael.drag.end." + dragi.el.id, dragi.end_scope || dragi.start_scope || dragi.move_scope || dragi.el, e);
+ }
+ drag = [];
+ },
+
+ /*\
+ * Raphael.el
+ [ property (object) ]
+ **
+ * You can add your own method to elements. This is usefull when you want to hack default functionality or
+ * want to wrap some common transformation or attributes in one method. In difference to canvas methods,
+ * you can redefine element method at any time. Expending element methods wouldn’t affect set.
+ > Usage
+ | Raphael.el.red = function () {
+ | this.attr({fill: "#f00"});
+ | };
+ | // then use it
+ | paper.circle(100, 100, 20).red();
+ \*/
+ elproto = R.el = {};
+
+ /*\
+ * Element.click
+ [ method ]
+ **
+ * Adds event handler for click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unclick
+ [ method ]
+ **
+ * Removes event handler for click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.dblclick
+ [ method ]
+ **
+ * Adds event handler for double click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.undblclick
+ [ method ]
+ **
+ * Removes event handler for double click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mousedown
+ [ method ]
+ **
+ * Adds event handler for mousedown for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmousedown
+ [ method ]
+ **
+ * Removes event handler for mousedown for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mousemove
+ [ method ]
+ **
+ * Adds event handler for mousemove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmousemove
+ [ method ]
+ **
+ * Removes event handler for mousemove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mouseout
+ [ method ]
+ **
+ * Adds event handler for mouseout for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmouseout
+ [ method ]
+ **
+ * Removes event handler for mouseout for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mouseover
+ [ method ]
+ **
+ * Adds event handler for mouseover for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmouseover
+ [ method ]
+ **
+ * Removes event handler for mouseover for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mouseup
+ [ method ]
+ **
+ * Adds event handler for mouseup for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmouseup
+ [ method ]
+ **
+ * Removes event handler for mouseup for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchstart
+ [ method ]
+ **
+ * Adds event handler for touchstart for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchstart
+ [ method ]
+ **
+ * Removes event handler for touchstart for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchmove
+ [ method ]
+ **
+ * Adds event handler for touchmove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchmove
+ [ method ]
+ **
+ * Removes event handler for touchmove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchend
+ [ method ]
+ **
+ * Adds event handler for touchend for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchend
+ [ method ]
+ **
+ * Removes event handler for touchend for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchcancel
+ [ method ]
+ **
+ * Adds event handler for touchcancel for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchcancel
+ [ method ]
+ **
+ * Removes event handler for touchcancel for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ for (var i = events.length; i--; ) {
+ (function(eventName) {
+ R[eventName] = elproto[eventName] = function(fn, scope) {
+ if (R.is(fn, "function")) {
+ this.events = this.events || [];
+ this.events.push({
+ name: eventName,
+ f: fn,
+ unbind: addEvent(this.shape || this.node || g.doc, eventName, fn, scope || this)
+ });
+ }
+ return this;
+ };
+ R["un" + eventName] = elproto["un" + eventName] = function(fn) {
+ var events = this.events || [],
+ l = events.length;
+ while (l--)
+ if (events[l].name == eventName && events[l].f == fn) {
+ events[l].unbind();
+ events.splice(l, 1);
+ !events.length && delete this.events;
+ return this;
+ }
+ return this;
+ };
+ })(events[i]);
+ }
+
+ /*\
+ * Element.data
+ [ method ]
+ **
+ * Adds or retrieves given value asociated with given key.
+ **
+ * See also @Element.removeData
+ > Parameters
+ - key (string) key to store data
+ - value (any) #optional value to store
+ = (object) @Element
+ * or, if value is not specified:
+ = (any) value
+ > Usage
+ | for (var i = 0, i < 5, i++) {
+ | paper.circle(10 + 15 * i, 10, 10)
+ | .attr({fill: "#000"})
+ | .data("i", i)
+ | .click(function () {
+ | alert(this.data("i"));
+ | });
+ | }
+ \*/
+ elproto.data = function(key, value) {
+ var data = eldata[this.id] = eldata[this.id] || {};
+ if (arguments.length == 1) {
+ if (R.is(key, object)) {
+ for (var i in key)
+ if (key[has](i)) {
+ this.data(i, key[i]);
+ }
+ return this;
+ }
+ eve("raphael.data.get." + this.id, this, data[key], key);
+ return data[key];
+ }
+ data[key] = value;
+ eve("raphael.data.set." + this.id, this, value, key);
+ return this;
+ };
+
+ /*\
+ * Element.removeData
+ [ method ]
+ **
+ * Removes value associated with an element by given key.
+ * If key is not provided, removes all the data of the element.
+ > Parameters
+ - key (string) #optional key
+ = (object) @Element
+ \*/
+ elproto.removeData = function(key) {
+ if (key == null) {
+ eldata[this.id] = {};
+ } else {
+ eldata[this.id] && delete eldata[this.id][key];
+ }
+ return this;
+ };
+
+ /*\
+ * Element.getData
+ [ method ]
+ **
+ * Retrieves the element data
+ = (object) data
+ \*/
+ elproto.getData = function () {
+ return clone(eldata[this.id] || {});
+ };
+
+ var downables = [],
+ mouseDown = function () {
+ this.untrack = addEvent(g.doc, 'mouseup', mouseUp, this);
+ },
+ mouseUp = function () {
+ this.untrack();
+ this.untrack = null;
+ return this.fn && this.fn.apply(this.scope || this.el, arguments);
+
+ };
+ elproto.mouseup = function (fn, scope, track) {
+ if (!track) {
+ return R.mouseup.apply(this, arguments);
+ }
+ downables.push(track = {
+ el: this,
+ fn: fn,
+ scope: scope
+ });
+ track.unbind = addEvent(this.shape || this.node || g.doc,
+ 'mousedown', mouseDown, track);
+
+ return this;
+ };
+
+ elproto.unmouseup = function (fn) {
+ var i = downables.length,
+ undowned;
+ while (i--) {
+ if (downables[i].el === this && downables[i].fn === fn) {
+ undowned = downables[i];
+ undowned.unbind();
+ undowned.untrack && undowned.untrack();
+ downables.splice(i, 1);
+ }
+ }
+ return undowned ? this : R.unmouseup.apply(this, arguments);
+ };
+
+ /*\
+ * Element.hover
+ [ method ]
+ **
+ * Adds event handlers for hover for the element.
+ > Parameters
+ - f_in (function) handler for hover in
+ - f_out (function) handler for hover out
+ - icontext (object) #optional context for hover in handler
+ - ocontext (object) #optional context for hover out handler
+ = (object) @Element
+ \*/
+ elproto.hover = function(f_in, f_out, scope_in, scope_out) {
+ return this.mouseover(f_in, scope_in).mouseout(f_out, scope_out || scope_in);
+ };
+
+ /*\
+ * Element.unhover
+ [ method ]
+ **
+ * Removes event handlers for hover for the element.
+ > Parameters
+ - f_in (function) handler for hover in
+ - f_out (function) handler for hover out
+ = (object) @Element
+ \*/
+ elproto.unhover = function(f_in, f_out) {
+ return this.unmouseover(f_in).unmouseout(f_out);
+ };
+ var draggable = [];
+
+ /*\
+ * Element.drag
+ [ method ]
+ **
+ * Adds event handlers for drag of the element.
+ > Parameters
+ - onmove (function) handler for moving
+ - onstart (function) handler for drag start
+ - onend (function) handler for drag end
+ - mcontext (object) #optional context for moving handler
+ - scontext (object) #optional context for drag start handler
+ - econtext (object) #optional context for drag end handler
+ * Additionaly following `drag` events will be triggered: `drag.start.` on start,
+ * `drag.end.` on end and `drag.move.` on every move. When element will be dragged over another element
+ * `drag.over.` will be fired as well.
+ *
+ * Start event and start handler will be called in specified context or in context of the element with following parameters:
+ o x (number) x position of the mouse
+ o y (number) y position of the mouse
+ o event (object) DOM event object
+ * Move event and move handler will be called in specified context or in context of the element with following parameters:
+ o dx (number) shift by x from the start point
+ o dy (number) shift by y from the start point
+ o x (number) x position of the mouse
+ o y (number) y position of the mouse
+ o event (object) DOM event object
+ * End event and end handler will be called in specified context or in context of the element with following parameters:
+ o event (object) DOM event object
+ = (object) @Element
+ \*/
+ elproto.drag = function(onmove, onstart, onend, move_scope, start_scope, end_scope) {
+ function start(e) {
+ (e.originalEvent || e).preventDefault();
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft;
+ this._drag.x = e.clientX + scrollX;
+ this._drag.y = e.clientY + scrollY;
+ this._drag.id = e.identifier;
+ !drag.length && R.mousemove(dragMove).mouseup(dragUp);
+ drag.push({
+ el: this,
+ move_scope: move_scope,
+ start_scope: start_scope,
+ end_scope: end_scope
+ });
+ onstart && eve.on("raphael.drag.start." + this.id, onstart);
+ onmove && eve.on("raphael.drag.move." + this.id, onmove);
+ onend && eve.on("raphael.drag.end." + this.id, onend);
+ eve("raphael.drag.start." + this.id, start_scope || move_scope || this, e.clientX + scrollX, e.clientY + scrollY, e);
+ }
+ this._drag = {};
+ draggable.push({
+ el: this,
+ start: start
+ });
+ this.mousedown(start);
+ return this;
+ };
+
+ /*\
+ * Element.onDragOver
+ [ method ]
+ **
+ * Shortcut for assigning event handler for `drag.over.` event, where id is id of the element (see @Element.id).
+ > Parameters
+ - f (function) handler for event, first argument would be the element you are dragging over
+ \*/
+ elproto.onDragOver = function(f) {
+ f ? eve.on("raphael.drag.over." + this.id, f) : eve.unbind("raphael.drag.over." + this.id);
+ };
+
+ /*\
+ * Element.undrag
+ [ method ]
+ **
+ * Removes all drag event handlers from given element.
+ \*/
+ elproto.undrag = function() {
+ var i = draggable.length;
+ while (i--)
+ if (draggable[i].el == this) {
+ this.unmousedown(draggable[i].start);
+ draggable.splice(i, 1);
+ eve.unbind("raphael.drag.*." + this.id);
+ }
+ !draggable.length && R.unmousemove(dragMove).unmouseup(dragUp);
+ };
+
+ elproto.follow = function(el, callback, stalk) {
+ if (el.removed || el.constructor !== R.el.constructor) {
+ return this;
+ }
+ el.followers.push({
+ el: this,
+ stalk: (stalk = {before: 'insertBefore', after: 'insertAfter'}[stalk]),
+ cb: callback
+ });
+
+ stalk && this[stalk](el);
+ return this;
+ };
+
+ elproto.unfollow = function(el) {
+ if (el.removed || el.constructor !== R.el.constructor) {
+ return this;
+ }
+ for (var i = 0, ii = el.followers.length; i < ii; i++) {
+ if (el.followers[i].el === this) {
+ el.followers.splice(i, 1);
+ break;
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Paper.hide
+ [ method ]
+ **
+ * Hides a paper
+ **
+ > Usage
+ | paper.hide();
+ \*/
+ paperproto.hide = function () {
+ var paper = this;
+ paper.canvas.style.visibility = "hidden";
+ return paper;
+ };
+
+ /*\
+ * Paper.show
+ [ method ]
+ **
+ * Shows a hidden paper
+ **
+ > Usage
+ | paper.show();
+ \*/
+ paperproto.show = function () {
+ var paper = this;
+ paper.canvas.style.visibility = E;
+ return paper;
+ };
+
+ /*\
+ * Paper.group
+ [ method ]
+ **
+ * Creates a group
+ **
+ > Parameters
+ **
+ - id (number) id of the group
+ = (object) Raphaël element object with type “group”
+ **
+ > Usage
+ | var g = paper.group();
+ \*/
+ paperproto.group = function () { // id
+ var paper = this,
+ out,
+ args = arguments,
+ group = lastArgIfGroup(args, true);
+
+ out = R._engine.group(paper, args[0], group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.circle
+ [ method ]
+ **
+ * Draws a circle.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the centre
+ - y (number) y coordinate of the centre
+ - r (number) radius
+ = (object) Raphaël element object with type “circle”
+ **
+ > Usage
+ | var c = paper.circle(50, 50, 40);
+ \*/
+ paperproto.circle = function () { // x, y, r
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.circle(paper, args[0] || 0, args[1] || 0,
+ args[2] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+
+ /*\
+ * Paper.rect
+ [ method ]
+ *
+ * Draws a rectangle.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the top left corner
+ - y (number) y coordinate of the top left corner
+ - width (number) width
+ - height (number) height
+ - r (number) #optional radius for rounded corners, default is 0
+ = (object) Raphaël element object with type “rect”
+ **
+ > Usage
+ | // regular rectangle
+ | var c = paper.rect(10, 10, 50, 50);
+ | // rectangle with rounded corners
+ | var c = paper.rect(40, 40, 50, 50, 10);
+ \*/
+ paperproto.rect = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.rect(paper, args[0] || 0, args[1] || 0, args[2] || 0,
+ args[3] || 0, args[4] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.ellipse
+ [ method ]
+ **
+ * Draws an ellipse.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the centre
+ - y (number) y coordinate of the centre
+ - rx (number) horizontal radius
+ - ry (number) vertical radius
+ = (object) Raphaël element object with type “ellipse”
+ **
+ > Usage
+ | var c = paper.ellipse(50, 50, 40, 20);
+ \*/
+ paperproto.ellipse = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.ellipse(this, args[0] || 0, args[1] || 0,
+ args[2] || 0, args[3] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.path
+ [ method ]
+ **
+ * Creates a path element by given path data string.
+ > Parameters
+ - pathString (string) #optional path string in SVG format.
+ * Path string consists of one-letter commands, followed by comma seprarated arguments in numercal form. Example:
+ | "M10,20L30,40"
+ * Here we can see two commands: “M”, with arguments `(10, 20)` and “L” with arguments `(30, 40)`. Upper case letter mean command is absolute, lower case—relative.
+ *
+ # Here is short list of commands available, for more details see SVG path string format .
+ # Command Name Parameters
+ # M moveto (x y)+
+ # Z closepath (none)
+ # L lineto (x y)+
+ # H horizontal lineto x+
+ # V vertical lineto y+
+ # C curveto (x1 y1 x2 y2 x y)+
+ # S smooth curveto (x2 y2 x y)+
+ # Q quadratic Bézier curveto (x1 y1 x y)+
+ # T smooth quadratic Bézier curveto (x y)+
+ # A elliptical arc (rx ry x-axis-rotation large-arc-flag sweep-flag x y)+
+ # R Catmull-Rom curveto *x1 y1 (x y)+
+ * * “Catmull-Rom curveto” is a not standard SVG command and added in 2.0 to make life easier.
+ * Note: there is a special case when path consist of just three commands: “M10,10R…z”. In this case path will smoothly connects to its beginning.
+ > Usage
+ | var c = paper.path("M10 10L90 90");
+ | // draw a diagonal line:
+ | // move to 10,10, line to 90,90
+ * For example of path strings, check out these icons: http://raphaeljs.com/icons/
+ \*/
+ paperproto.path = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ pathString,
+ out;
+
+ pathString = args[0];
+ pathString && !R.is(pathString, string) &&
+ !R.is(pathString[0], array) && (pathString += E);
+
+ out = R._engine.path(R.format[apply](R, arguments), paper, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.image
+ [ method ]
+ **
+ * Embeds an image into the surface.
+ **
+ > Parameters
+ **
+ - src (string) URI of the source image
+ - x (number) x coordinate position
+ - y (number) y coordinate position
+ - width (number) width of the image
+ - height (number) height of the image
+ = (object) Raphaël element object with type “image”
+ **
+ > Usage
+ | var c = paper.image("apple.png", 10, 10, 80, 80);
+ \*/
+ paperproto.image = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.image(paper, args[0] || "about:blank", args[1] || 0,
+ args[2] || 0, args[3] || 0, args[4] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.text
+ [ method ]
+ **
+ * Draws a text string. If you need line breaks, put “\n” in the string.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate position
+ - y (number) y coordinate position
+ - text (string) The text string to draw
+ = (object) Raphaël element object with type “text”
+ **
+ > Usage
+ | var t = paper.text(50, 50, "Raphaël\nkicks\nbutt!");
+ \*/
+ paperproto.text = function() {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.text(paper, args[0] || 0, args[1] || 0, Str(args[2] || E),
+ group);
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.set
+ [ method ]
+ **
+ * Creates array-like object to keep and operate several elements at once.
+ * Warning: it doesn’t create any elements for itself in the page, it just groups existing elements.
+ * Sets act as pseudo elements — all methods available to an element can be used on a set.
+ = (object) array-like object that represents set of elements
+ **
+ > Usage
+ | var st = paper.set();
+ | st.push(
+ | paper.circle(10, 10, 5),
+ | paper.circle(30, 10, 5)
+ | );
+ | st.attr({fill: "red"}); // changes the fill of both circles
+ \*/
+ paperproto.set = function(itemsArray) {
+ !R.is(itemsArray, "array") && (itemsArray = arraySplice.call(arguments, 0, arguments.length));
+ var out = new Set(itemsArray);
+ this.__set__ && this.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.setStart
+ [ method ]
+ **
+ * Creates @Paper.set. All elements that will be created after calling this method and before calling
+ * @Paper.setFinish will be added to the set.
+ **
+ > Usage
+ | paper.setStart();
+ | paper.circle(10, 10, 5),
+ | paper.circle(30, 10, 5)
+ | var st = paper.setFinish();
+ | st.attr({fill: "red"}); // changes the fill of both circles
+ \*/
+ paperproto.setStart = function(set) {
+ this.__set__ = set || this.set();
+ };
+
+ /*\
+ * Paper.setFinish
+ [ method ]
+ **
+ * See @Paper.setStart. This method finishes catching and returns resulting set.
+ **
+ = (object) set
+ \*/
+ paperproto.setFinish = function(set) {
+ var out = this.__set__;
+ delete this.__set__;
+ return out;
+ };
+
+ /*\
+ * Paper.setSize
+ [ method ]
+ **
+ * If you need to change dimensions of the canvas call this method
+ **
+ > Parameters
+ **
+ - width (number) new width of the canvas
+ - height (number) new height of the canvas
+ \*/
+ paperproto.setSize = function(width, height) {
+ return R._engine.setSize.call(this, width, height);
+ };
+
+ /*\
+ * Paper.setViewBox
+ [ method ]
+ **
+ * Sets the view box of the paper. Practically it gives you ability to zoom and pan whole paper surface by
+ * specifying new boundaries.
+ **
+ > Parameters
+ **
+ - x (number) new x position, default is `0`
+ - y (number) new y position, default is `0`
+ - w (number) new width of the canvas
+ - h (number) new height of the canvas
+ - fit (boolean) `true` if you want graphics to fit into new boundary box
+ \*/
+ paperproto.setViewBox = function(x, y, w, h, fit) {
+ return R._engine.setViewBox.call(this, x, y, w, h, fit);
+ };
+
+ /*\
+ * Paper.top
+ [ property ]
+ **
+ * Points to the topmost element on the paper
+ \*/
+ /*\
+ * Paper.bottom
+ [ property ]
+ **
+ * Points to the bottom element on the paper
+ \*/
+ paperproto.top = paperproto.bottom = null;
+
+ /*\
+ * Paper.raphael
+ [ property ]
+ **
+ * Points to the @Raphael object/function
+ \*/
+ paperproto.raphael = R;
+
+ var getOffset = function(elem) {
+ var box = elem.getBoundingClientRect(),
+ doc = elem.ownerDocument,
+ body = doc.body,
+ docElem = doc.documentElement,
+ clientTop = docElem.clientTop || body.clientTop || 0, clientLeft = docElem.clientLeft || body.clientLeft || 0,
+ top = box.top + (g.win.pageYOffset || docElem.scrollTop || body.scrollTop) - clientTop,
+ left = box.left + (g.win.pageXOffset || docElem.scrollLeft || body.scrollLeft) - clientLeft;
+ return {
+ y: top,
+ x: left
+ };
+ };
+
+ /*\
+ * Paper.getElementByPoint
+ [ method ]
+ **
+ * Returns you topmost element under given point.
+ **
+ = (object) Raphaël element object
+ > Parameters
+ **
+ - x (number) x coordinate from the top left corner of the window
+ - y (number) y coordinate from the top left corner of the window
+ > Usage
+ | paper.getElementByPoint(mouseX, mouseY).attr({stroke: "#f00"});
+ \*/
+ paperproto.getElementByPoint = function(x, y) {
+ var paper = this,
+ svg = paper.canvas,
+ target = g.doc.elementFromPoint(x, y);
+ if (g.win.opera && target.tagName == "svg") {
+ var so = getOffset(svg),
+ sr = svg.createSVGRect();
+ sr.x = x - so.x;
+ sr.y = y - so.y;
+ sr.width = sr.height = 1;
+ var hits = svg.getIntersectionList(sr, null);
+ if (hits.length) {
+ target = hits[hits.length - 1];
+ }
+ }
+ if (!target) {
+ return null;
+ }
+ while (target.parentNode && target != svg.parentNode && !target.raphael) {
+ target = target.parentNode;
+ }
+ target == paper.canvas.parentNode && (target = svg);
+ target = target && target.raphael ? paper.getById(target.raphaelid) : null;
+ return target;
+ };
+
+ /*\
+ * Paper.getElementsByBBox
+ [ method ]
+ **
+ * Returns set of elements that have an intersecting bounding box
+ **
+ > Parameters
+ **
+ - bbox (object) bbox to check with
+ = (object) @Set
+ \*/
+ paperproto.getElementsByBBox = function (bbox) {
+ var set = this.set();
+ this.forEach(function (el) {
+ if (R.isBBoxIntersect(el.getBBox(), bbox)) {
+ set.push(el);
+ }
+ });
+ return set;
+ };
+
+ paperproto.getById = function(id) {
+ var bot = this.bottom;
+ while (bot) {
+ if (bot.id == id) {
+ return bot;
+ }
+ bot = bot.next;
+ }
+ return null;
+ };
+
+ /*\
+ * Paper.forEach
+ [ method ]
+ **
+ * Executes given function for each element on the paper
+ *
+ * If callback function returns `false` it will stop loop running.
+ **
+ > Parameters
+ **
+ - callback (function) function to run
+ - thisArg (object) context object for the callback
+ = (object) Paper object
+ > Usage
+ | paper.forEach(function (el) {
+ | el.attr({ stroke: "blue" });
+ | });
+ \*/
+ paperproto.forEach = function(callback, thisArg) {
+ var bot = this.bottom;
+ while (bot) {
+ if (callback.call(thisArg, bot) === false) {
+ return this;
+ }
+ bot = bot.next;
+ }
+ return this;
+ };
+
+ /*\
+ * Paper.getElementsByPoint
+ [ method ]
+ **
+ * Returns set of elements that have common point inside
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the point
+ - y (number) y coordinate of the point
+ = (object) @Set
+ \*/
+ paperproto.getElementsByPoint = function(x, y) {
+ var set = this.set();
+ this.forEach(function(el) {
+ if (el.isPointInside(x, y)) {
+ set.push(el);
+ }
+ });
+ return set;
+ };
+ function x_y() {
+ return this.x + S + this.y;
+ }
+ function x_y_w_h() {
+ return this.x + S + this.y + S + this.width + " \xd7 " + this.height;
+ }
+
+ /*\
+ * Element.isPointInside
+ [ method ]
+ **
+ * Determine if given point is inside this element’s shape
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the point
+ - y (number) y coordinate of the point
+ = (boolean) `true` if point inside the shape
+ \*/
+ elproto.isPointInside = function(x, y) {
+ var rp = this.realPath = this.realPath || getPath[this.type](this),
+ tr;
+ return R.isPointInsidePath(((tr = this.attr('transform')) &&
+ tr.length && R.transformPath(rp, tr)) || rp, x, y);
+ };
+
+ /*\
+ * Element.getBBox
+ [ method ]
+ **
+ * Return bounding box for a given element
+ **
+ > Parameters
+ **
+ - isWithoutTransform (boolean) flag, `true` if you want to have bounding box before transformations. Default is `false`.
+ = (object) Bounding box object:
+ o {
+ o x: (number) top left corner x
+ o y: (number) top left corner y
+ o x2: (number) bottom right corner x
+ o y2: (number) bottom right corner y
+ o width: (number) width
+ o height: (number) height
+ o }
+ \*/
+ elproto.getBBox = function(isWithoutTransform) {
+ if (this.removed) {
+ return {};
+ }
+ var _ = this._;
+ if (isWithoutTransform) {
+ if (_.dirty || !_.bboxwt) {
+ this.realPath = getPath[this.type](this);
+ _.bboxwt = pathDimensions(this.realPath);
+ _.bboxwt.toString = x_y_w_h;
+ _.dirty = 0;
+ }
+ return _.bboxwt;
+ }
+ if (_.dirty || _.dirtyT || !_.bbox) {
+ if (_.dirty || !this.realPath) {
+ _.bboxwt = 0;
+ this.realPath = getPath[this.type](this);
+ }
+ _.bbox = pathDimensions(mapPath(this.realPath, this.matrix));
+ _.bbox.toString = x_y_w_h;
+ _.dirty = _.dirtyT = 0;
+ }
+ return _.bbox;
+ };
+
+ /*\
+ * Element.clone
+ [ method ]
+ **
+ = (object) clone of a given element
+ **
+ \*/
+ elproto.clone = function() {
+ if (this.removed) {
+ return null;
+ }
+ var o = this,
+ out = o.paper[o.type]().attr(o.attr());
+ o.__set__ && o.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Element.glow
+ [ method ]
+ **
+ * Return set of elements that create glow-like effect around given element. See @Paper.set.
+ *
+ * Note: Glow is not connected to the element. If you change element attributes it won’t adjust itself.
+ **
+ > Parameters
+ **
+ - glow (object) #optional parameters object with all properties optional:
+ o {
+ o width (number) size of the glow, default is `10`
+ o fill (boolean) will it be filled, default is `false`
+ o opacity (number) opacity, default is `0.5`
+ o offsetx (number) horizontal offset, default is `0`
+ o offsety (number) vertical offset, default is `0`
+ o color (string) glow colour, default is `black`
+ o }
+ = (object) @Paper.set of elements that represents glow
+ \*/
+ elproto.glow = function(glow) {
+ if (this.type == "text") {
+ return null;
+ }
+ glow = glow || {};
+ var s = {
+ width: (glow.width || 10) + (+this.attr("stroke-width") || 1),
+ fill: glow.fill || false,
+ opacity: glow.opacity || .5,
+ offsetx: glow.offsetx || 0,
+ offsety: glow.offsety || 0,
+ color: glow.color || "#000"
+ },
+ c = s.width / 2,
+ r = this.paper,
+ out = r.set(),
+ path = this.realPath || getPath[this.type](this);
+ path = this.matrix ? mapPath(path, this.matrix) : path;
+ for (var i = 1; i < c + 1; i++) {
+ out.push(r.path(path).attr({
+ stroke: s.color,
+ fill: s.fill ? s.color : "none",
+ "stroke-linejoin": "round",
+ "stroke-linecap": "round",
+ "stroke-width": +(s.width / c * i).toFixed(3),
+ opacity: +(s.opacity / c).toFixed(3)
+ }));
+ }
+ return out.insertBefore(this).translate(s.offsetx, s.offsety);
+ };
+ var curveslengths = {},
+ getPointAtSegmentLength = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, length) {
+ if (length == null) {
+ return bezlen(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y);
+ } else {
+ return R.findDotsAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, getTatLen(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, length));
+ }
+ },
+ getLengthFactory = function(istotal, subpath) {
+ return function(path, length, onlystart) {
+ path = path2curve(path);
+ var x, y, p, l, sp = "", subpaths = {}, point,
+ len = 0;
+ for (var i = 0, ii = path.length; i < ii; i++) {
+ p = path[i];
+ if (p[0] == "M") {
+ x = +p[1];
+ y = +p[2];
+ } else {
+ l = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6]);
+ if (len + l > length) {
+ if (subpath && !subpaths.start) {
+ point = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6], length - len);
+ sp += ["C" + point.start.x, point.start.y, point.m.x, point.m.y, point.x, point.y];
+ if (onlystart) {
+ return sp;
+ }
+ subpaths.start = sp;
+ sp = ["M" + point.x, point.y + "C" + point.n.x, point.n.y, point.end.x, point.end.y, p[5], p[6]].join();
+ len += l;
+ x = +p[5];
+ y = +p[6];
+ continue;
+ }
+ if (!istotal && !subpath) {
+ point = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6], length - len);
+ return {
+ x: point.x,
+ y: point.y,
+ alpha: point.alpha
+ };
+ }
+ }
+ len += l;
+ x = +p[5];
+ y = +p[6];
+ }
+ sp += p.shift() + p;
+ }
+ subpaths.end = sp;
+ point = istotal ? len : subpath ? subpaths : R.findDotsAtSegment(x, y, p[0], p[1], p[2], p[3], p[4], p[5], 1);
+ point.alpha && (point = {
+ x: point.x,
+ y: point.y,
+ alpha: point.alpha
+ });
+ return point;
+ };
+ };
+ var getTotalLength = getLengthFactory(1),
+ getPointAtLength = getLengthFactory(),
+ getSubpathsAtLength = getLengthFactory(0, 1);
+
+ R.getTotalLength = getTotalLength;
+
+ R.getPointAtLength = getPointAtLength;
+
+ R.getSubpath = function(path, from, to) {
+ if (this.getTotalLength(path) - to < 1e-6) {
+ return getSubpathsAtLength(path, from).end;
+ }
+ var a = getSubpathsAtLength(path, to, 1);
+ return from ? getSubpathsAtLength(a, from).end : a;
+ };
+
+ /*\
+ * Raphael.getTotalLength
+ [ method ]
+ **
+ * Returns length of the given path in pixels.
+ **
+ > Parameters
+ **
+ - path (string) SVG path string.
+ **
+ = (number) length.
+ \*/
+ elproto.getTotalLength = function() {
+ if (this.type != "path") {
+ return;
+ }
+ if (this.node.getTotalLength) {
+ return this.node.getTotalLength();
+ }
+ return getTotalLength(this.attrs.path);
+ };
+
+ /*\
+ * Raphael.getPointAtLength
+ [ method ]
+ **
+ * Return coordinates of the point located at the given length on the given path.
+ **
+ > Parameters
+ **
+ - path (string) SVG path string
+ - length (number)
+ **
+ = (object) representation of the point:
+ o {
+ o x: (number) x coordinate
+ o y: (number) y coordinate
+ o alpha: (number) angle of derivative
+ o }
+ \*/
+ elproto.getPointAtLength = function(length) {
+ if (this.type != "path") {
+ return;
+ }
+ return getPointAtLength(this.attrs.path, length);
+ };
+
+ /*\
+ * Raphael.getSubpath
+ [ method ]
+ **
+ * Return subpath of a given path from given length to given length.
+ **
+ > Parameters
+ **
+ - path (string) SVG path string
+ - from (number) position of the start of the segment
+ - to (number) position of the end of the segment
+ **
+ = (string) pathstring for the segment
+ \*/
+ elproto.getSubpath = function(from, to) {
+ if (this.type != "path") {
+ return;
+ }
+ return R.getSubpath(this.attrs.path, from, to);
+ };
+
+ /*\
+ * Raphael.easing_formulas
+ [ property ]
+ **
+ * Object that contains easing formulas for animation. You could extend it with your own. By default it has following list of easing:
+ #
+ # “linear”
+ # “<” or “easeIn” or “ease-in”
+ # “>” or “easeOut” or “ease-out”
+ # “<>” or “easeInOut” or “ease-in-out”
+ # “backIn” or “back-in”
+ # “backOut” or “back-out”
+ # “elastic”
+ # “bounce”
+ #
+ # See also Easing demo .
+ \*/
+ var ef = R.easing_formulas = {
+ linear: function(n) {
+ return n;
+ },
+ "<": function(n) {
+ return pow(n, 1.7);
+ },
+ ">": function(n) {
+ return pow(n, .48);
+ },
+ "<>": function(n) {
+ var q = .48 - n / 1.04,
+ Q = mathSqrt(.1734 + q * q),
+ x = Q - q,
+ X = pow(abs(x), 1 / 3) * (x < 0 ? -1 : 1),
+ y = -Q - q,
+ Y = pow(abs(y), 1 / 3) * (y < 0 ? -1 : 1),
+ t = X + Y + .5;
+ return (1 - t) * 3 * t * t + t * t * t;
+ },
+ backIn: function(n) {
+ var s = 1.70158;
+ return n * n * ((s + 1) * n - s);
+ },
+ backOut: function(n) {
+ n = n - 1;
+ var s = 1.70158;
+ return n * n * ((s + 1) * n + s) + 1;
+ },
+ elastic: function(n) {
+ if (n == !!n) {
+ return n;
+ }
+ return pow(2, -10 * n) * mathSin((n - .075) * (2 * PI) / .3) + 1;
+ },
+ bounce: function(n) {
+ var s = 7.5625,
+ p = 2.75,
+ l;
+ if (n < (1 / p)) {
+ l = s * n * n;
+ } else {
+ if (n < (2 / p)) {
+ n -= (1.5 / p);
+ l = s * n * n + .75;
+ } else {
+ if (n < (2.5 / p)) {
+ n -= (2.25 / p);
+ l = s * n * n + .9375;
+ } else {
+ n -= (2.625 / p);
+ l = s * n * n + .984375;
+ }
+ }
+ }
+ return l;
+ }
+ };
+ ef.easeIn = ef["ease-in"] = ef["<"];
+ ef.easeOut = ef["ease-out"] = ef[">"];
+ ef.easeInOut = ef["ease-in-out"] = ef["<>"];
+ ef["back-in"] = ef.backIn;
+ ef["back-out"] = ef.backOut;
+
+ var animationElements = [],
+ requestAnimFrame = window.requestAnimationFrame ||
+ window.webkitRequestAnimationFrame ||
+ window.mozRequestAnimationFrame ||
+ window.oRequestAnimationFrame ||
+ window.msRequestAnimationFrame ||
+ function(callback) {
+ setTimeout(callback, 16);
+ },
+ animation = function() {
+ var Now = +new Date,
+ l = 0;
+ for (; l < animationElements.length; l++) {
+ var e = animationElements[l];
+ if (e.el.removed || e.paused) {
+ continue;
+ }
+ var time = Now - e.start,
+ ms = e.ms,
+ easing = e.easing,
+ from = e.from,
+ diff = e.diff,
+ to = e.to,
+ t = e.t,
+ that = e.el,
+ set = {},
+ now,
+ init = {},
+ key;
+ if (e.initstatus) {
+ time = (e.initstatus * e.anim.top - e.prev) / (e.percent - e.prev) * ms;
+ e.status = e.initstatus;
+ delete e.initstatus;
+ e.stop && animationElements.splice(l--, 1);
+ } else {
+ e.status = (e.prev + (e.percent - e.prev) * (time / ms)) / e.anim.top;
+ }
+ if (time < 0) {
+ continue;
+ }
+ if (time < ms) {
+ var pos = easing(time / ms);
+ for (var attr in from)
+ if (from[has](attr)) {
+ switch (availableAnimAttrs[attr]) {
+ case nu:
+ now = +from[attr] + pos * ms * diff[attr];
+ break;
+ case "colour":
+ now = "rgb(" + [
+ upto255(round(from[attr].r + pos * ms * diff[attr].r)),
+ upto255(round(from[attr].g + pos * ms * diff[attr].g)),
+ upto255(round(from[attr].b + pos * ms * diff[attr].b))
+ ].join(",") + ")";
+ break;
+ case "path":
+ now = [];
+ for (var i = 0, ii = from[attr].length; i < ii; i++) {
+ now[i] = [from[attr][i][0]];
+ for (var j = 1, jj = from[attr][i].length; j < jj; j++) {
+ now[i][j] = (+from[attr][i][j] + pos * ms * diff[attr][i][j]).toFixed(4);
+ }
+ now[i] = now[i].join(S);
+ }
+ now = now.join(S);
+ break;
+ case "transform":
+ if (diff[attr].real) {
+ now = [];
+ for (i = 0, ii = from[attr].length; i < ii; i++) {
+ now[i] = [from[attr][i][0]];
+ for (j = 1, jj = from[attr][i].length; j < jj; j++) {
+ now[i][j] = from[attr][i][j] + pos * ms * diff[attr][i][j];
+ }
+ }
+ } else {
+ var get = function(i) {
+ return +from[attr][i] + pos * ms * diff[attr][i];
+ };
+ // now = [["r", get(2), 0, 0], ["t", get(3), get(4)], ["s", get(0), get(1), 0, 0]];
+ now = [["m", get(0), get(1), get(2), get(3), get(4), get(5)]];
+ }
+ break;
+ case "csv":
+ if (attr == "clip-rect") {
+ now = [];
+ i = 4;
+ while (i--) {
+ now[i] = +from[attr][i] + pos * ms * diff[attr][i];
+ }
+ }
+ break;
+ default:
+ var from2 = [][concat](from[attr]);
+ now = [];
+ i = that.ca[attr].length;
+ while (i--) {
+ now[i] = +from2[i] + pos * ms * diff[attr][i];
+ }
+ break;
+ }
+ set[attr] = now;
+ }
+ that.attr(set);
+ (function(id, that, anim) {
+ setTimeout(function() {
+ eve("raphael.anim.frame." + id, that, anim);
+ });
+ })(that.id, that, e.anim);
+ } else {
+ (function(f, el, a) {
+ setTimeout(function() {
+ eve("raphael.anim.frame." + el.id, el, a);
+ eve("raphael.anim.finish." + el.id, el, a);
+ R.is(f, "function") && f.call(el);
+ });
+ })(e.callback, that, e.anim);
+ that.attr(to);
+ animationElements.splice(l--, 1);
+ if (e.repeat > 1 && !e.next) {
+ for (key in to)
+ if (to[has](key)) {
+ init[key] = e.totalOrigin[key];
+ }
+ e.el.attr(init);
+ runAnimation(e.anim, e.el, e.anim.percents[0], null, e.totalOrigin, e.repeat - 1);
+ }
+ if (e.next && !e.stop) {
+ runAnimation(e.anim, e.el, e.next, null, e.totalOrigin, e.repeat);
+ }
+ }
+ }
+ R.svg && that && that.paper && that.paper.safari();
+ animationElements.length && requestAnimFrame(animation);
+ },
+ upto255 = function(color) {
+ return color > 255 ? 255 : color < 0 ? 0 : color;
+ };
+
+ /*\
+ * Element.animateWith
+ [ method ]
+ **
+ * Acts similar to @Element.animate, but ensure that given animation runs in sync with another given element.
+ **
+ > Parameters
+ **
+ - el (object) element to sync with
+ - anim (object) animation to sync with
+ - params (object) #optional final attributes for the element, see also @Element.attr
+ - ms (number) #optional number of milliseconds for animation to run
+ - easing (string) #optional easing type. Accept on of @Raphael.easing_formulas or CSS format: `cubic‐bezier(XX, XX, XX, XX)`
+ - callback (function) #optional callback function. Will be called at the end of animation.
+ * or
+ - element (object) element to sync with
+ - anim (object) animation to sync with
+ - animation (object) #optional animation object, see @Raphael.animation
+ **
+ = (object) original element
+ \*/
+ elproto.animateWith = function(el, anim, params, ms, easing, callback) {
+ var element = this;
+ if (element.removed) {
+ callback && callback.call(element);
+ return element;
+ }
+ var a = params instanceof Animation ? params : R.animation(params, ms, easing, callback),
+ x, y;
+ runAnimation(a, element, a.percents[0], null, element.attr());
+ for (var i = 0, ii = animationElements.length; i < ii; i++) {
+ if (animationElements[i].anim == anim && animationElements[i].el == el) {
+ animationElements[ii - 1].start = animationElements[i].start;
+ break;
+ }
+ }
+ return element;
+ //
+ //
+ // var a = params ? R.animation(params, ms, easing, callback) : anim,
+ // status = element.status(anim);
+ // return this.animate(a).status(a, status * anim.ms / a.ms);
+ };
+ function CubicBezierAtTime(t, p1x, p1y, p2x, p2y, duration) {
+ var cx = 3 * p1x,
+ bx = 3 * (p2x - p1x) - cx,
+ ax = 1 - cx - bx,
+ cy = 3 * p1y,
+ by = 3 * (p2y - p1y) - cy,
+ ay = 1 - cy - by;
+ function sampleCurveX(t) {
+ return ((ax * t + bx) * t + cx) * t;
+ }
+ function solve(x, epsilon) {
+ var t = solveCurveX(x, epsilon);
+ return ((ay * t + by) * t + cy) * t;
+ }
+ function solveCurveX(x, epsilon) {
+ var t0, t1, t2, x2, d2, i;
+ for (t2 = x, i = 0; i < 8; i++) {
+ x2 = sampleCurveX(t2) - x;
+ if (abs(x2) < epsilon) {
+ return t2;
+ }
+ d2 = (3 * ax * t2 + 2 * bx) * t2 + cx;
+ if (abs(d2) < 1e-6) {
+ break;
+ }
+ t2 = t2 - x2 / d2;
+ }
+ t0 = 0;
+ t1 = 1;
+ t2 = x;
+ if (t2 < t0) {
+ return t0;
+ }
+ if (t2 > t1) {
+ return t1;
+ }
+ while (t0 < t1) {
+ x2 = sampleCurveX(t2);
+ if (abs(x2 - x) < epsilon) {
+ return t2;
+ }
+ if (x > x2) {
+ t0 = t2;
+ } else {
+ t1 = t2;
+ }
+ t2 = (t1 - t0) / 2 + t0;
+ }
+ return t2;
+ }
+ return solve(t, 1 / (200 * duration));
+ }
+ elproto.onAnimation = function(f) {
+ f ? eve.on("raphael.anim.frame." + this.id, f) : eve.unbind("raphael.anim.frame." + this.id);
+ return this;
+ };
+ function Animation(anim, ms) {
+ var percents = [],
+ newAnim = {};
+ this.ms = ms;
+ this.times = 1;
+ if (anim) {
+ for (var attr in anim)
+ if (anim[has](attr)) {
+ newAnim[toFloat(attr)] = anim[attr];
+ percents.push(toFloat(attr));
+ }
+ percents.sort(sortByNumber);
+ }
+ this.anim = newAnim;
+ this.top = percents[percents.length - 1];
+ this.percents = percents;
+ }
+
+ /*\
+ * Animation.delay
+ [ method ]
+ **
+ * Creates a copy of existing animation object with given delay.
+ **
+ > Parameters
+ **
+ - delay (number) number of ms to pass between animation start and actual animation
+ **
+ = (object) new altered Animation object
+ | var anim = Raphael.animation({cx: 10, cy: 20}, 2e3);
+ | circle1.animate(anim); // run the given animation immediately
+ | circle2.animate(anim.delay(500)); // run the given animation after 500 ms
+ \*/
+ Animation.prototype.delay = function(delay) {
+ var a = new Animation(this.anim, this.ms);
+ a.times = this.times;
+ a.del = +delay || 0;
+ return a;
+ };
+
+ /*\
+ * Animation.repeat
+ [ method ]
+ **
+ * Creates a copy of existing animation object with given repetition.
+ **
+ > Parameters
+ **
+ - repeat (number) number iterations of animation. For infinite animation pass `Infinity`
+ **
+ = (object) new altered Animation object
+ \*/
+ Animation.prototype.repeat = function(times) {
+ var a = new Animation(this.anim, this.ms);
+ a.del = this.del;
+ a.times = math.floor(mmax(times, 0)) || 1;
+ return a;
+ };
+ function runAnimation(anim, element, percent, status, totalOrigin, times) {
+ percent = toFloat(percent);
+ var params,
+ isInAnim,
+ isInAnimSet,
+ percents = [],
+ next,
+ prev,
+ timestamp,
+ ms = anim.ms,
+ from = {},
+ to = {},
+ diff = {};
+ if (status) {
+ for (i = 0, ii = animationElements.length; i < ii; i++) {
+ var e = animationElements[i];
+ if (e.el.id == element.id && e.anim == anim) {
+ if (e.percent != percent) {
+ animationElements.splice(i, 1);
+ isInAnimSet = 1;
+ } else {
+ isInAnim = e;
+ }
+ element.attr(e.totalOrigin);
+ break;
+ }
+ }
+ } else {
+ status = +to; // NaN
+ }
+ for (var i = 0, ii = anim.percents.length; i < ii; i++) {
+ if (anim.percents[i] == percent || anim.percents[i] > status * anim.top) {
+ percent = anim.percents[i];
+ prev = anim.percents[i - 1] || 0;
+ ms = ms / anim.top * (percent - prev);
+ next = anim.percents[i + 1];
+ params = anim.anim[percent];
+ break;
+ } else if (status) {
+ element.attr(anim.anim[anim.percents[i]]);
+ }
+ }
+ if (!params) {
+ return;
+ }
+ if (!isInAnim) {
+ for (var attr in params)
+ if (params[has](attr)) {
+ if (availableAnimAttrs[has](attr) || element.ca[attr]) {
+ from[attr] = element.attr(attr);
+ (from[attr] == null) && (from[attr] = availableAttrs[attr]);
+ to[attr] = params[attr];
+ switch (availableAnimAttrs[attr]) {
+ case nu:
+ diff[attr] = (to[attr] - from[attr]) / ms;
+ break;
+ case "colour":
+ from[attr] = R.getRGB(from[attr]);
+ var toColour = R.getRGB(to[attr]);
+ diff[attr] = {
+ r: (toColour.r - from[attr].r) / ms,
+ g: (toColour.g - from[attr].g) / ms,
+ b: (toColour.b - from[attr].b) / ms
+ };
+ break;
+ case "path":
+ var pathes = path2curve(from[attr], to[attr]),
+ toPath = pathes[1];
+ from[attr] = pathes[0];
+ diff[attr] = [];
+ for (i = 0, ii = from[attr].length; i < ii; i++) {
+ diff[attr][i] = [0];
+ for (var j = 1, jj = from[attr][i].length; j < jj; j++) {
+ diff[attr][i][j] = (toPath[i][j] - from[attr][i][j]) / ms;
+ }
+ }
+ break;
+ case "transform":
+ var _ = element._,
+ eq = equaliseTransform(_[attr], to[attr]);
+ if (eq) {
+ from[attr] = eq.from;
+ to[attr] = eq.to;
+ diff[attr] = [];
+ diff[attr].real = true;
+ for (i = 0, ii = from[attr].length; i < ii; i++) {
+ diff[attr][i] = [from[attr][i][0]];
+ for (j = 1, jj = from[attr][i].length; j < jj; j++) {
+ diff[attr][i][j] = (to[attr][i][j] - from[attr][i][j]) / ms;
+ }
+ }
+ } else {
+ var m = (element.matrix || new Matrix),
+ to2 = {
+ _: {
+ transform: _.transform
+ },
+ getBBox: function() {
+ return element.getBBox(1);
+ }
+ };
+ from[attr] = [
+ m.a,
+ m.b,
+ m.c,
+ m.d,
+ m.e,
+ m.f
+ ];
+ extractTransform(to2, to[attr]);
+ to[attr] = to2._.transform;
+ diff[attr] = [
+ (to2.matrix.a - m.a) / ms,
+ (to2.matrix.b - m.b) / ms,
+ (to2.matrix.c - m.c) / ms,
+ (to2.matrix.d - m.d) / ms,
+ (to2.matrix.e - m.e) / ms,
+ (to2.matrix.f - m.f) / ms
+ ];
+ // from[attr] = [_.sx, _.sy, _.deg, _.dx, _.dy];
+ // var to2 = {_:{}, getBBox: function () { return element.getBBox(); }};
+ // extractTransform(to2, to[attr]);
+ // diff[attr] = [
+ // (to2._.sx - _.sx) / ms,
+ // (to2._.sy - _.sy) / ms,
+ // (to2._.deg - _.deg) / ms,
+ // (to2._.dx - _.dx) / ms,
+ // (to2._.dy - _.dy) / ms
+ // ];
+ }
+ break;
+ case "csv":
+ var values = Str(params[attr])[split](separator),
+ from2 = Str(from[attr])[split](separator);
+ if (attr == "clip-rect") {
+ from[attr] = from2;
+ diff[attr] = [];
+ i = from2.length;
+ while (i--) {
+ diff[attr][i] = (values[i] - from[attr][i]) / ms;
+ }
+ }
+ to[attr] = values;
+ break;
+ default:
+ values = [][concat](params[attr]);
+ from2 = [][concat](from[attr]);
+ diff[attr] = [];
+ i = element.ca[attr].length;
+ while (i--) {
+ diff[attr][i] = ((values[i] || 0) - (from2[i] || 0)) / ms;
+ }
+ break;
+ }
+ }
+ }
+ var easing = params.easing,
+ easyeasy = R.easing_formulas[easing];
+ if (!easyeasy) {
+ easyeasy = Str(easing).match(bezierrg);
+ if (easyeasy && easyeasy.length == 5) {
+ var curve = easyeasy;
+ easyeasy = function(t) {
+ return CubicBezierAtTime(t, +curve[1], +curve[2], +curve[3], +curve[4], ms);
+ };
+ } else {
+ easyeasy = pipe;
+ }
+ }
+ timestamp = params.start || anim.start || +new Date;
+ e = {
+ anim: anim,
+ percent: percent,
+ timestamp: timestamp,
+ start: timestamp + (anim.del || 0),
+ status: 0,
+ initstatus: status || 0,
+ stop: false,
+ ms: ms,
+ easing: easyeasy,
+ from: from,
+ diff: diff,
+ to: to,
+ el: element,
+ callback: params.callback,
+ prev: prev,
+ next: next,
+ repeat: times || anim.times,
+ origin: element.attr(),
+ totalOrigin: totalOrigin
+ };
+ animationElements.push(e);
+ if (status && !isInAnim && !isInAnimSet) {
+ e.stop = true;
+ e.start = new Date - ms * status;
+ if (animationElements.length == 1) {
+ return animation();
+ }
+ }
+ if (isInAnimSet) {
+ e.start = new Date - e.ms * status;
+ }
+ animationElements.length == 1 && requestAnimFrame(animation);
+ } else {
+ isInAnim.initstatus = status;
+ isInAnim.start = new Date - isInAnim.ms * status;
+ }
+ eve("raphael.anim.start." + element.id, element, anim);
+ }
+
+ /*\
+ * Raphael.animation
+ [ method ]
+ **
+ * Creates an animation object that can be passed to the @Element.animate or @Element.animateWith methods.
+ * See also @Animation.delay and @Animation.repeat methods.
+ **
+ > Parameters
+ **
+ - params (object) final attributes for the element, see also @Element.attr
+ - ms (number) number of milliseconds for animation to run
+ - easing (string) #optional easing type. Accept one of @Raphael.easing_formulas or CSS format: `cubic‐bezier(XX, XX, XX, XX)`
+ - callback (function) #optional callback function. Will be called at the end of animation.
+ **
+ = (object) @Animation
+ \*/
+ R.animation = function(params, ms, easing, callback) {
+ if (params instanceof Animation) {
+ return params;
+ }
+ if (R.is(easing, "function") || !easing) {
+ callback = callback || easing || null;
+ easing = null;
+ }
+ params = Object(params);
+ ms = +ms || 0;
+ var p = {},
+ json,
+ attr;
+ for (attr in params)
+ if (params[has](attr) && toFloat(attr) != attr && toFloat(attr) + "%" != attr) {
+ json = true;
+ p[attr] = params[attr];
+ }
+ if (!json) {
+ return new Animation(params, ms);
+ } else {
+ easing && (p.easing = easing);
+ callback && (p.callback = callback);
+ return new Animation({
+ 100: p
+ }, ms);
+ }
+ };
+
+ /*\
+ * Element.animate
+ [ method ]
+ **
+ * Creates and starts animation for given element.
+ **
+ > Parameters
+ **
+ - params (object) final attributes for the element, see also @Element.attr
+ - ms (number) number of milliseconds for animation to run
+ - easing (string) #optional easing type. Accept one of @Raphael.easing_formulas or CSS format: `cubic‐bezier(XX, XX, XX, XX)`
+ - callback (function) #optional callback function. Will be called at the end of animation.
+ * or
+ - animation (object) animation object, see @Raphael.animation
+ **
+ = (object) original element
+ \*/
+ elproto.animate = function(params, ms, easing, callback) {
+ var element = this;
+ if (element.removed) {
+ callback && callback.call(element);
+ return element;
+ }
+ var anim = params instanceof Animation ? params : R.animation(params, ms, easing, callback);
+ runAnimation(anim, element, anim.percents[0], null, element.attr());
+ return element;
+ };
+
+ /*\
+ * Element.setTime
+ [ method ]
+ **
+ * Sets the status of animation of the element in milliseconds. Similar to @Element.status method.
+ **
+ > Parameters
+ **
+ - anim (object) animation object
+ - value (number) number of milliseconds from the beginning of the animation
+ **
+ = (object) original element if `value` is specified
+ * Note, that during animation following events are triggered:
+ *
+ * On each animation frame event `anim.frame.`, on start `anim.start.` and on end `anim.finish.`.
+ \*/
+ elproto.setTime = function(anim, value) {
+ if (anim && value != null) {
+ this.status(anim, mmin(value, anim.ms) / anim.ms);
+ }
+ return this;
+ };
+
+ /*\
+ * Element.status
+ [ method ]
+ **
+ * Gets or sets the status of animation of the element.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ - value (number) #optional 0 – 1. If specified, method works like a setter and sets the status of a given animation to the value. This will cause animation to jump to the given position.
+ **
+ = (number) status
+ * or
+ = (array) status if `anim` is not specified. Array of objects in format:
+ o {
+ o anim: (object) animation object
+ o status: (number) status
+ o }
+ * or
+ = (object) original element if `value` is specified
+ \*/
+ elproto.status = function(anim, value) {
+ var out = [],
+ i = 0,
+ len,
+ e;
+ if (value != null) {
+ runAnimation(anim, this, -1, mmin(value, 1));
+ return this;
+ } else {
+ len = animationElements.length;
+ for (; i < len; i++) {
+ e = animationElements[i];
+ if (e.el.id == this.id && (!anim || e.anim == anim)) {
+ if (anim) {
+ return e.status;
+ }
+ out.push({
+ anim: e.anim,
+ status: e.status
+ });
+ }
+ }
+ if (anim) {
+ return 0;
+ }
+ return out;
+ }
+ };
+
+ /*\
+ * Element.pause
+ [ method ]
+ **
+ * Stops animation of the element with ability to resume it later on.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ **
+ = (object) original element
+ \*/
+ elproto.pause = function(anim) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.id == this.id && (!anim || animationElements[i].anim == anim)) {
+ if (eve("raphael.anim.pause." + this.id, this, animationElements[i].anim) !== false) {
+ animationElements[i].paused = true;
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Element.resume
+ [ method ]
+ **
+ * Resumes animation if it was paused with @Element.pause method.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ **
+ = (object) original element
+ \*/
+ elproto.resume = function(anim) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.id == this.id && (!anim || animationElements[i].anim == anim)) {
+ var e = animationElements[i];
+ if (eve("raphael.anim.resume." + this.id, this, e.anim) !== false) {
+ delete e.paused;
+ this.status(e.anim, e.status);
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Element.stop
+ [ method ]
+ **
+ * Stops animation of the element.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ **
+ = (object) original element
+ \*/
+ elproto.stop = function(anim) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.id == this.id && (!anim || animationElements[i].anim == anim)) {
+ if (eve("raphael.anim.stop." + this.id, this, animationElements[i].anim) !== false) {
+ animationElements.splice(i--, 1);
+ }
+ }
+ return this;
+ };
+ function stopAnimation(paper) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.paper == paper) {
+ animationElements.splice(i--, 1);
+ }
+ }
+ eve.on("raphael.remove", stopAnimation);
+ eve.on("raphael.clear", stopAnimation);
+ elproto.toString = function() {
+ return "Rapha\xebl\u2019s object";
+ };
+
+ elproto.toFront = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ parent = o.parent,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._tofront(o, parent)) {
+ parent.canvas.appendChild(thisNode);
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+ return o;
+ };
+
+ elproto.toBack = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ parent = o.parent,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._toback(o, parent)) {
+ parent.canvas.insertBefore(thisNode, parent.canvas.firstChild);
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+ return o;
+ };
+
+ elproto.insertAfter = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ thatNode = R._engine.getLastNode(element),
+ parentNode = element.parent.canvas,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (thatNode.nextSibling) {
+ parentNode.insertBefore(thisNode, thatNode.nextSibling);
+ }
+ else {
+ parentNode.appendChild(thisNode);
+ }
+ R._insertafter(o, element, o.parent, element.parent);
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+ return o;
+ };
+
+ elproto.insertBefore = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ thatNode = R._engine.getNode(element),
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ element.parent.canvas.insertBefore(thisNode, thatNode);
+ R._insertbefore(o, element, o.parent, element.parent);
+ o.parent = element.parent;
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+ return this;
+ };
+
+ elproto.appendChild = function (element) {
+ if (this.removed || this.type !== 'group') {
+ return this;
+ }
+
+ var group = this,
+ followers = group.followers,
+ follower,
+ thatNode,
+ i,
+ ii;
+
+ // If appending in same group, simply perform toFront().
+ if (element.parent === group) {
+ element.toFront();
+ return group;
+ }
+
+ thatNode = R._engine.getNode(element);
+
+ // first remove from own group
+ R._tear(element, element.parent);
+
+ group.canvas.appendChild(thatNode);
+ element.parent = group;
+
+ !group.bottom && (group.bottom = element);
+ element.prev = group.top;
+ element.next = null;
+ group.top && (group.top.next = element);
+ group.top = element;
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+
+ return group;
+ };
+
+ elproto.removeChild = function (element) {
+ if (this.removed || this.type !== 'group' || element.parent !== this) {
+ return this;
+ }
+
+ var o = this,
+ thatNode = R._engine.getNode(element),
+ paper = o.paper;
+
+ R._tear(element, o);
+ paper.canvas.appendChild(thatNode);
+
+ o.parent = paper;
+ !paper.bottom && (paper.bottom = o);
+
+ o.prev = paper.top;
+ paper.top && (paper.top.next = o);
+ paper.top = o;
+ o.next = null;
+
+ return o;
+ };
+
+ // Set
+ var Set = function(items) {
+ this.items = [];
+ this.length = 0;
+ this.type = "set";
+ if (items) {
+ for (var i = 0, ii = items.length; i < ii; i++) {
+ if (items[i] && (items[i].constructor == elproto.constructor || items[i].constructor == Set)) {
+ this[this.items.length] = this.items[this.items.length] = items[i];
+ this.length++;
+ }
+ }
+ }
+ },
+ setproto = Set.prototype;
+
+ /*\
+ * Set.push
+ [ method ]
+ **
+ * Adds each argument to the current set.
+ = (object) original element
+ \*/
+ setproto.push = function() {
+ var item,
+ len;
+ for (var i = 0, ii = arguments.length; i < ii; i++) {
+ item = arguments[i];
+ if (item && (item.constructor == elproto.constructor || item.constructor == Set)) {
+ len = this.items.length;
+ this[len] = this.items[len] = item;
+ this.length++;
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Set.pop
+ [ method ]
+ **
+ * Removes last element and returns it.
+ = (object) element
+ \*/
+ setproto.pop = function() {
+ this.length && delete this[this.length--];
+ return this.items.pop();
+ };
+
+ /*\
+ * Set.forEach
+ [ method ]
+ **
+ * Executes given function for each element in the set.
+ *
+ * If function returns `false` it will stop loop running.
+ **
+ > Parameters
+ **
+ - callback (function) function to run
+ - thisArg (object) context object for the callback
+ = (object) Set object
+ \*/
+ setproto.forEach = function(callback, thisArg) {
+ for (var i = 0, ii = this.items.length; i < ii; i++) {
+ if (callback.call(thisArg, this.items[i], i) === false) {
+ return this;
+ }
+ }
+ return this;
+ };
+ for (var method in elproto)
+ if (elproto[has](method)) {
+ setproto[method] = (function(methodname) {
+ return function() {
+ var arg = arguments;
+ return this.forEach(function(el) {
+ el[methodname][apply](el, arg);
+ });
+ };
+ })(method);
+ }
+ setproto.attr = function(name, value) {
+ if (name && R.is(name, array) && R.is(name[0], object)) {
+ for (var j = 0, jj = name.length; j < jj; j++) {
+ this.items[j].attr(name[j]);
+ }
+ } else {
+ for (var i = 0, ii = this.items.length; i < ii; i++) {
+ this.items[i].attr(name, value);
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Set.clear
+ [ method ]
+ **
+ * Removeds all elements from the set
+ \*/
+ setproto.clear = function() {
+ while (this.length) {
+ this.pop();
+ }
+ };
+
+ /*\
+ * Set.splice
+ [ method ]
+ **
+ * Removes given element from the set
+ **
+ > Parameters
+ **
+ - index (number) position of the deletion
+ - count (number) number of element to remove
+ - insertion… (object) #optional elements to insert
+ = (object) set elements that were deleted
+ \*/
+ setproto.splice = function(index, count, insertion) {
+ index = index < 0 ? mmax(this.length + index, 0) : index;
+ count = mmax(0, mmin(this.length - index, isNaN(count) && this.length || count));
+ var tail = [],
+ todel = [],
+ args = [],
+ i;
+ for (i = 2; i < arguments.length; i++) {
+ args.push(arguments[i]);
+ }
+ for (i = 0; i < count; i++) {
+ todel.push(this[index + i]);
+ }
+ for (; i < this.length - index; i++) {
+ tail.push(this[index + i]);
+ }
+ var arglen = args.length;
+ for (i = 0; i < arglen + tail.length; i++) {
+ this.items[index + i] = this[index + i] = i < arglen ? args[i] : tail[i - arglen];
+ }
+ i = this.items.length = this.length -= count - arglen;
+ while (this[i]) {
+ delete this[i++];
+ }
+ return new Set(todel);
+ };
+
+ /*\
+ * Set.exclude
+ [ method ]
+ **
+ * Removes given element from the set
+ **
+ > Parameters
+ **
+ - element (object) element to remove
+ = (boolean) `true` if object was found & removed from the set
+ \*/
+ setproto.exclude = function(el) {
+ for (var i = 0, ii = this.length; i < ii; i++)
+ if (this[i] == el) {
+ this.splice(i, 1);
+ return true;
+ }
+ };
+ setproto.animate = function(params, ms, easing, callback) {
+ (R.is(easing, "function") || !easing) && (callback = easing || null);
+ var len = this.items.length,
+ i = len,
+ item,
+ set = this,
+ collector;
+ if (!len) {
+ return this;
+ }
+ callback && (collector = function() {
+ !--len && callback.call(set);
+ });
+ easing = R.is(easing, string) ? easing : collector;
+ var anim = R.animation(params, ms, easing, collector);
+ item = this.items[--i].animate(anim);
+ while (i--) {
+ this.items[i] && !this.items[i].removed && this.items[i].animateWith(item, anim, anim);
+ }
+ return this;
+ };
+ setproto.insertAfter = function(el) {
+ var i = this.items.length;
+ while (i--) {
+ this.items[i].insertAfter(el);
+ }
+ return this;
+ };
+ setproto.getBBox = function() {
+ var x = [],
+ y = [],
+ x2 = [],
+ y2 = [];
+ for (var i = this.items.length; i--; )
+ if (!this.items[i].removed) {
+ var box = this.items[i].getBBox();
+ x.push(box.x);
+ y.push(box.y);
+ x2.push(box.x + box.width);
+ y2.push(box.y + box.height);
+ }
+ x = mmin[apply](0, x);
+ y = mmin[apply](0, y);
+ x2 = mmax[apply](0, x2);
+ y2 = mmax[apply](0, y2);
+ return {
+ x: x,
+ y: y,
+ x2: x2,
+ y2: y2,
+ width: x2 - x,
+ height: y2 - y
+ };
+ };
+ setproto.clone = function(s) {
+ s = new Set;
+ for (var i = 0, ii = this.items.length; i < ii; i++) {
+ s.push(this.items[i].clone());
+ }
+ return s;
+ };
+ setproto.toString = function() {
+ return "Rapha\xebl\u2018s set";
+ };
+
+ setproto.glow = function(glowConfig) {
+ var ret = this.paper.set();
+ this.forEach(function(shape, index){
+ var g = shape.glow(glowConfig);
+ if(g != null){
+ g.forEach(function(shape2, index2){
+ ret.push(shape2);
+ });
+ }
+ });
+ return ret;
+ };
+
+ /*\
+ * Raphael.registerFont
+ [ method ]
+ **
+ * Adds given font to the registered set of fonts for Raphaël. Should be used as an internal call from within Cufón’s font file.
+ * Returns original parameter, so it could be used with chaining.
+ # More about Cufón and how to convert your font form TTF, OTF, etc to JavaScript file.
+ **
+ > Parameters
+ **
+ - font (object) the font to register
+ = (object) the font you passed in
+ > Usage
+ | Cufon.registerFont(Raphael.registerFont({…}));
+ \*/
+ R.registerFont = function(font) {
+ if (!font.face) {
+ return font;
+ }
+ this.fonts = this.fonts || {};
+ var fontcopy = {
+ w: font.w,
+ face: {},
+ glyphs: {}
+ },
+ family = font.face["font-family"];
+ for (var prop in font.face)
+ if (font.face[has](prop)) {
+ fontcopy.face[prop] = font.face[prop];
+ }
+ if (this.fonts[family]) {
+ this.fonts[family].push(fontcopy);
+ } else {
+ this.fonts[family] = [fontcopy];
+ }
+ if (!font.svg) {
+ fontcopy.face["units-per-em"] = toInt(font.face["units-per-em"], 10);
+ for (var glyph in font.glyphs)
+ if (font.glyphs[has](glyph)) {
+ var path = font.glyphs[glyph];
+ fontcopy.glyphs[glyph] = {
+ w: path.w,
+ k: {},
+ d: path.d && "M" + path.d.replace(/[mlcxtrv]/g, function(command) {
+ return {
+ l: "L",
+ c: "C",
+ x: "z",
+ t: "m",
+ r: "l",
+ v: "c"
+ }
+ [command] || "M";
+ }) + "z"
+ };
+ if (path.k) {
+ for (var k in path.k)
+ if (path[has](k)) {
+ fontcopy.glyphs[glyph].k[k] = path.k[k];
+ }
+ }
+ }
+ }
+ return font;
+ };
+
+ /*\
+ * Paper.getFont
+ [ method ]
+ **
+ * Finds font object in the registered fonts by given parameters. You could specify only one word from the font name, like “Myriad” for “Myriad Pro”.
+ **
+ > Parameters
+ **
+ - family (string) font family name or any word from it
+ - weight (string) #optional font weight
+ - style (string) #optional font style
+ - stretch (string) #optional font stretch
+ = (object) the font object
+ > Usage
+ | paper.print(100, 100, "Test string", paper.getFont("Times", 800), 30);
+ \*/
+ paperproto.getFont = function(family, weight, style, stretch) {
+ stretch = stretch || "normal";
+ style = style || "normal";
+ weight = +weight || {
+ normal: 400,
+ bold: 700,
+ lighter: 300,
+ bolder: 800
+ }
+ [weight] || 400;
+ if (!R.fonts) {
+ return;
+ }
+ var font = R.fonts[family];
+ if (!font) {
+ var name = new RegExp("(^|\\s)" + family.replace(/[^\w\d\s+!~.:_-]/g, E) + "(\\s|$)", "i");
+ for (var fontName in R.fonts)
+ if (R.fonts[has](fontName)) {
+ if (name.test(fontName)) {
+ font = R.fonts[fontName];
+ break;
+ }
+ }
+ }
+ var thefont;
+ if (font) {
+ for (var i = 0, ii = font.length; i < ii; i++) {
+ thefont = font[i];
+ if (thefont.face["font-weight"] == weight && (thefont.face["font-style"] == style || !thefont.face["font-style"]) && thefont.face["font-stretch"] == stretch) {
+ break;
+ }
+ }
+ }
+ return thefont;
+ };
+
+ /*\
+ * Paper.print
+ [ method ]
+ **
+ * Creates path that represent given text written using given font at given position with given size.
+ * Result of the method is path element that contains whole text as a separate path.
+ **
+ > Parameters
+ **
+ - x (number) x position of the text
+ - y (number) y position of the text
+ - string (string) text to print
+ - font (object) font object, see @Paper.getFont
+ - size (number) #optional size of the font, default is `16`
+ - origin (string) #optional could be `"baseline"` or `"middle"`, default is `"middle"`
+ - letter_spacing (number) #optional number in range `-1..1`, default is `0`
+ = (object) resulting path element, which consist of all letters
+ > Usage
+ | var txt = r.print(10, 50, "print", r.getFont("Museo"), 30).attr({fill: "#fff"});
+ \*/
+ paperproto.print = function(x, y, string, font, size, origin, letter_spacing) {
+ origin = origin || "middle"; // baseline|middle
+ letter_spacing = mmax(mmin(letter_spacing || 0, 1), -1);
+ var letters = Str(string)[split](E),
+ shift = 0,
+ notfirst = 0,
+ path = E,
+ scale;
+ R.is(font, string) && (font = this.getFont(font));
+ if (font) {
+ scale = (size || 16) / font.face["units-per-em"];
+ var bb = font.face.bbox[split](separator),
+ top = +bb[0],
+ lineHeight = bb[3] - bb[1],
+ shifty = 0,
+ height = + bb[1] + (origin == "baseline" ? lineHeight + ( + font.face.descent) : lineHeight / 2);
+ for (var i = 0, ii = letters.length; i < ii; i++) {
+ if (letters[i] == "\n") {
+ shift = 0;
+ curr = 0;
+ notfirst = 0;
+ shifty += lineHeight;
+ } else {
+ var prev = notfirst && font.glyphs[letters[i - 1]] || {},
+ curr = font.glyphs[letters[i]];
+ shift += notfirst ? (prev.w || font.w) + (prev.k && prev.k[letters[i]] || 0) + (font.w * letter_spacing) : 0;
+ notfirst = 1;
+ }
+ if (curr && curr.d) {
+ path += R.transformPath(curr.d, ["t", shift * scale, shifty * scale, "s", scale, scale, top, height, "t", (x - top) / scale, (y - height) / scale]);
+ }
+ }
+ }
+ return this.path(path).attr({
+ fill: "#000",
+ stroke: "none"
+ });
+ };
+
+ /*\
+ * Paper.add
+ [ method ]
+ **
+ * Imports elements in JSON array in format `{type: type, }`
+ **
+ > Parameters
+ **
+ - json (array)
+ = (object) resulting set of imported elements
+ > Usage
+ | paper.add([
+ | {
+ | type: "circle",
+ | cx: 10,
+ | cy: 10,
+ | r: 5
+ | },
+ | {
+ | type: "rect",
+ | x: 10,
+ | y: 10,
+ | width: 10,
+ | height: 10,
+ | fill: "#fc0"
+ | }
+ | ]);
+ \*/
+ paperproto.add = function(json) {
+ if (R.is(json, "array")) {
+ var res = this.set(),
+ i = 0,
+ ii = json.length,
+ j;
+ for (; i < ii; i++) {
+ j = json[i] || {};
+ elements[has](j.type) && res.push(this[j.type]().attr(j));
+ }
+ }
+ return res;
+ };
+
+ /*\
+ * Raphael.format
+ [ method ]
+ **
+ * Simple format function. Replaces construction of type “`{}`” to the corresponding argument.
+ **
+ > Parameters
+ **
+ - token (string) string to format
+ - … (string) rest of arguments will be treated as parameters for replacement
+ = (string) formated string
+ > Usage
+ | var x = 10,
+ | y = 20,
+ | width = 40,
+ | height = 50;
+ | // this will draw a rectangular shape equivalent to "M10,20h40v50h-40z"
+ | paper.path(Raphael.format("M{0},{1}h{2}v{3}h{4}z", x, y, width, height, -width));
+ \*/
+ R.format = function(token, params) {
+ var args = R.is(params, array) ? [0][concat](params) : arguments;
+ token && R.is(token, string) && args.length - 1 && (token = token.replace(formatrg, function(str, i) {
+ return args[++i] == null ? E : args[i];
+ }));
+ return token || E;
+ };
+
+ /*\
+ * Raphael.fullfill
+ [ method ]
+ **
+ * A little bit more advanced format function than @Raphael.format. Replaces construction of type “`{}`” to the corresponding argument.
+ **
+ > Parameters
+ **
+ - token (string) string to format
+ - json (object) object which properties will be used as a replacement
+ = (string) formated string
+ > Usage
+ | // this will draw a rectangular shape equivalent to "M10,20h40v50h-40z"
+ | paper.path(Raphael.fullfill("M{x},{y}h{dim.width}v{dim.height}h{dim['negative width']}z", {
+ | x: 10,
+ | y: 20,
+ | dim: {
+ | width: 40,
+ | height: 50,
+ | "negative width": -40
+ | }
+ | }));
+ \*/
+ R.fullfill = (function() {
+ var tokenRegex = /\{([^\}]+)\}/g,
+ objNotationRegex = /(?:(?:^|\.)(.+?)(?=\[|\.|$|\()|\[('|")(.+?)\2\])(\(\))?/g, // matches .xxxxx or ["xxxxx"] to run over object properties
+ replacer = function(all, key, obj) {
+ var res = obj;
+ key.replace(objNotationRegex, function(all, name, quote, quotedName, isFunc) {
+ name = name || quotedName;
+ if (res) {
+ if (name in res) {
+ res = res[name];
+ }
+ typeof res == "function" && isFunc && (res = res());
+ }
+ });
+ res = (res == null || res == obj ? all : res) + "";
+ return res;
+ };
+ return function(str, obj) {
+ return String(str).replace(tokenRegex, function(all, key) {
+ return replacer(all, key, obj);
+ });
+ };
+ })();
+
+ /*\
+ * Raphael.ninja
+ [ method ]
+ **
+ * If you want to leave no trace of Raphaël (Well, Raphaël creates only one global variable `Raphael`, but anyway.) You can use `ninja` method.
+ * Beware, that in this case plugins could stop working, because they are depending on global variable existance.
+ **
+ = (object) Raphael object
+ > Usage
+ | (function (local_raphael) {
+ | var paper = local_raphael(10, 10, 320, 200);
+ | …
+ | })(Raphael.ninja());
+ \*/
+ R.ninja = function() {
+ oldRaphael.was ? (g.win.Raphael = oldRaphael.is) : delete Raphael;
+ return R;
+ };
+
+ var crispFixer = (R.vml && 0.5 || 0);
+
+ R.crispBound = cacher(function (x, y, w, h, s) {
+ var at = {},
+ normalizer;
+
+ x = x || 0;
+ y = y || 0;
+ w = w || 0;
+ h = h || 0;
+ s = s || 0;
+ normalizer = s % 2 / 2 + crispFixer;
+
+ // normalize for crisp edges
+ at.x = round(x + normalizer) - normalizer;
+ at.y = round(y + normalizer) - normalizer;
+ at.width = round(x + w + normalizer) - normalizer - at.x;
+ at.height = round(y + h + normalizer) - normalizer - at.y;
+ at['stroke-width'] = s;
+
+ // adjust to single pixel if resultant dimension is zero.
+ (at.width === 0 && w !== 0) && (at.width = 1);
+ (at.height === 0 && h !== 0) && (at.height = 1);
+
+ return at;
+ }, R);
+
+ elproto.crisp = function () {
+ var o = this,
+ attrs = o.attrs,
+ key,
+ attr = {},
+ values = o.attr(['x', 'y', 'width', 'height', 'stroke-width']);
+
+ values = R.crispBound(values.x, values.y, values.width, values.height,
+ values['stroke-width']);
+
+ for (key in values) {
+ if (attrs[key] === values[key]) { // only set attribute if changed
+ delete values[key];
+ }
+ }
+
+ return o.attr(values);
+ };
+
+ /*\
+ * Raphael.st
+ [ property (object) ]
+ **
+ * You can add your own method to elements and sets. It is wise to add a set method for each element method
+ * you added, so you will be able to call the same method on sets too.
+ **
+ * See also @Raphael.el.
+ > Usage
+ | Raphael.el.red = function () {
+ | this.attr({fill: "#f00"});
+ | };
+ | Raphael.st.red = function () {
+ | this.forEach(function (el) {
+ | el.red();
+ | });
+ | };
+ | // then use it
+ | paper.set(paper.circle(100, 100, 20), paper.circle(110, 100, 20)).red();
+ \*/
+ R.st = setproto;
+
+ /*\
+ * Raphael.define
+ [ method ]
+ **
+ * Allows a unified definition of composite shapes and other behaviours using
+ * simple directives.
+ **
+ > Parameters
+ **
+ - definition (object) the shape definition
+ \*/
+ R.define = function (name, init, ca, fn, e) {
+ var i,
+ ii;
+
+ // multi definition
+ if (R.is(name, array)) {
+ for (i = 0, ii = name.length; i < ii; i++) {
+ R.define(name[i]);
+ }
+ return;
+ }
+ // object definition
+ else if (R.is(name, object)) {
+ R.define(name.name, name[name.name], name.ca, name.fn, name.e);
+ return;
+ }
+ // invalid or duplicate definition
+ else if (!name || R.fn[name]) {
+ return;
+ }
+
+ R.fn[name] = function () {
+ var args = arguments,
+ element = init.apply(this, args),
+ key;
+
+ if (fn && R.is(fn, object)) {
+ for (key in fn) {
+ element[key] = fn[key];
+ }
+ }
+
+ if (e && R.is(e, object)) {
+ for (key in e) {
+ element[key] && element[key](e[key]);
+ }
+ }
+
+ if (ca) {
+ if (R.is(ca, 'function')) {
+ element.ca[name] = ca;
+ }
+ else {
+ for (key in ca) {
+ element.ca[key] = ca[key];
+ }
+ }
+
+ // Check if namesake ca exists and apply it
+ if (element.ca[name]) {
+ R._lastArgIfGroup(args, true); // purge group
+ element.attr(name, arraySlice.call(args))
+ }
+ }
+
+ return element;
+ };
+
+ if (ca) { R.fn[name].ca = ca; }
+ if (fn) { R.fn[name].fn = fn; }
+ if (e) { R.fn[name].e = e; }
+
+ return R.fn[name];
+ };
+ // Firefox <3.6 fix: http://webreflection.blogspot.com/2009/11/195-chars-to-help-lazy-loading.html
+ (function(doc, loaded, f) {
+ if (doc.readyState == null && doc.addEventListener) {
+ doc.addEventListener(loaded, f = function() {
+ doc.removeEventListener(loaded, f, false);
+ doc.readyState = "complete";
+ }, false);
+ doc.readyState = "loading";
+ }
+ function isLoaded() {
+ (/in/).test(doc.readyState) ? setTimeout(isLoaded, 9) : R.eve("raphael.DOMload");
+ }
+ isLoaded();
+ })(document, "DOMContentLoaded");
+
+ eve.on("raphael.DOMload", function() {
+ loaded = true;
+ });
+
+/**!
+* RedRaphael 1.0.0 - JavaScript Vector Library SVG Module
+* Copyright (c) 2012-2013 FusionCharts Technologies
+*
+* Raphael 2.1.0 - JavaScript Vector Library SVG Module
+* Copyright (c) 2008-2012 Dmitry Baranovskiy
+* Copyright © 2008-2012 Sencha Labs
+*
+* Licensed under the MIT license.
+*/
+(function(){
+ if (!R.svg) {
+ return;
+ }
+ var has = "hasOwnProperty",
+ Str = String,
+ toFloat = parseFloat,
+ toInt = parseInt,
+ math = Math,
+ mmax = math.max,
+ abs = math.abs,
+ pow = math.pow,
+ sqrt = math.sqrt,
+ separator = /[, ]+/,
+ zeroStrokeFix = !!(/AppleWebKit/.test(R._g.win.navigator.userAgent) &&
+ (!/Chrome/.test(R._g.win.navigator.userAgent) ||
+ R._g.win.navigator.appVersion.match(/Chrome\/(\d+)\./)[1] < 29)),
+ eve = R.eve,
+ E = "",
+ S = " ",
+ xlink = "http://www.w3.org/1999/xlink",
+ markers = {
+ block: "M5,0 0,2.5 5,5z",
+ classic: "M5,0 0,2.5 5,5 3.5,3 3.5,2z",
+ diamond: "M2.5,0 5,2.5 2.5,5 0,2.5z",
+ open: "M6,1 1,3.5 6,6",
+ oval: "M2.5,0A2.5,2.5,0,0,1,2.5,5 2.5,2.5,0,0,1,2.5,0z"
+ },
+ markerCounter = {},
+ updateReferenceUrl = function () {
+ return R._url = R._g.win.location.href.replace(/#.*?$/, E);
+ };
+
+ R.toString = function() {
+ return "Your browser supports SVG.\nYou are running Rapha\xebl " + this.version;
+ };
+
+ // Automatic gradient and other reference update on state change
+ R._url = (/msie/i.test(navigator.userAgent) && !window.opera) ?
+ E : updateReferenceUrl();
+ if (R._url && R._g.win.history.pushState) {
+ R._g.win.history.pushState = (function () {
+ var fn = R._g.win.history.pushState;
+ return function () {
+ var ret = fn.apply(R._g.win.history, arguments);
+ return updateReferenceUrl(), ret;
+ };
+ }());
+ R._g.win.addEventListener("popstate", updateReferenceUrl, false);
+ }
+
+ var $ = R._createNode = function(el, attr) {
+ if (attr) {
+ if (typeof el == "string") {
+ el = $(el);
+ }
+ for (var key in attr)
+ if (attr[has](key)) {
+ if (key.substring(0, 6) == "xlink:") {
+ el.setAttributeNS(xlink, key.substring(6), Str(attr[key]));
+ } else {
+ el.setAttribute(key, Str(attr[key]));
+ }
+ }
+ } else {
+ el = R._g.doc.createElementNS("http://www.w3.org/2000/svg", el);
+ }
+ return el;
+ },
+ gradientUnitNames = {
+ userSpaceOnUse: 'userSpaceOnUse',
+ objectBoundingBox: 'objectBoundingBox'
+ },
+ gradientSpreadNames = {
+ pad: 'pad',
+ redlect: 'reflect',
+ repeat: 'repeat'
+ },
+ addGradientFill = function(element, gradient) {
+ var type = "linear",
+ id = element.id + gradient,
+ fx = .5, fy = .5, r, cx, cy, units, spread,
+ o = element.node,
+ SVG = element.paper,
+ s = o.style,
+ el = R._g.doc.getElementById(id);
+ if (!el && SVG.defs) {
+ gradient = Str(gradient).replace(R._radial_gradient, function(all, opts) {
+ type = "radial";
+ opts = opts && opts.split(',') || [];
+ units = opts[5];
+ spread = opts[6];
+
+ var _fx = opts[0],
+ _fy = opts[1],
+ _r = opts[2],
+ _cx = opts[3],
+ _cy = opts[4],
+ shifted = (_fx && _fy),
+ dir,
+ sqx;
+
+ if (_r) {
+ r = /\%/.test(_r) ? _r : toFloat(_r);
+ }
+
+ if (units === gradientUnitNames.userSpaceOnUse) {
+ if (shifted) {
+ fx = _fx;
+ fy = _fy;
+ }
+ if (_cx && _cy) {
+ cx = _cx;
+ cy = _cy;
+ if (!shifted) {
+ fx = cx;
+ fy = cy;
+ }
+ }
+ return E;
+ }
+
+ if (shifted) {
+ fx = toFloat(_fx);
+ fy = toFloat(_fy);
+ dir = ((fy > .5) * 2 - 1);
+ (sqx = pow(fx - .5, 2)) + pow(fy - .5, 2) > .25 &&
+ (sqx < .25) && (fy = sqrt(.25 - sqx) * dir + .5) &&
+ fy !== .5 &&
+ (fy = fy.toFixed(5) - 1e-5 * dir);
+ }
+ if (_cx && _cy) {
+ cx = toFloat(_cx);
+ cy = toFloat(_cy);
+ dir = ((cy > .5) * 2 - 1);
+
+ (sqx = pow(cx - .5, 2)) + pow(cy - .5, 2) > .25 &&
+ (sqx < .25) && (cy = sqrt(.25 - sqx) * dir + .5) &&
+ cy !== .5 &&
+ (cy = cy.toFixed(5) - 1e-5 * dir);
+
+ if (!shifted) {
+ fx = cx;
+ fy = cy;
+ }
+ }
+
+ return E;
+ });
+ gradient = gradient.split(/\s*\-\s*/);
+ if (type == "linear") {
+ var angle = gradient.shift(),
+ specs = angle.match(/\((.*)\)/),
+ vector,
+ max;
+
+ specs = specs && specs[1] && specs[1].split(/\s*\,\s*/);
+ angle = -toFloat(angle);
+ if (isNaN(angle)) {
+ return null;
+ }
+ if (specs && specs.length) {
+ if (specs[0] in gradientUnitNames) {
+ units = specs.shift();
+ (specs[0] in gradientSpreadNames) &&
+ (spread = specs.shift());
+ }
+ else {
+ specs[4] && (units = specs[4]);
+ specs[5] && (spread = specs[5]);
+ }
+
+ /* @todo apply angle rotation and validation */
+ vector = [
+ specs[0] || "0%", specs[1] || "0%",
+ specs[2] || "100%", specs[3] || "0%"
+ ];
+ }
+ else {
+ vector = [0, 0, math.cos(R.rad(angle)), math.sin(R.rad(angle))];
+ max = 1 / (mmax(abs(vector[2]), abs(vector[3])) || 1);
+ vector[2] *= max;
+ vector[3] *= max;
+ if (vector[2] < 0) {
+ vector[0] = -vector[2];
+ vector[2] = 0;
+ }
+ if (vector[3] < 0) {
+ vector[1] = -vector[3];
+ vector[3] = 0;
+ }
+ }
+ }
+ var dots = R._parseDots(gradient);
+ if (!dots) {
+ return null;
+ }
+ id = id.replace(/[\(\)\s,\xb0#]/g, "_");
+
+ if (element.gradient && id !== element.gradient.id) {
+ SVG.defs.removeChild(element.gradient);
+ delete element.gradient;
+ }
+
+ if (!element.gradient) {
+ el = $(type + "Gradient", {
+ id: id
+ });
+ element.gradient = el;
+ (units in gradientUnitNames) &&
+ el.setAttribute('gradientUnits', Str(units));
+ (spread in gradientSpreadNames) &&
+ el.setAttribute('spreadMethod', Str(spread));
+ if (type === "radial") {
+ (r !== undefined) && el.setAttribute('r', Str(r));
+
+ if (cx !== undefined && cy !== undefined) {
+ el.setAttribute('cx', Str(cx));
+ el.setAttribute('cy', Str(cy));
+ }
+ el.setAttribute('fx', Str(fx));
+ el.setAttribute('fy', Str(fy));
+ }
+ else {
+ $(el, {
+ x1: vector[0],
+ y1: vector[1],
+ x2: vector[2],
+ y2: vector[3],
+ gradientTransform: element.matrix.invert()
+ });
+ }
+ SVG.defs.appendChild(el);
+ for (var i = 0, ii = dots.length; i < ii; i++) {
+ el.appendChild($("stop", {
+ offset: dots[i].offset ? dots[i].offset : i ? "100%" : "0%",
+ "stop-color": dots[i].color || "#fff",
+ //add stop opacity information
+ "stop-opacity": dots[i].opacity === undefined ? 1 : dots[i].opacity
+ }));
+ }
+ }
+ }
+ $(o, {
+ fill: "url('" + R._url + "#" + id + "')",
+ opacity: 1,
+ "fill-opacity": 1
+ });
+ s.fill = E;
+ s.opacity = 1;
+ s.fillOpacity = 1;
+ return 1;
+ },
+ updatePosition = function(o) {
+ var bbox = o.getBBox(1);
+ $(o.pattern, {
+ patternTransform: o.matrix.invert() + " translate(" + bbox.x + "," + bbox.y + ")"
+ });
+ },
+ addArrow = function(o, value, isEnd) {
+ if (o.type == "path") {
+ var values = Str(value).toLowerCase().split("-"),
+ p = o.paper,
+ se = isEnd ? "end" : "start",
+ node = o.node,
+ attrs = o.attrs,
+ stroke = attrs["stroke-width"],
+ i = values.length,
+ type = "classic",
+ from,
+ to,
+ dx,
+ refX,
+ attr,
+ w = 3,
+ h = 3,
+ t = 5;
+ while (i--) {
+ switch (values[i]) {
+ case "block":
+ case "classic":
+ case "oval":
+ case "diamond":
+ case "open":
+ case "none":
+ type = values[i];
+ break;
+ case "wide":
+ h = 5;
+ break;
+ case "narrow":
+ h = 2;
+ break;
+ case "long":
+ w = 5;
+ break;
+ case "short":
+ w = 2;
+ break;
+ }
+ }
+ if (type == "open") {
+ w += 2;
+ h += 2;
+ t += 2;
+ dx = 1;
+ refX = isEnd ? 4 : 1;
+ attr = {
+ fill: "none",
+ stroke: attrs.stroke
+ };
+ } else {
+ refX = dx = w / 2;
+ attr = {
+ fill: attrs.stroke,
+ stroke: "none"
+ };
+ }
+ if (o._.arrows) {
+ if (isEnd) {
+ o._.arrows.endPath && markerCounter[o._.arrows.endPath]--;
+ o._.arrows.endMarker && markerCounter[o._.arrows.endMarker]--;
+ } else {
+ o._.arrows.startPath && markerCounter[o._.arrows.startPath]--;
+ o._.arrows.startMarker && markerCounter[o._.arrows.startMarker]--;
+ }
+ } else {
+ o._.arrows = {};
+ }
+ if (type != "none") {
+ var pathId = "raphael-marker-" + type,
+ markerId = "raphael-marker-" + se + type + w + h + "-obj" + o.id;
+ if (!R._g.doc.getElementById(pathId)) {
+ p.defs.appendChild($($("path"), {
+ "stroke-linecap": "round",
+ d: markers[type],
+ id: pathId
+ }));
+ markerCounter[pathId] = 1;
+ } else {
+ markerCounter[pathId]++;
+ }
+ var marker = R._g.doc.getElementById(markerId),
+ use;
+ if (!marker) {
+ marker = $($("marker"), {
+ id: markerId,
+ markerHeight: h,
+ markerWidth: w,
+ orient: "auto",
+ refX: refX,
+ refY: h / 2
+ });
+ use = $($("use"), {
+ "xlink:href": "#" + pathId,
+ transform: (isEnd ? "rotate(180 " + w / 2 + " " + h / 2 + ") " : E) + "scale(" + w / t + "," + h / t + ")",
+ "stroke-width": (1 / ((w / t + h / t) / 2)).toFixed(4)
+ });
+ marker.appendChild(use);
+ p.defs.appendChild(marker);
+ markerCounter[markerId] = 1;
+ } else {
+ markerCounter[markerId]++;
+ use = marker.getElementsByTagName("use")[0];
+ }
+ $(use, attr);
+ var delta = dx * (type != "diamond" && type != "oval");
+ if (isEnd) {
+ from = o._.arrows.startdx * stroke || 0;
+ to = R.getTotalLength(attrs.path) - delta * stroke;
+ } else {
+ from = delta * stroke;
+ to = R.getTotalLength(attrs.path) - (o._.arrows.enddx * stroke || 0);
+ }
+ attr = {};
+ attr["marker-" + se] = "url('" + R._url + "#" + markerId + "')";
+ if (to || from) {
+ attr.d = Raphael.getSubpath(attrs.path, from, to);
+ }
+ $(node, attr);
+ o._.arrows[se + "Path"] = pathId;
+ o._.arrows[se + "Marker"] = markerId;
+ o._.arrows[se + "dx"] = delta;
+ o._.arrows[se + "Type"] = type;
+ o._.arrows[se + "String"] = value;
+ } else {
+ if (isEnd) {
+ from = o._.arrows.startdx * stroke || 0;
+ to = R.getTotalLength(attrs.path) - from;
+ } else {
+ from = 0;
+ to = R.getTotalLength(attrs.path) - (o._.arrows.enddx * stroke || 0);
+ }
+ o._.arrows[se + "Path"] && $(node, {
+ d: Raphael.getSubpath(attrs.path, from, to)
+ });
+ delete o._.arrows[se + "Path"];
+ delete o._.arrows[se + "Marker"];
+ delete o._.arrows[se + "dx"];
+ delete o._.arrows[se + "Type"];
+ delete o._.arrows[se + "String"];
+ }
+ for (attr in markerCounter)
+ if (markerCounter[has](attr) && !markerCounter[attr]) {
+ var item = R._g.doc.getElementById(attr);
+ item && item.parentNode.removeChild(item);
+ }
+ }
+ },
+ dasharray = {
+ "": [0],
+ "none": [0],
+ "-": [3, 1],
+ ".": [1, 1],
+ "-.": [3, 1, 1, 1],
+ "-..": [3, 1, 1, 1, 1, 1],
+ ". ": [1, 3],
+ "- ": [4, 3],
+ "--": [8, 3],
+ "- .": [4, 3, 1, 3],
+ "--.": [8, 3, 1, 3],
+ "--..": [8, 3, 1, 3, 1, 3]
+ },
+ addDashes = function(o, value, params) {
+ var predefValue = dasharray[Str(value).toLowerCase()];
+ value = predefValue || ((value !== undefined) && [].concat(value));
+ if (value) {
+ var width = o.attrs["stroke-width"] || "1",
+ butt = {
+ round: width,
+ square: width,
+ butt: 0
+ }[o.attrs["stroke-linecap"] || params["stroke-linecap"]] || 0,
+ i,
+ l = i = value.length;
+ if (predefValue) {
+ while (i--) {
+ value[i] = value[i] * width + ((i % 2) ? 1 : -1) * butt;
+ }
+ }
+ else {
+ for (i = 0; i < l; i += 2) {
+ value[i] -= butt;
+ value[i + 1] && (value[i + 1] += butt);
+ if (value[i] <= 0) {
+ value[i] = 0.1;
+ }
+ }
+ }
+ if (R.is(value, 'array')) {
+ $(o.node, {
+ "stroke-dasharray": value.join(",")
+ });
+ }
+ }
+ },
+ setFillAndStroke = R._setFillAndStroke = function(o, params) {
+ if (!o.paper.canvas) {
+ return;
+ }
+ var node = o.node,
+ attrs = o.attrs,
+ paper = o.paper,
+ s = node.style,
+ vis = s.visibility;
+
+ s.visibility = "hidden";
+ for (var att in params) {
+ if (params[has](att)) {
+ if (!R._availableAttrs[has](att)) {
+ continue;
+ }
+ var value = params[att];
+ attrs[att] = value;
+ switch (att) {
+ case "blur":
+ o.blur(value);
+ break;
+ case "href":
+ case "title":
+ case "target":
+ var pn = node.parentNode;
+ if (pn.tagName.toLowerCase() != "a") {
+ if (value == E) { break; }
+ var hl = $("a");
+ pn.insertBefore(hl, node);
+ hl.appendChild(node);
+ pn = hl;
+ }
+ if (att == "target") {
+ pn.setAttributeNS(xlink, "show", value == "blank" ? "new" : value);
+ } else {
+ pn.setAttributeNS(xlink, att, value);
+ }
+ node.titleNode = pn;
+ break;
+ case "cursor":
+ s.cursor = value;
+ break;
+ case "transform":
+ o.transform(value);
+ break;
+ case "rotation":
+ if (R.is(value, "array")) {
+ o.rotate.apply(o, value);
+ }
+ else {
+ o.rotate(value);
+ }
+ break;
+ case "arrow-start":
+ addArrow(o, value);
+ break;
+ case "arrow-end":
+ addArrow(o, value, 1);
+ break;
+ case "clip-path":
+ var pathClip = true;
+ case "clip-rect":
+ var rect = !pathClip && Str(value).split(separator);
+ o._.clipispath = !!pathClip;
+ if (pathClip || rect.length == 4) {
+ o.clip && o.clip.parentNode.parentNode.removeChild(o.clip.parentNode);
+ var el = $("clipPath"),
+ rc = $(pathClip ? "path" : "rect");
+ el.id = R.createUUID();
+ $(rc, pathClip ? {
+ d: value ? attrs['clip-path'] = R._pathToAbsolute(value) : R._availableAttrs.path,
+ fill: 'none'
+ } : {
+ x: rect[0],
+ y: rect[1],
+ width: rect[2],
+ height: rect[3],
+ transform: o.matrix.invert()
+ });
+ el.appendChild(rc);
+ paper.defs.appendChild(el);
+ $(node, {
+ "clip-path": "url('" + R._url +"#" + el.id + "')"
+ });
+ o.clip = rc;
+ }
+ if (!value) {
+ var path = node.getAttribute("clip-path");
+ if (path) {
+ var clip = R._g.doc.getElementById(path.replace(/(^url\(#|\)$)/g, E));
+ clip && clip.parentNode.removeChild(clip);
+ $(node, {
+ "clip-path": E
+ });
+ delete o.clip;
+ }
+ }
+ break;
+ case "path":
+ if (o.type == "path") {
+ $(node, {
+ d: value ? attrs.path = R._pathToAbsolute(value) : R._availableAttrs.path
+ });
+ o._.dirty = 1;
+ if (o._.arrows) {
+ "startString" in o._.arrows && addArrow(o, o._.arrows.startString);
+ "endString" in o._.arrows && addArrow(o, o._.arrows.endString, 1);
+ }
+ }
+ break;
+ case "width":
+ node.setAttribute(att, value);
+ o._.dirty = 1;
+ if (attrs.fx) {
+ att = "x";
+ value = attrs.x;
+ } else {
+ break;
+ }
+ case "x":
+ if (attrs.fx) {
+ value = -attrs.x - (attrs.width || 0);
+ }
+ case "rx":
+ if (att == "rx" && o.type == "rect") {
+ break;
+ }
+ case "cx":
+ node.setAttribute(att, value);
+ o.pattern && updatePosition(o);
+ o._.dirty = 1;
+ break;
+ case "height":
+ node.setAttribute(att, value);
+ o._.dirty = 1;
+ if (attrs.fy) {
+ att = "y";
+ value = attrs.y;
+ } else {
+ break;
+ }
+ case "y":
+ if (attrs.fy) {
+ value = -attrs.y - (attrs.height || 0);
+ }
+ case "ry":
+ if (att == "ry" && o.type == "rect") {
+ break;
+ }
+ case "cy":
+ node.setAttribute(att, value);
+ o.pattern && updatePosition(o);
+ o._.dirty = 1;
+ break;
+ case "r":
+ if (o.type == "rect") {
+ $(node, {
+ rx: value,
+ ry: value
+ });
+ } else {
+ node.setAttribute(att, value);
+ }
+ o._.dirty = 1;
+ break;
+ case "src":
+ if (o.type == "image") {
+ node.setAttributeNS(xlink, "href", value);
+ }
+ break;
+ case "stroke-width":
+ if (o._.sx != 1 || o._.sy != 1) {
+ value /= mmax(abs(o._.sx), abs(o._.sy)) || 1;
+ }
+ if (paper._vbSize) {
+ value *= paper._vbSize;
+ }
+ if (zeroStrokeFix && value === 0) {
+ value = 0.000001;
+ }
+ node.setAttribute(att, value);
+ if (attrs["stroke-dasharray"]) {
+ addDashes(o, attrs["stroke-dasharray"], params);
+ }
+ if (o._.arrows) {
+ "startString" in o._.arrows && addArrow(o, o._.arrows.startString);
+ "endString" in o._.arrows && addArrow(o, o._.arrows.endString, 1);
+ }
+ break;
+ case "stroke-dasharray":
+ addDashes(o, value, params);
+ break;
+ case "fill":
+ var isURL = Str(value).match(R._ISURL);
+ if (isURL) {
+ el = $("pattern");
+ var ig = $("image");
+ el.id = R.createUUID();
+ $(el, {
+ x: 0,
+ y: 0,
+ patternUnits: "userSpaceOnUse",
+ height: 1,
+ width: 1
+ });
+ $(ig, {
+ x: 0,
+ y: 0,
+ "xlink:href": isURL[1]
+ });
+ el.appendChild(ig);
+
+ (function(el) {
+ R._preload(isURL[1], function() {
+ var w = this.offsetWidth,
+ h = this.offsetHeight;
+ $(el, {
+ width: w,
+ height: h
+ });
+ $(ig, {
+ width: w,
+ height: h
+ });
+ paper.safari();
+ });
+ })(el);
+ paper.defs.appendChild(el);
+ $(node, {
+ fill: "url('" + R._url + "#" + el.id + "')"
+ });
+ o.pattern = el;
+ o.pattern && updatePosition(o);
+ break;
+ }
+ var clr = R.getRGB(value);
+ if (!clr.error) {
+ delete params.gradient;
+ delete attrs.gradient;
+ !R.is(attrs.opacity, "undefined") &&
+ R.is(params.opacity, "undefined") &&
+ $(node, {
+ opacity: attrs.opacity
+ });
+ !R.is(attrs["fill-opacity"], "undefined") &&
+ R.is(params["fill-opacity"], "undefined") &&
+ $(node, {
+ "fill-opacity": attrs["fill-opacity"]
+ });
+ } else if ((o.type == "circle" || o.type == "ellipse" || Str(value).charAt() != "r") && addGradientFill(o, value)) {
+ if ("opacity" in attrs || "fill-opacity" in attrs) {
+ var gradient = R._g.doc.getElementById(node.getAttribute("fill").replace(/^url\(#|\)$/g, E));
+ if (gradient) {
+ var stops = gradient.getElementsByTagName("stop");
+ $(stops[stops.length - 1], {
+ "stop-opacity": ("opacity" in attrs ? attrs.opacity : 1) * ("fill-opacity" in attrs ? attrs["fill-opacity"] : 1)
+ });
+ }
+ }
+ attrs.gradient = value;
+ attrs.fill = "none";
+ break;
+ }
+ if (clr[has]("opacity")) {
+ $(node, {
+ "fill-opacity": (s.fillOpacity =
+ (clr.opacity > 1 ? clr.opacity / 100 : clr.opacity))
+ });
+ o._.opacitydirty = true;
+ }
+ else if (o._.opacitydirty && R.is(attrs['fill-opacity'], "undefined") &&
+ R.is(params["fill-opacity"], "undefined")) {
+ node.removeAttribute('fill-opacity');
+ s.fillOpacity = E;
+ delete o._.opacitydirty;
+ }
+ case "stroke":
+ clr = R.getRGB(value);
+ node.setAttribute(att, clr.hex);
+ att == "stroke" && clr[has]("opacity") && $(node, {
+ "stroke-opacity": clr.opacity > 1 ? clr.opacity / 100 : clr.opacity
+ });
+ if (att == "stroke" && o._.arrows) {
+ "startString" in o._.arrows && addArrow(o, o._.arrows.startString);
+ "endString" in o._.arrows && addArrow(o, o._.arrows.endString, 1);
+ }
+ break;
+ case "gradient":
+ (o.type == "circle" || o.type == "ellipse" || Str(value).charAt() != "r") && addGradientFill(o, value);
+ break;
+ case 'line-height': // do not apply
+ case 'vertical-align': // do not apply
+ break;
+ case "visibility":
+ value === 'hidden' ? o.hide() : o.show();
+ break;
+ case "opacity":
+ if (attrs.gradient && !attrs[has]("stroke-opacity")) {
+ $(node, {
+ "stroke-opacity": value > 1 ? value / 100 : value
+ });
+ }
+ // fall
+ case "fill-opacity":
+ if (attrs.gradient) {
+ gradient = R._g.doc.getElementById(node.getAttribute("fill").replace(/^url\(#|\)$/g, E));
+ if (gradient) {
+ stops = gradient.getElementsByTagName("stop");
+ $(stops[stops.length - 1], {
+ "stop-opacity": value
+ });
+ }
+ break;
+ }
+ default:
+ att == "font-size" && (value = toInt(value, 10) + "px");
+ var cssrule = att.replace(/(\-.)/g, function(w) {
+ return w.substring(1).toUpperCase();
+ });
+ s[cssrule] = value;
+ o._.dirty = 1;
+ node.setAttribute(att, value);
+ break;
+ }
+ }
+ }
+
+ tuneText(o, params);
+ s.visibility = vis;
+ },
+ leading = 1.2,
+ tuneText = function(el, params) {
+ if (el.type != "text" || !(params[has]("text") || params[has]("font") ||
+ params[has]("font-size") || params[has]("x") || params[has]("y") ||
+ params[has]("line-height") || params[has]("vertical-align"))) {
+ return;
+ }
+ var a = el.attrs,
+ node = el.node,
+ computedStyle = node.firstChild && R._g.doc.defaultView.getComputedStyle(node.firstChild, E),
+ fontSize = computedStyle ? toFloat(R._g.doc.defaultView.getComputedStyle(node.firstChild, E).getPropertyValue("font-size")) : 10,
+ lineHeight = toFloat(params['line-height'] || a['line-height']) || fontSize * leading,
+ valign = a[has]("vertical-align") ? a["vertical-align"] : "middle";
+
+ if (isNaN(lineHeight)) {
+ lineHeight = fontSize * leading;
+ }
+
+ valign = valign === 'top' ? -0.5 : (valign === 'bottom' ? 0.5 : 0);
+
+ if (params[has]("text") && (params.text !== a.text || el._textdirty)) {
+ a.text = params.text;
+ while (node.firstChild) {
+ node.removeChild(node.firstChild);
+ }
+ var texts = Str(params.text).split(/\n| /ig),
+ tspans = [],
+ tspan;
+ for (var i = 0, ii = texts.length; i < ii; i++) {
+ tspan = $("tspan");
+ if (i) {
+ $(tspan, {
+ dy: lineHeight,
+ x: a.x
+ });
+ } else {
+ $(tspan, {
+ dy: lineHeight * texts.length * valign,
+ x: a.x
+ });
+ }
+ if (!texts[i]) { // preserve blank lines
+ tspan.setAttributeNS("http://www.w3.org/XML/1998/namespace",
+ "xml:space","preserve");
+ texts[i] = " ";
+ }
+ tspan.appendChild(R._g.doc.createTextNode(texts[i]));
+ node.appendChild(tspan);
+ tspans[i] = tspan;
+ }
+ el._textdirty = false;
+ } else {
+ tspans = node.getElementsByTagName("tspan");
+ for (i = 0, ii = tspans.length; i < ii; i++)
+ if (i) {
+ $(tspans[i], {
+ dy: lineHeight,
+ x: a.x
+ });
+ } else {
+ $(tspans[0], {
+ dy: lineHeight * tspans.length * valign,
+ x: a.x
+ });
+ }
+ }
+ $(node, {
+ x: a.x,
+ y: a.y
+ });
+ el._.dirty = 1;
+ var bb = el._getBBox(),
+ dif = a.y - (bb.y + bb.height / 2);
+
+ // If the bbox is calculated then we need to make additional adjustments,
+ // to account for the fact that the calculated bbox already has the
+ // text alignment, both horizontal and vertical, included in the calculation.
+ if (bb.isCalculated) {
+ switch (a['vertical-align']) {
+ case "top":
+ dif = bb.height * .75;
+ break;
+ case "bottom":
+ dif = - (bb.height * .25);
+ break;
+ default:
+ dif = a.y - (bb.y + bb.height * .25);
+ break;
+ };
+ }
+
+ dif && R.is(dif, "finite") && tspans[0] && $(tspans[0], {
+ dy: dif
+ });
+ },
+ Element = function(node, svg, group) {
+ var o = this,
+ parent = group || svg;
+
+ o.node = o[0] = node;
+ node.raphael = true;
+ node.raphaelid = o.id = R._oid++;
+
+ o.matrix = R.matrix();
+ o.realPath = null;
+
+ o.attrs = o.attrs || {};
+ o.styles = o.styles || {};
+ o.followers = o.followers || [];
+
+ o.paper = svg;
+ o.ca = o.customAttributes = o.customAttributes ||
+ new svg._CustomAttributes();
+
+ o._ = {
+ transform: [],
+ sx: 1,
+ sy: 1,
+ deg: 0,
+ dx: 0,
+ dy: 0,
+ dirty: 1
+ };
+
+ o.parent = parent;
+ !parent.bottom && (parent.bottom = o);
+
+ o.prev = parent.top;
+ parent.top && (parent.top.next = o);
+ parent.top = o;
+ o.next = null;
+ },
+ elproto = R.el;
+
+ Element.prototype = elproto;
+ elproto.constructor = Element;
+
+ R._engine.getNode = function (el) {
+ var node = el.node || el[0].node;
+ return node.titleNode || node;
+ };
+ R._engine.getLastNode = function (el) {
+ var node = el.node || el[el.length - 1].node;
+ return node.titleNode || node;
+ };
+
+ R._engine.path = function(pathString, SVG, group) {
+ var el = $("path");
+
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (SVG.canvas && SVG.canvas.appendChild(el));
+
+ var p = new Element(el, SVG, group);
+ p.type = "path";
+ setFillAndStroke(p, {
+ fill: "none",
+ stroke: "#000",
+ path: pathString
+ });
+ return p;
+ };
+
+ elproto.rotate = function(deg, cx, cy) {
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ deg = Str(deg).split(separator);
+ if (deg.length - 1) {
+ cx = toFloat(deg[1]);
+ cy = toFloat(deg[2]);
+ }
+ deg = toFloat(deg[0]);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ cx = bbox.x + bbox.width / 2;
+ cy = bbox.y + bbox.height / 2;
+ }
+ o.transform(o._.transform.concat([["r", deg, cx, cy]]));
+ return o;
+ };
+
+ elproto.scale = function(sx, sy, cx, cy) {
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ sx = Str(sx).split(separator);
+ if (sx.length - 1) {
+ sy = toFloat(sx[1]);
+ cx = toFloat(sx[2]);
+ cy = toFloat(sx[3]);
+ }
+ sx = toFloat(sx[0]);
+ (sy == null) && (sy = sx);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ }
+ cx = cx == null ? bbox.x + bbox.width / 2 : cx;
+ cy = cy == null ? bbox.y + bbox.height / 2 : cy;
+ o.transform(o._.transform.concat([["s", sx, sy, cx, cy]]));
+ return o;
+ };
+
+ elproto.translate = function(dx, dy) {
+ var o = this;
+ if (o.removed) {
+ return o;
+ }
+ dx = Str(dx).split(separator);
+ if (dx.length - 1) {
+ dy = toFloat(dx[1]);
+ }
+ dx = toFloat(dx[0]) || 0;
+ dy = +dy || 0;
+ o.transform(o._.transform.concat([["t", dx, dy]]));
+ return o;
+ };
+
+ elproto.transform = function(tstr) {
+ var o = this,
+ _ = o._,
+ sw;
+
+ if (tstr == null) {
+ return _.transform;
+ }
+ R._extractTransform(o, tstr);
+
+ o.clip && !_.clipispath && $(o.clip, {
+ transform: o.matrix.invert()
+ });
+ o.pattern && updatePosition(o);
+ o.node && $(o.node, {
+ transform: o.matrix
+ });
+
+ if (_.sx != 1 || _.sy != 1) {
+ sw = o.attrs[has]("stroke-width") ? o.attrs["stroke-width"] : 1;
+ o.attr({
+ "stroke-width": sw
+ });
+ }
+
+ return o;
+ };
+
+ elproto.hide = function() {
+ var o = this;
+ !o.removed && o.paper.safari(o.node.style.display = "none");
+ return o;
+ };
+
+ elproto.show = function() {
+ var o = this;
+ !o.removed && o.paper.safari(o.node.style.display = E);
+ return o;
+ };
+
+ elproto.remove = function() {
+ if (this.removed || !this.parent.canvas) {
+ return;
+ }
+
+ var o = this,
+ node = R._engine.getNode(o),
+ paper = o.paper,
+ defs = paper.defs,
+ i;
+
+
+ paper.__set__ && paper.__set__.exclude(o);
+ eve.unbind("raphael.*.*." + o.id);
+
+ if (o.gradient && defs) {
+ defs.removeChild(o.gradient);
+ }
+ while (i = o.followers.pop()) {
+ i.el.remove();
+ }
+ o.parent.canvas.removeChild(node);
+ R._tear(o, paper);
+ for (i in o) {
+ o[i] = typeof o[i] === "function" ? R._removedFactory(i) : null;
+ }
+ o.removed = true;
+ };
+ elproto._getBBox = function() {
+ var o = this,
+ node = o.node,
+ bbox = {},
+ a = o.attrs,
+ align,
+ hide;
+
+ if (node.style.display === "none") {
+ o.show();
+ hide = true;
+ }
+
+ try {
+ bbox = node.getBBox();
+
+ if (o.type == "text") {
+ // If bbox does not have x / y, which is possible in certain
+ // environments, we mathematically calculate these values by
+ // using x, y (adjusted using the values of text-anchor, and
+ // vertical-align attributes), of the element along with the
+ // width and height provided by the getBBox().
+ if (bbox.x === undefined) {
+ bbox.isCalculated = true;
+ align = a['text-anchor'];
+ bbox.x = (a.x || 0) - (bbox.width * ((align === "start") ?
+ 0 : (align === "middle") ? 0.5 : 1));
+ }
+
+ if (bbox.y === undefined) {
+ bbox.isCalculated = true;
+ align = a['vertical-align'];
+ bbox.y = (a.y || 0) - (bbox.height * ((align === "bottom") ?
+ 1 : (align === "middle") ? 0.5 : 0));
+ }
+ }
+
+ } catch (e) {
+ // Firefox 3.0.x plays badly here
+ } finally {
+ bbox = bbox || {};
+ }
+ hide && o.hide();
+ return bbox;
+ };
+
+ elproto.css = function (name, value) {
+ // do not parse css in case element is removed.
+ if (this.removed) {
+ return this;
+ }
+
+ // process as getter when a single key is sent as parameter.
+ if (value == null && R.is(name, "string")) {
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.styles) {
+ out[name] = this.styles[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ // process as getter when multiple keys are pre-sent as array.
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.styles(name[i]);
+ }
+ return out;
+ }
+ // convert single key-value setter into object style standard.
+ if (value != null) {
+ var params = {};
+ params[name] = value;
+ } else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+ // iterate on keys and set style or raise events.
+ var otherkey, doattrs = {};
+ for (var key in params) {
+ otherkey = key.replace(/\B([A-Z]{1})/g, "-$1").toLowerCase();
+
+ // If keys are supported via attr then use attr instead of css.
+ if (R._availableAttrs[has](otherkey) || otherkey === 'color') {
+ // Replace "color" with fill
+ (otherkey === 'color' && this.type === 'text') && (otherkey = 'fill');
+
+ doattrs[otherkey] = params[key];
+ doattrs.dirty = true;
+ continue;
+ }
+ eve("raphael.css." + otherkey + "." + this.id, this, params[key], otherkey);
+ this.node.style[otherkey] = params[key];
+ this.styles[otherkey] = params[key];
+ }
+ // run on followers
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ this.followers[i].el.css(params);
+ }
+ // apply css via attrs
+ if (doattrs[has]("dirty")) {
+ delete doattrs.dirty;
+ this.attr(doattrs);
+ }
+ return this;
+ };
+
+ elproto.attr = function(name, value) {
+ if (this.removed) {
+ return this;
+ }
+ if (name == null) {
+ var res = {};
+ for (var a in this.attrs)
+ if (this.attrs[has](a)) {
+ res[a] = this.attrs[a];
+ }
+ res.gradient && res.fill == "none" && (res.fill = res.gradient) && delete res.gradient;
+ res.transform = this._.transform;
+ res.visibility = this.node.style.display === "none" ? "hidden" : "visible";
+ return res;
+ }
+ if (value == null && R.is(name, "string")) {
+ if (name == "fill" && this.attrs.fill == "none" && this.attrs.gradient) {
+ return this.attrs.gradient;
+ }
+ if (name == "transform") {
+ return this._.transform;
+ }
+ if (name == "visibility") {
+ return this.node.style.display === "none" ? "hidden" : "visible";
+ }
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.attrs) {
+ out[name] = this.attrs[name];
+ } else if (R.is(this.ca[name], "function")) {
+ out[name] = this.ca[name].def;
+ } else {
+ out[name] = R._availableAttrs[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.attr(name[i]);
+ }
+ return out;
+ }
+ if (value != null) {
+ var params = {};
+ params[name] = value;
+ } else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+ for (var key in params) {
+ eve("raphael.attr." + key + "." + this.id, this, params[key], key);
+ }
+ var todel = {};
+ for (key in this.ca) {
+ if (this.ca[key] && params[has](key) && R.is(this.ca[key], "function") && !this.ca['_invoked' + key]) {
+
+ this.ca['_invoked'+key] = true; // prevent recursion
+ var par = this.ca[key].apply(this, [].concat(params[key]));
+ delete this.ca['_invoked'+key];
+
+ for (var subkey in par) {
+ if (par[has](subkey)) {
+ params[subkey] = par[subkey];
+ }
+ }
+ this.attrs[key] = params[key];
+ if (par === false) {
+ todel[key] = params[key];
+ delete params[key];
+ }
+ }
+ }
+
+ setFillAndStroke(this, params);
+
+ var follower;
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ follower = this.followers[i];
+ (follower.cb && !follower.cb.call(follower.el, params, this)) ||
+ follower.el.attr(params);
+ }
+
+ for (subkey in todel) {
+ params[subkey] = todel[subkey];
+ }
+ return this;
+ };
+
+ elproto.blur = function(size) {
+ // Experimental. No Safari support. Use it on your own risk.
+ var t = this;
+ if (+size !== 0) {
+ var fltr = $("filter"),
+ blur = $("feGaussianBlur");
+ t.attrs.blur = size;
+ fltr.id = R.createUUID();
+ $(blur, {
+ stdDeviation: +size || 1.5
+ });
+ fltr.appendChild(blur);
+ t.paper.defs.appendChild(fltr);
+ t._blur = fltr;
+ $(t.node, {
+ filter: "url('" + R._url + "#" + fltr.id + "')"
+ });
+ } else {
+ if (t._blur) {
+ t._blur.parentNode.removeChild(t._blur);
+ delete t._blur;
+ delete t.attrs.blur;
+ }
+ t.node.removeAttribute("filter");
+ }
+ };
+
+ elproto.on = function(eventType, handler) {
+ if (this.removed) {
+ return this;
+ }
+
+ var fn = handler;
+ if (R.supportsTouch) {
+ eventType = R._touchMap[eventType] ||
+ (eventType === 'click' && 'touchstart') || eventType;
+ fn = function(e) {
+ e.preventDefault();
+ handler();
+ };
+ }
+ this.node['on'+ eventType] = fn;
+ return this;
+ };
+
+
+ R._engine.group = function(svg, id, group) {
+ var el = $("g");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var g = new Element(el, svg, group);
+ g.type = "group";
+ g.canvas = g.node;
+ g.top = null;
+ g.bottom = null;
+ id && el.setAttribute('class', ['red', id, g.id].join('-'));
+
+ return g;
+ };
+
+ R._engine.circle = function(svg, x, y, r, group) {
+ var el = $("circle");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ cx: x,
+ cy: y,
+ r: r,
+ fill: "none",
+ stroke: "#000"
+ };
+ res.type = "circle";
+ $(el, res.attrs);
+ return res;
+ };
+ R._engine.rect = function(svg, x, y, w, h, r, group) {
+ var el = $("rect");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ x: x,
+ y: y,
+ width: w,
+ height: h,
+ r: r || 0,
+ rx: r || 0,
+ ry: r || 0,
+ fill: "none",
+ stroke: "#000"
+ };
+ res.type = "rect";
+ $(el, res.attrs);
+ return res;
+ };
+ R._engine.ellipse = function(svg, x, y, rx, ry, group) {
+ var el = $("ellipse");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ cx: x,
+ cy: y,
+ rx: rx,
+ ry: ry,
+ fill: "none",
+ stroke: "#000"
+ };
+ res.type = "ellipse";
+ $(el, res.attrs);
+ return res;
+ };
+ R._engine.image = function(svg, src, x, y, w, h, group) {
+ var el = $("image");
+ $(el, {
+ x: x,
+ y: y,
+ width: w,
+ height: h,
+ preserveAspectRatio: "none"
+ });
+ el.setAttributeNS(xlink, "href", src);
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ x: x,
+ y: y,
+ width: w,
+ height: h,
+ src: src
+ };
+ res.type = "image";
+ return res;
+ };
+ R._engine.text = function(svg, x, y, text, group) {
+ var el = $("text");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ x: x,
+ y: y,
+ "text-anchor": "middle",
+ "vertical-align": "middle",
+ text: text,
+ //font: R._availableAttrs.font,
+ stroke: "none",
+ fill: "#000"
+ };
+ res.type = "text";
+ res._textdirty = true;
+ setFillAndStroke(res, res.attrs);
+ return res;
+ };
+ /* @diffend */
+
+ R._engine.setSize = function(width, height) {
+ this.width = width || this.width;
+ this.height = height || this.height;
+ this.canvas.setAttribute("width", this.width);
+ this.canvas.setAttribute("height", this.height);
+ if (this._viewBox) {
+ this.setViewBox.apply(this, this._viewBox);
+ }
+ return this;
+ };
+ R._engine.create = function() {
+ var con = R._getContainer.apply(0, arguments),
+ container = con && con.container,
+ x = con.x,
+ y = con.y,
+ width = con.width,
+ height = con.height;
+ if (!container) {
+ throw new Error("SVG container not found.");
+ }
+ var cnvs = $("svg"),
+ css = "overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);"+
+ "-webkit-user-select:none;-moz-user-select:-moz-none;-khtml-user-select:none;"+
+ "-ms-user-select:none;user-select:none;-o-user-select:none;cursor:default;",
+ isFloating;
+ x = x || 0;
+ y = y || 0;
+ width = width || 512;
+ height = height || 342;
+ $(cnvs, {
+ height: height,
+ version: 1.1,
+ width: width,
+ xmlns: "http://www.w3.org/2000/svg"
+ });
+ if (container == 1) {
+ cnvs.style.cssText = css + "position:absolute;left:" + x + "px;top:" + y + "px";
+ R._g.doc.body.appendChild(cnvs);
+ isFloating = 1;
+ } else {
+ cnvs.style.cssText = css + "position:relative";
+ if (container.firstChild) {
+ container.insertBefore(cnvs, container.firstChild);
+ } else {
+ container.appendChild(cnvs);
+ }
+ }
+ container = new R._Paper;
+ container.width = width;
+ container.height = height;
+ container.canvas = cnvs;
+ container.clear();
+ container._left = container._top = 0;
+ isFloating && (container.renderfix = function() {
+ });
+ container.renderfix();
+ return container;
+ };
+ R._engine.setViewBox = function(x, y, w, h, fit) {
+ eve("raphael.setViewBox", this, this._viewBox, [x, y, w, h, fit]);
+ var size = mmax(w / this.width, h / this.height),
+ top = this.top,
+ aspectRatio = fit ? "meet" : "xMinYMin",
+ vb,
+ sw;
+ if (x == null) {
+ if (this._vbSize) {
+ size = 1;
+ }
+ delete this._vbSize;
+ vb = "0 0 " + this.width + S + this.height;
+ } else {
+ this._vbSize = size;
+ vb = x + S + y + S + w + S + h;
+ }
+ $(this.canvas, {
+ viewBox: vb,
+ preserveAspectRatio: aspectRatio
+ });
+ while (size && top) {
+ sw = "stroke-width" in top.attrs ? top.attrs["stroke-width"] : 1;
+ top.attr({
+ "stroke-width": sw
+ });
+ top._.dirty = 1;
+ top._.dirtyT = 1;
+ top = top.prev;
+ }
+ this._viewBox = [x, y, w, h, !!fit];
+ return this;
+ };
+
+ R.prototype.renderfix = function() {
+ var cnvs = this.canvas,
+ s = cnvs.style,
+ pos;
+ try {
+ pos = cnvs.getScreenCTM() || cnvs.createSVGMatrix();
+ } catch (e) {
+ pos = cnvs.createSVGMatrix();
+ }
+ var left = -pos.e % 1,
+ top = - pos.f % 1;
+ if (left || top) {
+ if (left) {
+ this._left = (this._left + left) % 1;
+ s.left = this._left + "px";
+ }
+ if (top) {
+ this._top = (this._top + top) % 1;
+ s.top = this._top + "px";
+ }
+ }
+ };
+
+ R.prototype.clear = function() {
+ eve("raphael.clear", this);
+ var c = this.canvas;
+ while (c.firstChild) {
+ c.removeChild(c.firstChild);
+ }
+ this.bottom = this.top = null;
+ (this.desc = $("desc")).appendChild(R._g.doc.createTextNode(R.is(R.desc, "string") && R.desc ||
+ "Created with Red Rapha\xebl " + R.version));
+ c.appendChild(this.desc);
+ c.appendChild(this.defs = $("defs"));
+ };
+
+ R.prototype.remove = function() {
+ eve("raphael.remove", this);
+ this.canvas.parentNode && this.canvas.parentNode.removeChild(this.canvas);
+ for (var i in this) {
+ this[i] = typeof this[i] == "function" ? R._removedFactory(i) : null;
+ }
+ this.removed = true;
+ };
+ var setproto = R.st;
+ for (var method in elproto)
+ if (elproto[has](method) && !setproto[has](method)) {
+ setproto[method] = (function(methodname) {
+ return function() {
+ var arg = arguments;
+ return this.forEach(function(el) {
+ el[methodname].apply(el, arg);
+ });
+ };
+ })(method);
+ }
+})();
+
+/**!
+* RedRaphael 1.0.0 - JavaScript Vector Library VML Module
+* Copyright (c) 2012-2013 FusionCharts Technologies
+*
+* Raphael 2.1.0 - JavaScript Vector Library VML Module
+* Copyright (c) 2008-2012 Dmitry Baranovskiy
+* Copyright © 2008-2012 Sencha Labs
+*
+* Licensed under the MIT license.
+*/
+
+(function(){
+ if (!R.vml) {
+ return;
+ }
+ var has = "hasOwnProperty",
+ Str = String,
+ toFloat = parseFloat,
+ math = Math,
+ round = math.round,
+ mmax = math.max,
+ mmin = math.min,
+ sqrt = math.sqrt,
+ abs = math.abs,
+ fillString = "fill",
+ separator = /[, ]+/,
+ eve = R.eve,
+ ms = " progid:DXImageTransform.Microsoft",
+ S = " ",
+ E = "",
+ map = {
+ M: "m",
+ L: "l",
+ C: "c",
+ Z: "x",
+ m: "t",
+ l: "r",
+ c: "v",
+ z: "x"
+ },
+ bites = /([clmz]),?([^clmz]*)/gi,
+ blurregexp = / progid:\S+Blur\([^\)]+\)/g,
+ val = /-?[^,\s-]+/g,
+ cssDot = "position:absolute;left:0;top:0;width:1px;height:1px",
+ zoom = 21600,
+ pathTypes = {
+ path: 1,
+ rect: 1,
+ image: 1
+ },
+ ovalTypes = {
+ circle: 1,
+ ellipse: 1
+ },
+ path2vml = function(path) {
+ var total = /[ahqstv]/ig,
+ command = R._pathToAbsolute;
+ Str(path).match(total) && (command = R._path2curve);
+ total = /[clmz]/g;
+ if (command == R._pathToAbsolute && !Str(path).match(total)) {
+ var res = Str(path).replace(bites, function(all, command, args) {
+ var vals = [],
+ isMove = command.toLowerCase() == "m",
+ res = map[command];
+ args.replace(val, function(value) {
+ if (isMove && vals.length == 2) {
+ res += vals + map[command == "m" ? "l" : "L"];
+ vals = [];
+ }
+ vals.push(round(value * zoom));
+ });
+ return res + vals;
+ });
+
+ return res || 'm0,0';
+ }
+ var pa = command(path), p, r;
+ res = [];
+ for (var i = 0, ii = pa.length; i < ii; i++) {
+ p = pa[i];
+ r = pa[i][0].toLowerCase();
+ r == "z" && (r = "x");
+ for (var j = 1, jj = p.length; j < jj; j++) {
+ r += round(p[j] * zoom) + (j != jj - 1 ? "," : E);
+ }
+ res.push(r);
+ }
+ return res.length ? res.join(S) : 'm0,0';
+ },
+ compensation = function(deg, dx, dy) {
+ var m = R.matrix();
+ m.rotate(-deg, .5, .5);
+ return {
+ dx: m.x(dx, dy),
+ dy: m.y(dx, dy)
+ };
+ },
+ setCoords = function(p, sx, sy, dx, dy, deg) {
+ var _ = p._,
+ m = p.matrix,
+ fillpos = _.fillpos,
+ o = p.node,
+ s = o.style,
+ y = 1,
+ flip = "",
+ dxdy,
+ kx = zoom / sx,
+ ky = zoom / sy;
+ s.visibility = "hidden";
+ if (!sx || !sy) {
+ return;
+ }
+ o.coordsize = abs(kx) + S + abs(ky);
+ s.rotation = deg * (sx * sy < 0 ? -1 : 1);
+ if (deg) {
+ var c = compensation(deg, dx, dy);
+ dx = c.dx;
+ dy = c.dy;
+ }
+ sx < 0 && (flip += "x");
+ sy < 0 && (flip += " y") && (y = -1);
+ s.flip = flip;
+ o.coordorigin = (dx * -kx) + S + (dy * -ky);
+ if (fillpos || _.fillsize) {
+ var fill = o.getElementsByTagName(fillString);
+ fill = fill && fill[0];
+ o.removeChild(fill);
+ if (fillpos) {
+ c = compensation(deg, m.x(fillpos[0], fillpos[1]), m.y(fillpos[0], fillpos[1]));
+ fill.position = c.dx * y + S + c.dy * y;
+ }
+ if (_.fillsize) {
+ fill.size = _.fillsize[0] * abs(sx) + S + _.fillsize[1] * abs(sy);
+ }
+ o.appendChild(fill);
+ }
+ s.visibility = "visible";
+ };
+ R._url = E;
+ R.toString = function() {
+ return "Your browser doesn\u2019t support SVG. Falling down to VML.\nYou are running Rapha\xebl " + this.version;
+ };
+ var addArrow = function(o, value, isEnd) {
+ var values = Str(value).toLowerCase().split("-"),
+ se = isEnd ? "end" : "start",
+ i = values.length,
+ type = "classic",
+ w = "medium",
+ h = "medium";
+ while (i--) {
+ switch (values[i]) {
+ case "block":
+ case "classic":
+ case "oval":
+ case "diamond":
+ case "open":
+ case "none":
+ type = values[i];
+ break;
+ case "wide":
+ case "narrow":
+ h = values[i];
+ break;
+ case "long":
+ case "short":
+ w = values[i];
+ break;
+ }
+ }
+ var stroke = o.node.getElementsByTagName("stroke")[0];
+ stroke[se + "arrow"] = type;
+ stroke[se + "arrowlength"] = w;
+ stroke[se + "arrowwidth"] = h;
+ },
+ setFillAndStroke = R._setFillAndStroke = function(o, params) {
+ if (!o.paper.canvas) return;
+ // o.paper.canvas.style.display = "none";
+ o.attrs = o.attrs || {};
+ var node = o.node,
+ a = o.attrs,
+ s = node.style,
+ xy,
+ newpath = pathTypes[o.type] && (params.x != a.x || params.y != a.y || params.width != a.width || params.height != a.height || params.cx != a.cx || params.cy != a.cy || params.rx != a.rx || params.ry != a.ry || params.r != a.r),
+ isOval = ovalTypes[o.type] && (a.cx != params.cx || a.cy != params.cy || a.r != params.r || a.rx != params.rx || a.ry != params.ry),
+ isGroup = o.type === 'group',
+ res = o;
+
+
+ for (var par in params)
+ if (params[has](par)) {
+ a[par] = params[par];
+ }
+ if (newpath) {
+ a.path = R._getPath[o.type](o);
+ o._.dirty = 1;
+ }
+ params.href && (node.href = params.href);
+ params.title && (node.title = params.title);
+ params.target && (node.target = params.target);
+ params.cursor && (s.cursor = params.cursor);
+ "blur" in params && o.blur(params.blur);
+ if (params.path && o.type == "path" || newpath) {
+ node.path = path2vml(~Str(a.path).toLowerCase().indexOf("r") ? R._pathToAbsolute(a.path) : a.path);
+ if (o.type == "image") {
+ o._.fillpos = [a.x, a.y];
+ o._.fillsize = [a.width, a.height];
+ setCoords(o, 1, 1, 0, 0, 0);
+ }
+ }
+ "transform" in params && o.transform(params.transform);
+ if ("rotation" in params) {
+ var rotation = params.rotation;
+ if (R.is(rotation, "array")) {
+ o.rotate.apply(o, rotation);
+ }
+ else {
+ o.rotate(rotation);
+ }
+ }
+ if ("visibility" in params) {
+ params.visibility === 'hidden' ? o.hide() : o.show();
+ }
+ if (isOval) {
+ var cx = +a.cx,
+ cy = +a.cy,
+ rx = +a.rx || +a.r || 0,
+ ry = + a.ry || + a.r || 0;
+ node.path = R.format("ar{0},{1},{2},{3},{4},{1},{4},{1}x", round((cx - rx) * zoom), round((cy - ry) * zoom), round((cx + rx) * zoom), round((cy + ry) * zoom), round(cx * zoom));
+ }
+ if ("clip-rect" in params) {
+ var rect = Str(params["clip-rect"]).split(separator);
+
+ if (rect.length == 4) {
+ rect[0] = +rect[0];
+ rect[1] = +rect[1];
+ rect[2] = +rect[2] + rect[0];
+ rect[3] = +rect[3] + rect[1];
+
+ /* @todo create separate element for group clip-rect to
+ * avoid unclipping issue */
+ var div = isGroup ? node : (node.clipRect ||
+ R._g.doc.createElement("div")),
+ offset,
+ dstyle = div.style;
+
+ if (isGroup) {
+ o.clip = rect.slice(); // copy param
+ offset = o.matrix.offset();
+ offset = [toFloat(offset[0]), toFloat(offset[1])];
+ // invert matrix calculation
+ rect[0] -= offset[0];
+ rect[1] -= offset[1];
+ rect[2] -= offset[0];
+ rect[3] -= offset[1];
+ // Fix for bug in ie clip-auto when height/width is not defined
+ /* @todo set dynamic w/h based on clip bounds or find
+ * another workaround fix */
+ dstyle.width = "10800px";
+ dstyle.height = "10800px";
+ }
+ else if (!node.clipRect) {
+ dstyle.top = "0";
+ dstyle.left = "0";
+ dstyle.width = o.paper.width + "px";
+ dstyle.height = o.paper.height + "px";
+ node.parentNode.insertBefore(div, node);
+ div.appendChild(node);
+ node.clipRect = div;
+ }
+ dstyle.position = "absolute";
+ dstyle.clip = R.format("rect({1}px {2}px {3}px {0}px)", rect);
+ }
+ if (!params["clip-rect"]) {
+ if (isGroup && o.clip) {
+ node.style.clip = "rect(auto auto auto auto)";
+ delete o.clip;
+ }
+ else if (node.clipRect) {
+ node.clipRect.style.clip = "rect(auto auto auto auto)";
+ }
+ }
+ }
+ if (o.textpath) {
+ var textpathStyle = o.textpath.style;
+ params.font && (textpathStyle.font = params.font);
+ params["font-family"] && (textpathStyle.fontFamily = '"' + params["font-family"].split(",")[0].replace(/^['"]+|['"]+$/g, E) + '"');
+ params["font-size"] && (textpathStyle.fontSize = params["font-size"]);
+ params["font-weight"] && (textpathStyle.fontWeight = params["font-weight"]);
+ params["font-style"] && (textpathStyle.fontStyle = params["font-style"]);
+ }
+ if ("arrow-start" in params) {
+ addArrow(res, params["arrow-start"]);
+ }
+ if ("arrow-end" in params) {
+ addArrow(res, params["arrow-end"], 1);
+ }
+ if (params.opacity != null ||
+ params["stroke-width"] != null ||
+ params.fill != null ||
+ params.src != null ||
+ params.stroke != null ||
+ params["stroke-width"] != null ||
+ params["stroke-opacity"] != null ||
+ params["fill-opacity"] != null ||
+ params["stroke-dasharray"] != null ||
+ params["stroke-miterlimit"] != null ||
+ params["stroke-linejoin"] != null ||
+ params["stroke-linecap"] != null) {
+ var fill = node.getElementsByTagName(fillString),
+ newfill = false,
+ fillOpacity = -1;
+ fill = fill && fill[0];
+ !fill && (newfill = fill = createNode(fillString));
+ if (o.type == "image" && params.src) {
+ fill.src = params.src;
+ }
+ params.fill && (fill.on = true);
+ if (fill.on == null || params.fill == "none" || params.fill === null) {
+ fill.on = false;
+ }
+ if (fill.on && params.fill) {
+ var isURL = Str(params.fill).match(R._ISURL);
+ if (isURL) {
+ fill.parentNode == node && node.removeChild(fill);
+ fill.rotate = true;
+ fill.src = isURL[1];
+ fill.type = "tile";
+ var bbox = o.getBBox(1);
+ fill.position = bbox.x + S + bbox.y;
+ o._.fillpos = [bbox.x, bbox.y];
+
+ R._preload(isURL[1], function() {
+ o._.fillsize = [this.offsetWidth, this.offsetHeight];
+ });
+ } else {
+ var color = R.getRGB(params.fill);
+ fill.color = color.hex;
+ fill.src = E;
+ fill.type = "solid";
+ if (color.error && (res.type in {
+ circle: 1,
+ ellipse: 1
+ } || Str(params.fill).charAt() != "r") && addGradientFill(res, params.fill, fill)) {
+ a.fill = "none";
+ a.gradient = params.fill;
+ fill.rotate = false;
+ }
+ else if ("opacity" in color && !("fill-opacity" in params)) {
+ fillOpacity = color.opacity;
+ }
+ }
+ }
+ if (fillOpacity !== -1 || "fill-opacity" in params || "opacity" in params) {
+ var opacity = ((+a["fill-opacity"] + 1 || 2) - 1) * ((+a.opacity + 1 || 2) - 1) * ((+fillOpacity + 1 || 2) - 1);
+ opacity = mmin(mmax(opacity, 0), 1);
+ fill.opacity = opacity;
+ if (fill.src) {
+ fill.color = "none";
+ }
+ }
+ node.appendChild(fill);
+ var stroke = (node.getElementsByTagName("stroke") && node.getElementsByTagName("stroke")[0]),
+ newstroke = false;
+ !stroke && (newstroke = stroke = createNode("stroke"));
+ if ((params.stroke && params.stroke != "none") ||
+ params["stroke-width"] ||
+ params["stroke-opacity"] != null ||
+ params["stroke-dasharray"] ||
+ params["stroke-miterlimit"] ||
+ params["stroke-linejoin"] ||
+ params["stroke-linecap"]) {
+ stroke.on = true;
+ }
+ (params.stroke == "none" || params.stroke === null || stroke.on == null || params.stroke == 0 || params["stroke-width"] == 0) && (stroke.on = false);
+ var strokeColor = R.getRGB(('stroke' in params) ? params.stroke : a.stroke);
+ stroke.on && params.stroke && (stroke.color = strokeColor.hex);
+ opacity = ((+a["stroke-opacity"] + 1 || 2) - 1) * ((+a.opacity + 1 || 2) - 1) * ((+strokeColor.opacity + 1 || 2) - 1);
+ var width = (toFloat(params["stroke-width"]) || 1) * .75;
+ opacity = mmin(mmax(opacity, 0), 1);
+ params["stroke-width"] == null && (width = a["stroke-width"]);
+ params["stroke-width"] && (stroke.weight = width);
+ width && width < 1 && (opacity *= width) && (stroke.weight = 1);
+ stroke.opacity = opacity;
+
+ params["stroke-linejoin"] && (stroke.joinstyle = params["stroke-linejoin"]) || newstroke && (newstroke.joinstyle = 'miter');
+ stroke.miterlimit = params["stroke-miterlimit"] || 8;
+ params["stroke-linecap"] && (stroke.endcap = params["stroke-linecap"] == "butt" ? "flat" : params["stroke-linecap"] == "square" ? "square" : "round");
+ if (params["stroke-dasharray"]) {
+ var dasharray = {
+ "-": "shortdash",
+ ".": "shortdot",
+ "-.": "shortdashdot",
+ "-..": "shortdashdotdot",
+ ". ": "dot",
+ "- ": "dash",
+ "--": "longdash",
+ "- .": "dashdot",
+ "--.": "longdashdot",
+ "--..": "longdashdotdot"
+ };
+ stroke.dashstyle = dasharray[has](params["stroke-dasharray"]) ? dasharray[params["stroke-dasharray"]] :
+ ((params["stroke-dasharray"].join && params["stroke-dasharray"].join(' ')) || E);
+ }
+ newstroke && node.appendChild(stroke);
+ }
+ if (res.type == "text") {
+ res.paper.canvas.style.display = E;
+ var span = res.paper.span,
+ m = 100,
+ fontSize = a.font && a.font.match(/\d+(?:\.\d*)?(?=px)/),
+ lineHeight = a['line-height'] && (a['line-height']+E).match(/\d+(?:\.\d*)?(?=px)/);
+ s = span.style;
+ a.font && (s.font = a.font);
+ a["font-family"] && (s.fontFamily = a["font-family"]);
+ a["font-weight"] && (s.fontWeight = a["font-weight"]);
+ a["font-style"] && (s.fontStyle = a["font-style"]);
+ fontSize = toFloat(a["font-size"] || fontSize && fontSize[0]) || 10;
+ s.fontSize = fontSize * m + "px";
+ lineHeight = toFloat(a["line-height"] || lineHeight && lineHeight[0]) || 12;
+ a["line-height"] && (s.lineHeight = lineHeight * m + 'px');
+ res.textpath.string && (span.innerHTML = Str(res.textpath.string).replace(/"));
+ var brect = span.getBoundingClientRect();
+ res.W = a.w = (brect.right - brect.left) / m;
+ res.H = a.h = (brect.bottom - brect.top) / m;
+ // res.paper.canvas.style.display = "none";
+ res.X = a.x;
+ res.Y = a.y;
+ var leading = lineHeight - fontSize;
+ switch(a["vertical-align"]) {
+ case "top":
+ res.bby = res.H / 2; // + leading;
+ break;
+ case "bottom":
+ res.bby = -res.H / 2; // - leading;
+ break;
+ default:
+ res.bby = 0;
+ }
+
+ ("x" in params || "y" in params || res.bby !== undefined) && (res.path.v = R.format("m{0},{1}l{2},{1}", round(a.x * zoom), round((a.y + (res.bby || 0)) * zoom), round(a.x * zoom) + 1));
+ var dirtyattrs = ["x", "y", "text", "font", "font-family", "font-weight", "font-style", "font-size", "line-height"];
+ for (var d = 0, dd = dirtyattrs.length; d < dd; d++)
+ if (dirtyattrs[d] in params) {
+ res._.dirty = 1;
+ break;
+ }
+
+ // text-anchor emulation
+ switch (a["text-anchor"]) {
+ case "start":
+ res.textpath.style["v-text-align"] = "left";
+ res.bbx = res.W / 2;
+ break;
+ case "end":
+ res.textpath.style["v-text-align"] = "right";
+ res.bbx = -res.W / 2;
+ break;
+ default:
+ res.textpath.style["v-text-align"] = "center";
+ res.bbx = 0;
+ break;
+ }
+ res.textpath.style["v-text-kern"] = true;
+ }
+ // res.paper.canvas.style.display = E;
+ },
+ addGradientFill = function(o, gradient, fill) {
+ o.attrs = o.attrs || {};
+ var attrs = o.attrs,
+ pow = Math.pow,
+ opacity,
+ oindex,
+ type = "linear",
+ fxfy = ".5 .5";
+ o.attrs.gradient = gradient;
+ gradient = Str(gradient).replace(R._radial_gradient, function(all, opts) {
+ type = "radial";
+ opts = opts && opts.split(',') || [];
+
+ // fx,fy of vml is cx,cy of svg
+ var cx = opts[0],
+ cy = opts[1],
+ r = opts[2],
+ fx = opts[3],
+ fy = opts[4],
+ units = opts[5];
+ if (fx && fy) {
+ fx = toFloat(fx);
+ fy = toFloat(fy);
+ pow(fx - .5, 2) + pow(fy - .5, 2) > .25 && (fy = sqrt(.25 - pow(fx - .5, 2)) * ((fy > .5) * 2 - 1) + .5);
+ fxfy = fx + S + fy;
+ }
+ return E;
+ });
+ gradient = gradient.split(/\s*\-\s*/);
+ if (type == "linear") {
+ var angle = gradient.shift();
+ angle = -toFloat(angle);
+ if (isNaN(angle)) {
+ return null;
+ }
+ }
+ var dots = R._parseDots(gradient);
+ if (!dots) {
+ return null;
+ }
+ o = o.shape || o.node;
+ if (dots.length) {
+ o.removeChild(fill);
+ fill.on = true;
+ fill.method = "none";
+ fill.color = dots[0].color;
+ fill.color2 = dots[dots.length - 1].color;
+ //For VML use first and last available alpha
+ var clrs = [],
+ opacity1 = 1,
+ opacity2 = dots[0].opacity === undefined ? 1 : dots[0].opacity;
+ for (var i = 0, ii = dots.length; i < ii; i++) {
+ dots[i].offset && clrs.push(dots[i].offset + S + dots[i].color);
+ if (dots[i].opacity !== undefined) {
+ opacity1 = dots[i].opacity;//update with latest avaible opacity
+ }
+ }
+ fill.colors = clrs.length ? clrs.join() : "0% " + fill.color;
+ //set opacity1 & opacity2
+ fill.opacity = opacity1;
+ fill['o:opacity2'] = opacity2;
+ if (type == "radial") {
+ fill.type = "gradientTitle";
+ fill.focus = "100%";
+ fill.focussize = "0 0";
+ fill.focusposition = fxfy;
+ fill.angle = 0;
+ } else {
+ // fill.rotate= true;
+ fill.type = "gradient";
+ fill.angle = (270 - angle) % 360;
+ }
+ o.appendChild(fill);
+ }
+ return 1;
+ },
+ Element = function(node, vml, group) {
+ var o = this,
+ parent = group || vml;
+
+ o.node = o[0] = node;
+ node.raphael = true;
+ node.raphaelid = o.id = R._oid++;
+
+ o.X = 0;
+ o.Y = 0;
+
+ o.attrs = o.attrs || {};
+ o.styles = o.styles || {};
+ o.followers = o.followers || [];
+
+ o.paper = vml;
+ o.ca = o.customAttributes = o.customAttributes ||
+ new vml._CustomAttributes();
+
+ o.matrix = R.matrix();
+ o._ = {
+ transform: [],
+ sx: 1,
+ sy: 1,
+ dx: 0,
+ dy: 0,
+ deg: 0,
+ dirty: 1,
+ dirtyT: 1
+ };
+
+ o.parent = parent;
+ !parent.bottom && (parent.bottom = o);
+
+ o.prev = parent.top;
+ parent.top && (parent.top.next = o);
+ parent.top = o;
+ o.next = null;
+ };
+ var elproto = R.el;
+
+ Element.prototype = elproto;
+ elproto.constructor = Element;
+
+ elproto.transform = function(tstr) {
+ if (tstr == null) {
+ return this._.transform;
+ }
+ var vbs = this.paper._viewBoxShift,
+ vbt = vbs ? "s" + [vbs.scale, vbs.scale] + "-1-1t" + [vbs.dx, vbs.dy] : E,
+ oldt;
+
+ if (vbs) {
+ oldt = tstr = Str(tstr).replace(/\.{3}|\u2026/g, this._.transform || E);
+ }
+
+ R._extractTransform(this, vbt + tstr);
+
+ var matrix = this.matrix.clone(),
+ skew = this.skew,
+ o = this.node,
+ split,
+ isGrad = ~Str(this.attrs.fill).indexOf("-"),
+ isPatt = !Str(this.attrs.fill).indexOf("url(");
+ matrix.translate(-.5, -.5);
+ if (isPatt || isGrad || this.type == "image") {
+ skew.matrix = "1 0 0 1";
+ skew.offset = "0 0";
+ split = matrix.split();
+ if ((isGrad && split.noRotation) || !split.isSimple) {
+ o.style.filter = matrix.toFilter();
+ var bb = this.getBBox(),
+ bbt = this.getBBox(1),
+ xget = bb.x2 && bbt.x2 && 'x2' || 'x',
+ yget = bb.y2 && bbt.y2 && 'y2' || 'y',
+ dx = bb[xget] - bbt[xget],
+ dy = bb[yget] - bbt[yget];
+ o.coordorigin = (dx * -zoom) + S + (dy * -zoom);
+ setCoords(this, 1, 1, dx, dy, 0);
+ } else {
+ o.style.filter = E;
+ setCoords(this, split.scalex, split.scaley, split.dx, split.dy, split.rotate);
+ }
+ } else {
+ o.style.filter = E;
+ skew.matrix = Str(matrix);
+ skew.offset = matrix.offset();
+ }
+ oldt && (this._.transform = oldt);
+
+ return this;
+ };
+ elproto.rotate = function(deg, cx, cy) {
+ if (this.removed) {
+ return this;
+ }
+ if (deg == null) {
+ return;
+ }
+ deg = Str(deg).split(separator);
+ if (deg.length - 1) {
+ cx = toFloat(deg[1]);
+ cy = toFloat(deg[2]);
+ }
+ deg = toFloat(deg[0]);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ var bbox = this.getBBox(1);
+ cx = bbox.x + bbox.width / 2;
+ cy = bbox.y + bbox.height / 2;
+ }
+ this._.dirtyT = 1;
+ this.transform(this._.transform.concat([["r", deg, cx, cy]]));
+ return this;
+ };
+ elproto.translate = function(dx, dy) {
+ if (this.removed) {
+ return this;
+ }
+ dx = Str(dx).split(separator);
+ if (dx.length - 1) {
+ dy = toFloat(dx[1]);
+ }
+ dx = toFloat(dx[0]) || 0;
+ dy = +dy || 0;
+ if (this._.bbox) {
+ this._.bbox.x += dx;
+ this._.bbox.y += dy;
+ }
+ this.transform(this._.transform.concat([["t", dx, dy]]));
+ return this;
+ };
+ elproto.scale = function(sx, sy, cx, cy) {
+ if (this.removed) {
+ return this;
+ }
+ sx = Str(sx).split(separator);
+ if (sx.length - 1) {
+ sy = toFloat(sx[1]);
+ cx = toFloat(sx[2]);
+ cy = toFloat(sx[3]);
+ isNaN(cx) && (cx = null);
+ isNaN(cy) && (cy = null);
+ }
+ sx = toFloat(sx[0]);
+ (sy == null) && (sy = sx);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ var bbox = this.getBBox(1);
+ }
+ cx = cx == null ? bbox.x + bbox.width / 2 : cx;
+ cy = cy == null ? bbox.y + bbox.height / 2 : cy;
+
+ this.transform(this._.transform.concat([["s", sx, sy, cx, cy]]));
+ this._.dirtyT = 1;
+ return this;
+ };
+ elproto.hide = function(soft) {
+ var o = this;
+ !o.removed && (o.node.style.display = "none");
+ return o;
+ };
+
+ elproto.show = function(soft) {
+ var o = this;
+ !o.removed && (o.node.style.display = E);
+ return o;
+ };
+ elproto._getBBox = function() {
+ if (this.removed) {
+ return {};
+ }
+ return {
+ x: this.X + (this.bbx || 0) - this.W / 2,
+ y: this.Y + (this.bby || 0) - this.H / 2,
+ width: this.W,
+ height: this.H
+ };
+ };
+ elproto.remove = function() {
+ if (this.removed || !this.parent.canvas) {
+ return;
+ }
+ var i,
+ thisNode = R._engine.getNode(this);
+ this.paper.__set__ && this.paper.__set__.exclude(this);
+ eve.unbind("raphael.*.*." + this.id);
+ while (i = this.followers.pop()) {
+ i.el.remove();
+ }
+ this.shape && this.shape.parentNode.removeChild(this.shape);
+ thisNode.parentNode.removeChild(thisNode);
+ R._tear(this, this.paper);
+ for (var i in this) {
+ this[i] = typeof this[i] == "function" ? R._removedFactory(i) : null;
+ }
+ this.removed = true;
+ };
+ elproto.css = function (name, value) {
+ // do not parse css in case element is removed.
+ if (this.removed) {
+ return this;
+ }
+
+ // process as getter when a single key is sent as parameter.
+ if (value == null && R.is(name, "string")) {
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.styles) {
+ out[name] = this.styles[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ // process as getter when multiple keys are pre-sent as array.
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.styles(name[i]);
+ }
+ return out;
+ }
+ // convert single key-value setter into object style standard.
+ if (value != null) {
+ var params = {};
+ params[name] = value;
+ } else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+ // iterate on keys and set style or raise events.
+ var otherkey, doattrs = {};
+ for (var key in params) {
+ otherkey = key.replace(/\B([A-Z]{1})/g, "-$1").toLowerCase();
+ // Replace "color" with fill
+ (otherkey === 'color' && this.type === 'text') && (otherkey = 'fill');
+ // If keys are supported via attr then use attr instead of css.
+ if (R._availableAttrs[has](otherkey)) {
+ doattrs[otherkey] = params[key];
+ doattrs.dirty = true;
+ continue;
+ }
+ eve("raphael.css." + otherkey + "." + this.id, this, params[key], otherkey);
+ (params[key] != undefined) && (this.node.style[otherkey] = params[key]);
+ this.styles[otherkey] = params[key];
+ }
+
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ this.followers[i].el.css(params);
+ }
+
+ // apply css via attrs
+ if (doattrs[has]("dirty")) {
+ delete doattrs.dirty;
+ this.attr(doattrs);
+ }
+
+ return this;
+ };
+ elproto.attr = function(name, value) {
+ if (this.removed) {
+ return this;
+ }
+ if (name == null) {
+ var res = {};
+ for (var a in this.attrs)
+ if (this.attrs[has](a)) {
+ res[a] = this.attrs[a];
+ }
+ res.gradient && res.fill == "none" && (res.fill = res.gradient) && delete res.gradient;
+ res.transform = this._.transform;
+ res.visibility = this.node.style.display === "none" ? "hidden" : "visible";
+ return res;
+ }
+ if (value == null && R.is(name, "string")) {
+ if (name == fillString && this.attrs.fill == "none" && this.attrs.gradient) {
+ return this.attrs.gradient;
+ }
+ if (name == "visibility") {
+ return this.node.style.display === "none" ? "hidden" : "visible";
+ }
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.attrs) {
+ out[name] = this.attrs[name];
+ } else if (R.is(this.ca[name], "function")) {
+ out[name] = this.ca[name].def;
+ } else {
+ out[name] = R._availableAttrs[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ if (this.attrs && value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.attr(name[i]);
+ }
+ return out;
+ }
+ var params;
+ if (value != null) {
+ params = {};
+ params[name] = value;
+ }
+ value == null && R.is(name, "object") && (params = name);
+ for (var key in params) {
+ eve("raphael.attr." + key + "." + this.id, this, params[key], key);
+ }
+ if (params) {
+ var todel = {};
+ for (key in this.ca)
+ if (this.ca[key] && params[has](key) && R.is(this.ca[key], "function") && !this.ca['_invoked' + key]) {
+ this.ca['_invoked' + key] = true; // prevent recursion
+ var par = this.ca[key].apply(this, [].concat(params[key]));
+ delete this.ca['_invoked' + key];
+
+ for (var subkey in par) {
+ if (par[has](subkey)) {
+ params[subkey] = par[subkey];
+ }
+ }
+ this.attrs[key] = params[key];
+ if (par === false) {
+ todel[key] = params[key];
+ delete params[key];
+ }
+ }
+ // this.paper.canvas.style.display = "none";
+ if ('text' in params && this.type == "text") {
+ this.textpath.string = params.text.replace(/ /ig, '\n');
+ }
+ setFillAndStroke(this, params);
+ var follower;
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ follower = this.followers[i];
+ (follower.cb && !follower.cb.call(follower.el, params, this)) ||
+ follower.el.attr(params);
+ }
+ for (var subkey in todel) {
+ params[subkey] = todel[subkey];
+ }
+ // this.paper.canvas.style.display = E;
+ }
+ return this;
+ };
+
+ elproto.blur = function(size) {
+ var s = this.node.runtimeStyle,
+ f = s.filter;
+ f = f.replace(blurregexp, E);
+ if (+size !== 0) {
+ this.attrs.blur = size;
+ s.filter = f + S + ms + ".Blur(pixelradius=" + (+size || 1.5) + ")";
+ s.margin = R.format("-{0}px 0 0 -{0}px", round(+size || 1.5));
+ } else {
+ s.filter = f;
+ s.margin = 0;
+ delete this.attrs.blur;
+ }
+ return this;
+ };
+
+ elproto.on = function(eventType, handler) {
+ if (this.removed) {
+ return this;
+ }
+
+ this.node['on'+ eventType] = function() {
+ var evt = R._g.win.event;
+ evt.target = evt.srcElement;
+ handler(evt);
+ };
+ return this;
+ };
+
+ R._engine.getNode = function (el) {
+ var node = el.node || el[0].node;
+ return node.clipRect || node;
+ };
+ R._engine.getLastNode = function (el) {
+ var node = el.node || el[el.length - 1].node;
+ return node.clipRect || node;
+ };
+
+ R._engine.group = function(vml, id, group) {
+ var el = R._g.doc.createElement("div"),
+ p = new Element(el, vml, group);
+
+ el.style.cssText = cssDot;
+
+ id && (el.className = ['red', id, p.id].join('-'));
+ (group || vml).canvas.appendChild(el);
+
+ p.type = 'group';
+ p.canvas = p.node;
+ p.transform = R._engine.group.transform;
+ p.top = null;
+ p.bottom = null;
+
+ return p;
+ };
+
+ R._engine.group.transform = function(tstr) {
+ if (tstr == null) {
+ return this._.transform;
+ }
+
+ var o = this,
+ s = o.node.style,
+ c = o.clip,
+ vbs = o.paper._viewBoxShift,
+ vbt = vbs ? "s" + [vbs.scale, vbs.scale] + "-1-1t" + [vbs.dx, vbs.dy] : E,
+ oldt,
+ matrix,
+ offset,
+ tx,
+ ty;
+
+ if (vbs) {
+ oldt = tstr = Str(tstr).replace(/\.{3}|\u2026/g, o._.transform || E);
+ }
+ R._extractTransform(o, vbt + tstr);
+ matrix = o.matrix;
+ offset = matrix.offset();
+ tx = toFloat(offset[0]) || 0;
+ ty = toFloat(offset[1]) || 0;
+
+ s.left = tx + "px";
+ s.top = ty + "px";
+ s.zoom = (o._.tzoom = matrix.get(0)) + E;
+
+ /* @todo try perform relative group transform, thus avoiding
+ * transform on clipping */
+ c && (s.clip = R.format("rect({1}px {2}px {3}px {0}px)", [
+ c[0] - tx, c[1] - ty, c[2] - tx, c[3] - ty
+ ]));
+
+ return o;
+ };
+
+ R._engine.path = function(pathString, vml, group) {
+ var el = createNode("shape");
+ el.style.cssText = cssDot;
+ el.coordsize = zoom + S + zoom;
+ el.coordorigin = vml.coordorigin;
+ var p = new Element(el, vml, group),
+ attr = {
+ fill: "none",
+ stroke: "#000"
+ };
+
+ pathString && (attr.path = pathString);
+ p.type = "path";
+ p.path = [];
+ p.Path = E;
+ setFillAndStroke(p, attr);
+ (group || vml).canvas.appendChild(el);
+
+ var skew = createNode("skew");
+ skew.on = true;
+ el.appendChild(skew);
+ p.skew = skew;
+ return p;
+ };
+
+ R._engine.rect = function(vml, x, y, w, h, r, group) {
+ var path = R._rectPath(x, y, w, h, r),
+ res = vml.path(path, group),
+ a = res.attrs;
+ res.X = a.x = x;
+ res.Y = a.y = y;
+ res.W = a.width = w;
+ res.H = a.height = h;
+ a.r = r;
+ a.path = path;
+ res.type = "rect";
+ return res;
+ };
+ R._engine.ellipse = function(vml, x, y, rx, ry, group) {
+ var res = vml.path(undefined, group);
+ res.X = x - rx;
+ res.Y = y - ry;
+ res.W = rx * 2;
+ res.H = ry * 2;
+ res.type = "ellipse";
+ setFillAndStroke(res, {
+ cx: x,
+ cy: y,
+ rx: rx,
+ ry: ry
+ });
+ return res;
+ };
+ R._engine.circle = function(vml, x, y, r, group) {
+ var res = vml.path(undefined, group);
+ res.X = x - r;
+ res.Y = y - r;
+ res.W = res.H = r * 2;
+ res.type = "circle";
+ setFillAndStroke(res, {
+ cx: x,
+ cy: y,
+ r: r
+ });
+ return res;
+ };
+ R._engine.image = function(vml, src, x, y, w, h, group) {
+ var path = R._rectPath(x, y, w, h),
+ res = vml.path(path, group).attr({
+ stroke: "none"
+ }),
+ a = res.attrs,
+ node = res.node,
+ fill = node.getElementsByTagName(fillString)[0];
+ a.src = src;
+ res.X = a.x = x;
+ res.Y = a.y = y;
+ res.W = a.width = w;
+ res.H = a.height = h;
+ a.path = path;
+ res.type = "image";
+ fill.parentNode == node && node.removeChild(fill);
+ fill.rotate = true;
+ fill.src = src;
+ fill.type = "tile";
+ res._.fillpos = [x, y];
+ res._.fillsize = [w, h];
+ node.appendChild(fill);
+ setCoords(res, 1, 1, 0, 0, 0);
+ return res;
+ };
+ R._engine.text = function(vml, x, y, text, group) {
+ var el = createNode("shape"),
+ path = createNode("path"),
+ o = createNode("textpath");
+ x = x || 0;
+ y = y || 0;
+ text = text;
+ path.v = R.format("m{0},{1}l{2},{1}", round(x * zoom), round(y * zoom), round(x * zoom) + 1);
+ path.textpathok = true;
+ o.string = Str(text).replace(/ /ig, '\n');
+ o.on = true;
+ el.style.cssText = cssDot;
+ el.coordsize = zoom + S + zoom;
+ el.coordorigin = "0 0";
+ var p = new Element(el, vml, group),
+ attr = {
+ fill: "#000",
+ stroke: "none",
+ text: text
+ };
+
+ p.shape = el;
+ p.path = path;
+ p.textpath = o;
+ p.type = "text";
+ p.attrs.text = Str(text || E);
+ p.attrs.x = x;
+ p.attrs.y = y;
+ p.attrs.w = 1;
+ p.attrs.h = 1;
+ setFillAndStroke(p, attr);
+ el.appendChild(o);
+ el.appendChild(path);
+ (group || vml).canvas.appendChild(el);
+
+ var skew = createNode("skew");
+ skew.on = true;
+ el.appendChild(skew);
+ p.skew = skew;
+ return p;
+ };
+
+ R._engine.setSize = function(width, height) {
+ var cs = this.canvas.style;
+ this.width = width;
+ this.height = height;
+ width == +width && (width += "px");
+ height == +height && (height += "px");
+ cs.width = width;
+ cs.height = height;
+ cs.clip = "rect(0 " + width + " " + height + " 0)";
+ if (this._viewBox) {
+ R._engine.setViewBox.apply(this, this._viewBox);
+ }
+ return this;
+ };
+ R._engine.setViewBox = function(x, y, w, h, fit) {
+ eve("raphael.setViewBox", this, this._viewBox, [x, y, w, h, fit]);
+ var width = this.width,
+ height = this.height,
+ size = 1 / mmax(w / width, h / height),
+ H, W;
+ if (fit) {
+ H = height / h;
+ W = width / w;
+ if (w * H < width) {
+ x -= (width - w * H) / 2 / H;
+ }
+ if (h * W < height) {
+ y -= (height - h * W) / 2 / W;
+ }
+ }
+ this._viewBox = [x, y, w, h, !!fit];
+ this._viewBoxShift = {
+ dx: -x,
+ dy: -y,
+ scale: size
+ };
+ this.forEach(function(el) {
+ el.transform("...");
+ });
+ return this;
+ };
+ var createNode;
+ R._engine.initWin = function(win) {
+ var doc = win.document;
+ doc.createStyleSheet().addRule(".rvml", "behavior:url(#default#VML)");
+ try {
+ !doc.namespaces.rvml && doc.namespaces.add("rvml", "urn:schemas-microsoft-com:vml");
+ createNode = R._createNode = function(tagName, attrs) {
+ var el = doc.createElement(''),
+ prop;
+ for (prop in attrs) {
+ el[prop] = Str(attrs[prop]);
+ }
+ return el;
+ };
+ } catch (e) {
+ createNode = R._createNode = function(tagName, attrs) {
+ var el = doc.createElement('<' + tagName + ' xmlns="urn:schemas-microsoft.com:vml" class="rvml">'),
+ prop;
+ for (prop in attrs) {
+ el[prop] = Str(attrs[prop]);
+ }
+ return el;
+ };
+ }
+ };
+ R._engine.initWin(R._g.win);
+ R._engine.create = function() {
+ var con = R._getContainer.apply(0, arguments),
+ container = con.container,
+ height = con.height,
+ s,
+ width = con.width,
+ x = con.x,
+ y = con.y;
+ if (!container) {
+ throw new Error("VML container not found.");
+ }
+ var res = new R._Paper,
+ c = res.canvas = R._g.doc.createElement("div"),
+ cs = c.style;
+ x = x || 0;
+ y = y || 0;
+ width = width || 512;
+ height = height || 342;
+ res.width = width;
+ res.height = height;
+ width == +width && (width += "px");
+ height == +height && (height += "px");
+ res.coordsize = zoom * 1e3 + S + zoom * 1e3;
+ res.coordorigin = "0 0";
+ res.span = R._g.doc.createElement("span");
+ res.span.style.cssText = "position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;";
+ c.appendChild(res.span);
+ cs.cssText = R.format("top:0;left:0;width:{0};height:{1};display:inline-block;cursor:default;position:relative;clip:rect(0 {0} {1} 0);overflow:hidden", width, height);
+ if (container == 1) {
+ R._g.doc.body.appendChild(c);
+ cs.left = x + "px";
+ cs.top = y + "px";
+ cs.position = "absolute";
+ } else {
+ if (container.firstChild) {
+ container.insertBefore(c, container.firstChild);
+ } else {
+ container.appendChild(c);
+ }
+ }
+ res.renderfix = function() {
+ };
+ return res;
+ };
+ R.prototype.clear = function() {
+ eve("raphael.clear", this);
+ this.canvas.innerHTML = E;
+ this.span = R._g.doc.createElement("span");
+ this.span.style.cssText = "position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;display:inline;";
+ this.canvas.appendChild(this.span);
+ this.bottom = this.top = null;
+ };
+ R.prototype.remove = function() {
+ eve("raphael.remove", this);
+ this.canvas.parentNode.removeChild(this.canvas);
+ for (var i in this) {
+ this[i] = typeof this[i] == "function" ? R._removedFactory(i) : null;
+ }
+ return true;
+ };
+
+ var setproto = R.st;
+ for (var method in elproto)
+ if (elproto[has](method) && !setproto[has](method)) {
+ setproto[method] = (function(methodname) {
+ return function() {
+ var arg = arguments;
+ return this.forEach(function(el) {
+ el[methodname].apply(el, arg);
+ });
+ };
+ })(method);
+ }
+})();
+
+/*jslint forin: true, regexp: true, todo: true, white: false, browser: true,
+ sloppy: true, white: true, eqeq: false, newcap: true, nomen: true */
+
+/*global FusionCharts */
+
+/**
+ * Raphael Canvas Extension
+ */
+
+(function(){
+ if (!R.canvas) {
+ return;
+ }
+ var win = R._g.win,
+ doc = R._g.doc,
+ g = R._g,
+
+ STRING = 'string',
+ PX = 'px',
+
+ separator = /[, ]+/,
+
+ Str = win.String,
+ toInt = win.parseInt,
+ toFloat = win.parseFloat,
+
+ math = win.Math,
+ mmax = math.max,
+ mmin = math.min,
+ pi = math.PI,
+ mathFloor = math.floor,
+
+ eve = R.eve,
+ paperproto = R.fn,
+ elproto = R.el,
+ setproto = R.st,
+
+ clone = R.clone,
+ deg2rad = pi / 180,
+ rad2deg = 180 / pi,
+
+ DEFAULT_FILL = "#fff",
+ DEFAULT_STROKE = "#000",
+ has = "hasOwnProperty",
+ S = " ",
+ /* @todo: detect touch */
+ supportsTouch = (('ontouchstart' in win) || (navigator.msMaxTouchPoints > 0)),
+ events = ("click dblclick mousedown mousemove mouseout mouseover mouseup touchstart touchmove touchend touchcancel").split(S),
+ noHandle = false,
+
+ $,
+ FauxNode,
+ Element,
+ draggable = [],
+ drag = [],
+ dragMove = function(e) {
+ var x = e.clientX,
+ y = e.clientY,
+ scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft,
+ dragi,
+ j = drag.length;
+ while (j--) {
+ dragi = drag[j];
+ if (supportsTouch) {
+ var i = e.touches.length,
+ touch;
+ while (i--) {
+ touch = e.touches[i];
+ if (touch.identifier == dragi.el._drag.id) {
+ x = touch.clientX;
+ y = touch.clientY;
+ (e.originalEvent ? e.originalEvent : e).preventDefault();
+ break;
+ }
+ }
+ } else {
+ e.preventDefault();
+ }
+
+ //var node = dragi.el.node;
+ //o,
+ //next = node.nextSibling,
+ //parent = node.parentNode,
+ //display = node.style.display;
+
+ /* @todo: implement raphael.drag.over */
+
+ //g.win.opera && parent.removeChild(node);
+ //node.style.display = "none";
+ //o = dragi.el.paper.getElementByPoint(x, y);
+ //node.style.display = display;
+ //g.win.opera && (next ? parent.insertBefore(node, next) : parent.appendChild(node));
+ //o && eve("raphael.drag.over." + dragi.el.id, dragi.el, o);
+ x += scrollX;
+ y += scrollY;
+ eve("raphael.drag.move." + dragi.el.id, dragi.move_scope || dragi.el, x - dragi.el._drag.x, y - dragi.el._drag.y, x, y, e);
+ }
+ },
+ dragUp = function(e) {
+ R.unmousemove(dragMove).unmouseup(dragUp);
+ var i = drag.length,
+ dragi;
+ while (i--) {
+ dragi = drag[i];
+ dragi.el._drag = {};
+ eve("raphael.drag.end." + dragi.el.id, dragi.end_scope || dragi.start_scope || dragi.move_scope || dragi.el, e);
+ }
+ drag = [];
+ };
+
+
+ if (!R.canvas) {
+ return;
+ }
+
+ $ = R._createNode = function(el, attr) {
+ if (attr) {
+ if (typeof el === STRING) {
+ el = $(el);
+ }
+ for (var key in attr)
+ if (attr.hasOwnProperty(key)) {
+ el.setAttribute(key, Str(attr[key]));
+ }
+ } else {
+ el = doc.createElement(el);
+ }
+ return el;
+ };
+
+ R._getConnectedNodes = function (node) {
+ return {
+ above: [],
+ below: []
+ };
+ };
+
+ R._getTargetNode = function (coords) {
+ var x = coords[0],
+ y = coords[1];
+
+
+ };
+
+ R._containerEventHandler = function (event) {
+ event = event || win.event;
+
+ if (noHandle) {
+ return;
+ }
+
+ /* @todo: do not use offsetX and offsetY */
+ var x = event.offsetX,//mathFloor(event.pageX || (event.clientX + doc.body.scrollLeft + doc.documentElement.scrollLeft) || 0),
+ y = event.offsetY,//mathFloor(event.pageY || (event.clientY + doc.body.scrollTop + doc.documentElement.scrollTop) || 0),
+ type = event.type,
+ node = R._getTargetNode([x, y]);
+
+ //console.log(type + ":: x: " + x + " y: " + y);
+
+ /*
+ if (typeof node.listeners[type] === 'function') {
+ node.listeners[type].call(node, e);
+ }*/
+
+ };
+
+
+
+
+
+ FauxNode = function (parent) {
+ this.type = "basic";
+
+ this.owner = parent;
+
+ // Element reference needed to get attrs so that there is no need to
+ // maintain a redundant copy.
+ this._rElement = null;
+
+ // False if this element does not need mouse interactivity.
+ this.mouseInteractions = false;
+ this.matrix = null;
+
+ // An outline path that will be the outline of the shape or in case of
+ // text the rect that bounds the text.
+ this.outlinePath = null;
+
+ this.conf = {};
+ };
+
+ FauxNode.prototype = {
+
+ constructor: FauxNode,
+
+ /**
+ * This method does the complete rendering of the element, including
+ * (re)setting the bbox and image map.
+ *
+ * @returns {_L10.FauxNode.prototype}
+ */
+ render: function () {
+
+ var o = this;
+
+ o.draw();
+ o.setBBox();
+
+ return o;
+ },
+
+ /**
+ * Applies the transforms and clipping to the context and draws the
+ * element.
+ *
+ * @returns {undefined}
+ */
+ draw: function () {
+ var o = this,
+ ctx = o.context,
+ el = o._rElement,
+ m = el.matrix,
+ isClipped = o.isClipped,
+ attrs = o.validateAttrs(),
+ cr,
+ end;
+
+
+ // After the validation the attributes need to be set to the element
+ // as validateAttrs works on a copy of the attrs object.
+ el.attrs = attrs;
+
+ ctx.save();
+
+ ctx.fillStyle = attrs.fill;
+ ctx.strokeStyle = attrs.stroke;
+ ctx.lineWidth = attrs['stroke-width'];
+
+ // Applying the clip before the context is transformed as is the case
+ // in case of SVG.
+ if (cr = attrs['clip-rect']) {
+ cr = cr.split(" ");
+ ctx.rect(cr[0], cr[1], cr[2], cr[3]);
+ ctx.clip();
+ isClipped = o.isClipped = true;
+ }
+
+ o.applyTransform(m);
+
+ o.paint();
+
+ ctx.restore();
+ },
+
+ /**
+ * Parse the attributes provided to paint the element shape using the
+ * canvas context.
+ *
+ * Overridden by individual derived FauxNodes
+ */
+ paint: function () {
+ },
+
+ /**
+ * The redraw of the FauxNodes is to be handled by the CanvasObjectModel
+ * instance as it involves redrawing all the elements corresponsing to
+ * that canvas in the proper order.
+ */
+ redraw: function () {
+ this.COMInstance.redraw(this);
+ },
+
+ /**
+ * Clears the rectangle corresponding to the bounding box of the node.
+ */
+ clear: function () {
+ var o = this,
+ ctx = o.context,
+ bbox = o._bbox;
+
+ /* @todo: while clearing the stroke-width also needs to be accounted for. */
+ if (bbox) {
+ ctx.clearRect(bbox.x, bbox.y, bbox.width, bbox.height);
+ }
+ },
+
+ /**
+ * Creates the area node in the image map so that mouse interactivity
+ * can be emulated using it.
+ */
+ addMouseInteractivity: function () {
+ var o = this,
+ attrs = o._rElement.attrs,
+ bbox = o._bbox,
+ map = o.owner.wrapper._map,
+ shape = (o.type === 'circle') ? 'circle' : 'rect',
+ coords = (shape === 'circle') ? [attrs.cx, attrs.cy, attrs.r].join(",")
+ : [bbox.x, bbox.y, bbox.x2, bbox.y2].join(","),
+ area,
+ end;
+
+ area = $('area', {shape: shape, coords: coords});
+ if (map.firstChild) {
+ map.insertBefore(area, map.firstChild);
+ }
+ else {
+ map.appendChild(area);
+ }
+
+ /* @todo: should be put this check here */
+ o._mouseArea = area;
+
+ // Needed for paths.
+ o.eventListeners = {};
+ },
+
+
+ /**
+ * Update the coords of the area node (image map) corresponding to the FauxNode
+ * after it has been modified by changing attributes.
+ */
+ updateMapAreaCoords: function () {
+ var o = this,
+ oArea = o._mouseArea,
+ bbox = o._bbox;
+
+ if (oArea) {
+ //o.context.strokeRect(bbox.x, bbox.y, bbox.width, bbox.height);
+ if (o instanceof CircleFauxNode) {
+ var r = bbox.width / 2;
+ oArea.setAttribute('coords',
+ [bbox.x + r, bbox.y + r, r].join(","));
+ }
+ else {
+ oArea.setAttribute('coords',
+ [bbox.x, bbox.y, bbox.x2, bbox.y2].join(","));
+ }
+ }
+ },
+
+ /**
+ * Applies the transform given the transformation matrix to the context.
+ *
+ * @param {type} matrix
+ * @param {type} bbox
+ * @param {type} dontsetbbox
+ *
+ * @returns {_L10.FauxNode.prototype.transformBBox.tbox}
+ */
+ applyTransform: function (m) {
+ var o = this,
+ ctx = o.context,
+ split;
+
+ if (m) {
+ split = m.split();
+ ctx.translate(split.dx, split.dy);
+ !split.noRotation && ctx.rotate(deg2rad * split.rotate);
+ ctx.scale(split.scalex, split.scaley);
+ }
+ },
+
+ /**
+ *
+ */
+ setBBox: function () {
+ var o = this,
+ el = o._rElement,
+ m = el.matrix,
+ parent = o.owner,
+ pm = parent.getTransformMatrix && parent.getTransformMatrix();
+
+ if (pm) {
+ pm = pm.clone();
+ pm.add(m);
+ m = pm;
+ }
+
+ if (o.outlinePath) {
+ o._bbox = R.pathBBox(R.transformPath(o.outlinePath, m.toTransformString()).toString());
+ }
+ else {
+ o.setShapeBBox(m);
+ }
+
+ o._mouseArea && o.updateMapAreaCoords();
+ },
+
+ getBBox: function () {
+ return this._bbox;
+ },
+
+ drawPath: function (path) {
+ var o = this,
+ ctx = o.context,
+ len = (path && path.length) || 0,
+ pp = PathParser,
+ i = 0,
+ command,
+ x,
+ y,
+ end;
+
+ // The PathParser object has been borrowed from canvg. All 3rd party attributions implied.
+ pp.reset();
+ pp.setTokens(path);
+
+ //var bb = new BoundingBox();
+ if (ctx != null) {
+ ctx.beginPath();
+ }
+
+ while (!pp.isEnd()) {
+ pp.nextCommand();
+ switch (pp.command) {
+ case 'M':
+ case 'm':
+ var p = pp.getAsCurrentPoint();
+ pp.addMarker(p);
+ //bb.addPoint(p.x, p.y);
+ if (ctx != null) {
+ ctx.moveTo(p.x, p.y);
+ }
+ pp.start = pp.current;
+ while (!pp.isCommandOrEnd()) {
+ var p = pp.getAsCurrentPoint();
+ pp.addMarker(p, pp.start);
+ //bb.addPoint(p.x, p.y);
+ if (ctx != null) {
+ ctx.lineTo(p.x, p.y);
+ }
+ }
+ break;
+ case 'L':
+ case 'l':
+ while (!pp.isCommandOrEnd()) {
+ var c = pp.current;
+ var p = pp.getAsCurrentPoint();
+ pp.addMarker(p, c);
+ //bb.addPoint(p.x, p.y);
+ if (ctx != null) {
+ ctx.lineTo(p.x, p.y);
+ }
+ }
+ break;
+ case 'H':
+ case 'h':
+ while (!pp.isCommandOrEnd()) {
+ var newP = new Point((pp.isRelativeCommand() ? pp.current.x : 0) + pp.getScalar(), pp.current.y);
+ pp.addMarker(newP, pp.current);
+ pp.current = newP;
+ //bb.addPoint(pp.current.x, pp.current.y);
+ if (ctx != null) {
+ ctx.lineTo(pp.current.x, pp.current.y);
+ }
+ }
+ break;
+ case 'V':
+ case 'v':
+ while (!pp.isCommandOrEnd()) {
+ var newP = new Point(pp.current.x, (pp.isRelativeCommand() ? pp.current.y : 0) + pp.getScalar());
+ pp.addMarker(newP, pp.current);
+ pp.current = newP;
+ //bb.addPoint(pp.current.x, pp.current.y);
+ if (ctx != null) {
+ ctx.lineTo(pp.current.x, pp.current.y);
+ }
+ }
+ break;
+ case 'C':
+ case 'c':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var p1 = pp.getPoint();
+ var cntrl = pp.getAsControlPoint();
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, p1);
+ //bb.addBezierCurve(curr.x, curr.y, p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) {
+ ctx.bezierCurveTo(p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ }
+ break;
+ case 'S':
+ case 's':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var p1 = pp.getReflectedControlPoint();
+ var cntrl = pp.getAsControlPoint();
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, p1);
+ //bb.addBezierCurve(curr.x, curr.y, p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) {
+ ctx.bezierCurveTo(p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ }
+ break;
+ case 'Q':
+ case 'q':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var cntrl = pp.getAsControlPoint();
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, cntrl);
+ //bb.addQuadraticCurve(curr.x, curr.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) ctx.quadraticCurveTo(cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ break;
+ case 'T':
+ case 't':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var cntrl = pp.getReflectedControlPoint();
+ pp.control = cntrl;
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, cntrl);
+ //bb.addQuadraticCurve(curr.x, curr.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) ctx.quadraticCurveTo(cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ break;
+ case 'A':
+ case 'a':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var rx = pp.getScalar();
+ var ry = pp.getScalar();
+ var xAxisRotation = pp.getScalar() * (Math.PI / 180.0);
+ var largeArcFlag = pp.getScalar();
+ var sweepFlag = pp.getScalar();
+ var cp = pp.getAsCurrentPoint();
+
+ // Conversion from endpoint to center parameterization
+ // http://www.w3.org/TR/SVG11/implnote.html#ArcImplementationNotes
+ // x1', y1'
+ var currp = new Point(
+ Math.cos(xAxisRotation) * (curr.x - cp.x) / 2.0 + Math.sin(xAxisRotation) * (curr.y - cp.y) / 2.0,
+ -Math.sin(xAxisRotation) * (curr.x - cp.x) / 2.0 + Math.cos(xAxisRotation) * (curr.y - cp.y) / 2.0
+ );
+ // adjust radii
+ var l = Math.pow(currp.x,2)/Math.pow(rx,2)+Math.pow(currp.y,2)/Math.pow(ry,2);
+ if (l > 1) {
+ rx *= Math.sqrt(l);
+ ry *= Math.sqrt(l);
+ }
+ // cx', cy'
+ var s = (largeArcFlag == sweepFlag ? -1 : 1) * Math.sqrt(
+ ((Math.pow(rx,2)*Math.pow(ry,2))-(Math.pow(rx,2)*Math.pow(currp.y,2))-(Math.pow(ry,2)*Math.pow(currp.x,2))) /
+ (Math.pow(rx,2)*Math.pow(currp.y,2)+Math.pow(ry,2)*Math.pow(currp.x,2))
+ );
+ if (isNaN(s)) s = 0;
+ var cpp = new Point(s * rx * currp.y / ry, s * -ry * currp.x / rx);
+ // cx, cy
+ var centp = new Point(
+ (curr.x + cp.x) / 2.0 + Math.cos(xAxisRotation) * cpp.x - Math.sin(xAxisRotation) * cpp.y,
+ (curr.y + cp.y) / 2.0 + Math.sin(xAxisRotation) * cpp.x + Math.cos(xAxisRotation) * cpp.y
+ );
+ // vector magnitude
+ var m = function(v) { return Math.sqrt(Math.pow(v[0],2) + Math.pow(v[1],2)); }
+ // ratio between two vectors
+ var r = function(u, v) { return (u[0]*v[0]+u[1]*v[1]) / (m(u)*m(v)) }
+ // angle between two vectors
+ var a = function(u, v) { return (u[0]*v[1] < u[1]*v[0] ? -1 : 1) * Math.acos(r(u,v)); }
+ // initial angle
+ var a1 = a([1,0], [(currp.x-cpp.x)/rx,(currp.y-cpp.y)/ry]);
+ // angle delta
+ var u = [(currp.x-cpp.x)/rx,(currp.y-cpp.y)/ry];
+ var v = [(-currp.x-cpp.x)/rx,(-currp.y-cpp.y)/ry];
+ var ad = a(u, v);
+ if (r(u,v) <= -1) ad = Math.PI;
+ if (r(u,v) >= 1) ad = 0;
+
+ // for markers
+ var dir = 1 - sweepFlag ? 1.0 : -1.0;
+ var ah = a1 + dir * (ad / 2.0);
+ var halfWay = new Point(
+ centp.x + rx * Math.cos(ah),
+ centp.y + ry * Math.sin(ah)
+ );
+ pp.addMarkerAngle(halfWay, ah - dir * Math.PI / 2);
+ pp.addMarkerAngle(cp, ah - dir * Math.PI);
+
+ //bb.addPoint(cp.x, cp.y); // TODO: this is too naive, make it better
+ if (ctx != null) {
+ var r = rx > ry ? rx : ry;
+ var sx = rx > ry ? 1 : rx / ry;
+ var sy = rx > ry ? ry / rx : 1;
+
+ ctx.translate(centp.x, centp.y);
+ ctx.rotate(xAxisRotation);
+ ctx.scale(sx, sy);
+ ctx.arc(0, 0, r, a1, a1 + ad, 1 - sweepFlag);
+ ctx.scale(1/sx, 1/sy);
+ ctx.rotate(-xAxisRotation);
+ ctx.translate(-centp.x, -centp.y);
+ }
+ }
+ break;
+ case 'Z':
+ case 'z':
+ if (ctx != null) ctx.closePath();
+ pp.current = pp.start;
+ }
+ }
+
+ o.outlinePath = path;
+
+ return o;
+ },
+
+ /**
+ * Methods to add and remove event listeners emulating the DOM of
+ * standard browsers (and also the non-standard one).
+ */
+ addEventListener: function () {
+ var o = this,
+ args = arguments,
+ eventName = args && args[0],
+ handler = args && args[1],
+ area,
+ checkPathHandler,
+ end;
+
+ if (!o._mouseArea) {
+ o.addMouseInteractivity();
+ }
+
+ area = o._mouseArea;
+
+ if (typeof eventName === 'string' && typeof handler === 'function') {
+ /*
+ * If the shape has an associated path then we need to check if
+ * the mouse is within the co-ordinates of the path.
+ */
+ if (o._path) {
+
+ /*
+ * If the event being listened to is mouseover, mouseout or
+ * mousemove then the mouse position has to be constantly
+ * monitored and the event handler called explicitly when
+ * appropriate.
+ */
+ if (eventName === 'mouseover' || eventName === 'mouseout' || eventName === 'mousemove') {
+
+ if (!o._mousemoveAdded) {
+ var startListening = (function (node) {
+
+ var isInside = false,
+ isOutside = true,
+ transition = false;
+
+ return function (event) {
+ /* @todo: replace layerX and layerY with
+ * standard ways of determining mouse position.
+ */
+ var x = event.layerX,
+ y = event.layerY;
+
+ transition = false;
+ /*
+ * Check if the current position of the mouse
+ * pointer lies within the path or not.
+ */
+ if (R.isPointInsidePath(node._transformPath, x, y)) {
+ /*
+ * @todo: fix isPointInsidePath to return the
+ * proper result when the mouse pointer is on
+ * the same horizontal/vertical lines as one
+ * of the vertices of the path.
+ */
+ isInside = true;
+ if (isOutside) {
+ isOutside = false;
+ transition = true;
+ }
+ }
+ else {
+ isOutside = true;
+ if (isInside) {
+ isInside = false;
+ transition = true;
+ }
+ }
+
+ // Based on the state of the flags, fire the
+ // appropriate event handlers.
+ if (isOutside && transition && node.eventListeners['mouseout']) {
+ node.eventListeners['mouseout'].apply(this, arguments);
+ }
+ if (isInside) {
+ if (transition && node.eventListeners['mouseover']) {
+ node.eventListeners['mouseover'].apply(this, arguments);
+ }
+ if (node.eventListeners['mousemove']) {
+ node.eventListeners['mousemove'].apply(this, arguments);
+ }
+ }
+ };
+ })(o);
+
+ area.addEventListener('mousemove', startListening, false);
+ o._mousemoveAdded = true;
+ }
+
+ o.eventListeners[eventName] = handler;
+ }
+ else {
+ var checkPathHandler = (function (node, handler) {
+ return function (event) {
+ if (R.isPointInsidePath(node._path, event.layerX, event.layerY)) {
+ handler.apply(this, arguments);
+ }
+ };
+ }(o, handler));
+
+ area.addEventListener(eventName, checkPathHandler, false);
+ }
+ }
+ else {
+ area.addEventListener(eventName, handler, false);
+ }
+ }
+ },
+
+ removeEventListener: function () {
+ var o = this,
+ args = arguments,
+ eventName = args && args[0],
+ handler = args && args[1],
+ area,
+ end;
+
+ if (!o._mouseArea) {
+ return;
+ }
+ area = o._mouseArea;
+
+ if (typeof eventName === 'string' && typeof handler === 'function') {
+ area.removeEventListener(eventName, handler);
+ }
+ },
+
+ attachEvent: function () {
+ },
+
+ detachEvent: function () {
+ },
+
+ validateAttrs: function (attrs) {
+
+ var o = this,
+ elAttrs = clone(o._rElement.attrs),
+ attr,
+ val;
+
+ if (attrs === null) {
+ if (o._isValid) {
+ return elAttrs;
+ }
+ else {
+ o._isValid = true;
+ }
+ }
+
+ attrs = attrs || elAttrs;
+
+ for (attr in attrs) {
+ val = attrs[attr];
+
+ switch (attr) {
+
+ default:
+ continue;
+ }
+ }
+
+ return attrs;
+ },
+
+ attrs: function () {
+ }
+ };
+
+
+
+
+ var NodeListItem = function (node) {
+ this.node = node;
+ this.next = null;
+ this.prev = null;
+ },
+
+ NodeList = function () {
+ this.top = null;
+ this.bottom = null;
+ };
+
+ NodeList.prototype = {
+
+ constructor: NodeList,
+
+ add: function (node) {
+
+ node = new NodeListItem(node);
+
+ if (!this.bottom) {
+ this.bottom = node;
+ }
+ if (this.top) {
+ this.top.next = node;
+ }
+ node.next = null;
+ node.prev = this.top;
+
+ this.top = node;
+ },
+
+ addList: function (list) {
+ if (!this.bottom) {
+ this.bottom = list.bottom;
+ }
+
+ if (this.top) {
+ this.top.next = list.bottom;
+ list.bottom.prev = this.top;
+ }
+
+ this.top = list.top;
+ },
+
+ toFront: function (node) {
+ if (this.top === node) {
+ return false;
+ }
+
+ if (this.bottom === node) {
+ this.bottom = node.next;
+ }
+
+ //var map = node.node.canvas._map,
+ // area = node.node._mouseArea;
+
+ node.prev && (node.prev.next = node.next);
+ node.next && (node.next.prev = node.prev);
+
+ this.top.next = node;
+ node.prev = this.top;
+ node.next = null;
+
+ this.top = node;
+
+ /*if (map.firstChild) {
+ map.insertBefore(area, map.firstChild);
+ }
+ else {
+ map.appendChild(area);
+ }
+
+ node.redraw();*/
+ },
+
+ toBack: function (node) {
+ if (this.bottom === node) {
+ return false;
+ }
+
+ if (this.top === node) {
+ this.top = node.prev;
+ }
+
+ //var map = node.canvas._map,
+ // area = node._mouseArea;
+
+ node.prev && (node.prev.next = node.next);
+ node.next && (node.next.prev = node.prev);
+
+ this.bottom.prev = node;
+ node.prev = null;
+ node.next = this.bottom;
+
+ this.bottom = node;
+
+ /*map.appendChild(area);
+
+ node.redraw();*/
+ },
+
+ insertBefore: function () {
+ },
+
+ insertAfter: function () {
+ },
+
+ each: function (fn, args) {
+ var item = this.bottom;
+
+ while (item) {
+ fn.apply(item.node, args);
+ item = item.next;
+ }
+ },
+
+ iterate: function (fn, args) {
+ var item = this.bottom,
+ retVal = true;
+
+ while (item) {
+ retVal = fn.apply(item.node, args);
+
+ if (retVal === false) {
+ break;
+ }
+
+ item = item.next;
+ }
+ },
+
+ dispose: function () {
+
+ this.each(function () {
+ this.node.dispose && this.node.dispose();
+ });
+
+ this.top = null;
+ this.bottom = null;
+ },
+ };
+
+ /**
+ *
+ ncowner, a NodeCollection that corresponds to the collection of which
+ the layer neing created is a part
+
+ above, a NodeCollection iterator, that indicated the collection the layer has to be rendered. If
+ not provided then this is the first layer of ncowner.
+ */
+ var CanvasLayer = function (ncowner, canvas) {
+ this.items = new NodeList();
+
+ this.owner = ncowner;
+ //this.above = above;
+ this.element = null;
+
+ if (canvas) {
+ this.element = canvas;
+ }
+ else {
+ this.init();
+ }
+ };
+
+ CanvasLayer.prototype = {
+ constructor: CanvasLayer,
+
+ appendChild: function () {
+ var o = this,
+ ownerWrapper = o.owner.wrapper,
+ ele = this.element;
+
+ if (ownerWrapper._image) {
+ ownerWrapper.insertBefore(ele, ownerWrapper._image);
+ }
+ else {
+ ownerWrapper.appendChild(ele);
+ }
+ },
+
+ insertBefore: function () {
+
+ },
+
+ insertAfter: function () {
+
+ },
+
+ init: function () {
+ this.element = $("canvas");
+ // CHECKPOINT: width and height in %?
+ $(this.element, {
+ width: this.owner.wrapper.offsetWidth,
+ height: this.owner.wrapper.offsetHeight
+ });
+
+ this.element.style.cssText = "position:absolute;left:0;top:0;";
+
+ this.appendChild();
+ },
+
+ getCanvas: function () {
+ return this.element;
+ },
+
+ getContext: function () {
+ return this.element.getContext('2d');
+ },
+
+ addToLayer: function (node) {
+ this.items.add(node);
+ },
+
+ mergeWithLayerOnTop: function (layerObj) {
+ this.items.addList(layerObj.items);
+ layerObj.dispose(true);
+ },
+
+ mergeWithLayerOnBottom: function (layerObj) {
+ layerObj.items.addList(this.items);
+ this.items = layerObj.items;
+ layerObj.dispose(true);
+ },
+
+ dispose: function (softDispose) {
+
+ if (!softDispose) {
+ this.items.each(function () {
+ this.dispose();
+ });
+ }
+
+ this.items = null;
+ this.owner = null;
+
+ this.element.parentNode.removeChild(this.element);
+ this.element = null;
+ }
+ };
+
+ var NodeCollection = function (parent, wrapper, canvas) {
+
+ this.nodeItems = new NodeList();
+ this.collectionItems = new NodeList();
+ this.layerItems = new NodeList();
+
+ this.owner = this.parent = parent;
+ this.layerOnTop = null;
+
+ this.currentLayer = null;
+ this.baseLayer = null;
+
+ if (wrapper) {
+ this.wrapper = wrapper;
+ this.currentLayer = this.baseLayer = new CanvasLayer(this, canvas);
+ }
+ else {
+ this.init();
+ }
+ };
+
+ NodeCollection.prototype = {
+ constructor: NodeCollection,
+
+ init: function () {
+
+ var o = this,
+ parent = o.parent,
+ imageMap = parent.wrapper._image,
+ wrapper = $("div");
+
+ // Hacky but need a refernce to the image map to addEventListeners.
+ wrapper.style.cssText = "width:100%;height:100%;position:absolute;left:0;top:0;";
+
+ wrapper._map = parent.wrapper._map;
+
+ if (imageMap) {
+ parent.wrapper.insertBefore(wrapper, imageMap);
+ }
+ else {
+ parent.wrapper.appendChild(wrapper);
+ }
+
+ o.wrapper = wrapper;
+ o.currentLayer = o.baseLayer = new CanvasLayer(o);
+ },
+
+ getCurrentContext: function () {
+ return this.currentLayer.getContext();
+ },
+
+ setLayerOnTop: function (layerObj) {
+ this.layerOnTop = layerObj;
+ },
+
+ getCurrentCanvas: function () {
+ return this.currentLayer.getCanvas();
+ },
+
+ addNode: function (node) {
+ this.nodeItems.add(node);
+
+ if (node.type === "group") {
+ this.addCollection(node);
+ }
+ else {
+ this.currentLayer.addToLayer(node);
+ }
+ },
+
+ addCollection: function (collectionNode) {
+
+ collectionNode = collectionNode || new NodeCollection(this);
+
+ this.collectionItems.add(collectionNode);
+ this.currentLayer = new CanvasLayer(this);
+ this.layerItems.add(this.currentLayer);
+ collectionNode.setLayerOnTop(this.currentLayer);
+ },
+
+ dispose: function () {
+ this.nodeItems.dispose();
+ this.collectionItems.dispose();
+ this.layerItems.dispose();
+ this.owner = this.parent = null;
+
+ this.ownerLayer = null;
+ this.currentLayer = null;
+ this.baseLayer = null;
+ }
+ };
+
+ /**
+ * The CanvasObjectModel will be a layer of abstraction above the individual
+ * FauxNodes created to emulate the DOM in case of canvas rendering.
+ * The engine will be the point of contact for Raphael._engine that will be
+ * the direct consumer of the FauxNodes.
+ */
+ var CanvasObjectModel = function (cnvs, wrpr, width, height) {
+
+ var com = this,
+ root = new NodeCollection(null, wrpr, cnvs);
+
+ //root.set
+
+ com.width = width;
+ com.height = height;
+
+ com.createNode = function (type, parent) {
+
+ parent = parent || root;
+
+ var node,
+ nodeItems = parent.nodeItems,
+ layer = parent.currentLayer,
+ canvasEle = layer.getCanvas();
+
+ switch (type) {
+ case 'rect':
+ node = new RectFauxNode(parent);
+ break;
+
+ case 'circle':
+ node = new CircleFauxNode(parent);
+ break;
+
+ case 'path':
+ node = new PathFauxNode(parent);
+ break;
+
+ case 'text':
+ node = new TextFauxNode(parent);
+ break;
+
+ case 'group':
+ node = new GroupFauxNode(parent);
+ parent.addCollection(node);
+ break;
+
+ default:
+ node = new FauxNode(canvasEle);
+ }
+
+ node.COMInstance = this;
+ nodeItems.add(node);
+ layer.addToLayer(node);
+
+ return node;
+ };
+
+ /**
+ * This method is needed to redraw a node. Redraw is to be handled at the
+ * COM level as redrawing one node needs all the (connected) nodes to be
+ * redrawn in the right order.
+ *
+ * @param {type} node The node that needs to be redrawn.
+ *
+ * @returns {undefined}
+ */
+ com.redraw = function (node) {
+
+ // Check if node is a group or a shape.
+ var nodeList,
+ childNode,
+ layer;
+
+ if (node.type === "group") {
+ nodeList = node.nodeItems;
+ node.render();
+ }
+ else {
+
+ layer = node.layer;
+ nodeList = layer.items;
+ childNode = nodeList.bottom;
+
+ // Clear the canvas.
+ layer.element.width = layer.element.width;
+
+ while (childNode) {
+ fNode = childNode.node;
+ (fNode.type !== "group") && fNode.render();
+ childNode = childNode.next;
+ }
+
+ }
+
+ };
+
+ com.insertBefore = function (node) {
+
+ };
+
+ com.insertAfter = function (node) {
+
+ };
+
+ com.removeNode = function (node) {
+
+ };
+
+ com.refreshNode = function (node) {
+
+ };
+
+ com.refreshAll = function () {
+
+ };
+ };
+
+
+
+ Point = function(x, y) {
+ this.x = x;
+ this.y = y;
+ }
+
+ Point.prototype.angleTo = function(p) {
+ return Math.atan2(p.y - this.y, p.x - this.x);
+ }
+
+ Point.prototype.applyTransform = function(v) {
+ var xp = this.x * v[0] + this.y * v[2] + v[4];
+ var yp = this.x * v[1] + this.y * v[3] + v[5];
+ this.x = xp;
+ this.y = yp;
+ }
+
+ PathParser = new (function() {
+
+ this.tokens = null;
+
+ this.setTokens = function (d) {
+ if (typeof d === 'string') {
+ this.tokens = d.split(' ');
+ }
+ else {
+ this.tokens = d;
+ }
+ };
+
+ this.reset = function() {
+ this.i = -1;
+ this.command = '';
+ this.previousCommand = '';
+ this.start = new Point(0, 0);
+ this.control = new Point(0, 0);
+ this.current = new Point(0, 0);
+ this.points = [];
+ this.angles = [];
+ };
+
+ this.isEnd = function() {
+ return this.i >= this.tokens.length - 1;
+ }
+
+ this.isCommandOrEnd = function() {
+ if (this.isEnd()) {
+ return true;
+ }
+
+ return this.tokens[this.i + 1].toString().match(/^[A-Za-z]$/) != null;
+ }
+
+ this.isRelativeCommand = function() {
+ switch(this.command)
+ {
+ case 'm':
+ case 'l':
+ case 'h':
+ case 'v':
+ case 'c':
+ case 's':
+ case 'q':
+ case 't':
+ case 'a':
+ case 'z':
+ return true;
+ break;
+ }
+ return false;
+ }
+
+ this.getToken = function() {
+ this.i++;
+ return this.tokens[this.i];
+ }
+
+ this.getScalar = function() {
+ return parseFloat(this.getToken());
+ }
+
+ this.nextCommand = function() {
+ this.previousCommand = this.command;
+ this.command = this.getToken();
+ }
+
+ this.getPoint = function() {
+ var p = new Point(this.getScalar(), this.getScalar());
+ return this.makeAbsolute(p);
+ }
+
+ this.getAsControlPoint = function() {
+ var p = this.getPoint();
+ this.control = p;
+ return p;
+ }
+
+ this.getAsCurrentPoint = function() {
+ var p = this.getPoint();
+ this.current = p;
+ return p;
+ }
+
+ this.getReflectedControlPoint = function() {
+ if (this.previousCommand.toLowerCase() != 'c' &&
+ this.previousCommand.toLowerCase() != 's' &&
+ this.previousCommand.toLowerCase() != 'q' &&
+ this.previousCommand.toLowerCase() != 't' ){
+ return this.current;
+ }
+
+ // reflect point
+ var p = new Point(2 * this.current.x - this.control.x, 2 * this.current.y - this.control.y);
+ return p;
+ }
+
+ this.makeAbsolute = function(p) {
+ if (this.isRelativeCommand()) {
+ p.x += this.current.x;
+ p.y += this.current.y;
+ }
+ return p;
+ }
+
+ this.addMarker = function(p, from, priorTo) {
+ // if the last angle isn't filled in because we didn't have this point yet ...
+ if (priorTo != null && this.angles.length > 0 && this.angles[this.angles.length-1] == null) {
+ this.angles[this.angles.length-1] = this.points[this.points.length-1].angleTo(priorTo);
+ }
+ this.addMarkerAngle(p, from == null ? null : from.angleTo(p));
+ }
+
+ this.addMarkerAngle = function(p, a) {
+ this.points.push(p);
+ this.angles.push(a);
+ }
+
+ this.getMarkerPoints = function() {
+ return this.points;
+ }
+
+ this.getMarkerAngles = function() {
+ for (var i=0; i maxR) {
+ attrs.r = maxR;
+ }
+
+ if (val < 0) {
+ attrs.r = 0;
+ }
+ break;
+
+ case "width":
+ case "height":
+ if (val < 0) {
+ attrs[attr] = 0;
+ };
+ break;
+
+ default:
+ continue;
+ }
+ }
+
+ return attrs;
+ },
+
+ setShapeBBox: function (m) {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attrs,
+ sX = m.get(0),
+ sY = m.get(3),
+ tX = m.get(4),
+ tY = m.get(5),
+ strokeW = attrs['stroke-width'];
+
+ o._bbox = {
+ x: ((attrs.x * sX) + tX) - strokeW,
+ y: ((attrs.y * sY) + tY) - strokeW,
+ width: (attrs.width * sX) + (2 * strokeW),
+ height: (attrs.height * sY) + (2 * strokeW),
+ };
+
+ o._bbox.x2 = o._bbox.x + o._bbox.width;
+ o._bbox.y2 = o._bbox.y + o._bbox.height;
+
+ o.X = o._bbox.x;
+ o.Y = o._bbox.y;
+ o.W = o._bbox.width;
+ o.H = o._bbox.height;
+ }
+ });
+
+ var CircleFauxNode = function (parentObj) {
+ this.type = "circle";
+ this._isValid = false;
+
+ this.parent = this.owner = parentObj;
+ this.context = parentObj.getCurrentContext();
+ this.layer = parentObj.currentLayer;
+ },
+
+ PathFauxNode = function (parentObj) {
+ this.type = "path";
+ this._isValid = false;
+
+ this.parent = this.owner = parentObj;
+ this.context = parentObj.getCurrentContext();
+ this.layer = parentObj.currentLayer;
+ },
+
+ TextFauxNode = function (parentObj) {
+ this.type = "text";
+ this._isValid = false;
+
+ this.parent = this.owner = parentObj;
+ this.context = parentObj.getCurrentContext();
+ this.layer = parentObj.currentLayer;
+ },
+
+ GroupFauxNode = function (parent, width, height) {
+
+ this.type = "group";
+
+ this.nodeItems = new NodeList();
+ this.collectionItems = new NodeList();
+ this.layerItems = new NodeList();
+
+ this.owner = this.parent = parent;
+ this.layerOnTop = parent.currentLayer;
+
+ this.currentLayer = null;
+ this.baseLayer = null;
+
+ this.init();
+ };
+
+ CircleFauxNode.prototype = R.extend(new FauxNode(), {
+
+ constructor: CircleFauxNode,
+
+ paint: function () {
+ var o = this,
+ ctx = o.context,
+ attrs = o.validateAttrs(),
+ x = attrs.cx,
+ y = attrs.cy,
+ r = attrs.r,
+ /* @todo: provide support for rx, ry */
+ rx = r || attrs.rx,
+ ry = r || attrs.ry;
+
+ if (attrs.r) {
+
+ o.drawPath(["M", x + r, y, "A", rx, ry, 0, 1, 0, x - r, y, "A", rx, ry, 0, 1, 0, x + r, y, "Z"]);
+
+ if (attrs['stroke-width']) {
+ var strokeAlpha = attrs['stroke-opacity'] === undefined ? attrs['opacity'] : attrs['stroke-opacity'];
+ if (strokeAlpha !== undefined) {
+ ctx.globalAlpha = strokeAlpha;
+ }
+ ctx.stroke();
+ }
+ var fillAlpha = attrs['fill-opacity'] === undefined ? attrs['opacity'] : attrs['fill-opacity'];
+ if (fillAlpha !== undefined) {
+ ctx.globalAlpha = fillAlpha;
+ }
+ ctx.fill();
+ }
+
+ return;
+ },
+
+ setShapeBBox: function (m) {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attrs,
+ sX = m.get(0),
+ sY = m.get(3),
+ tX = m.get(4),
+ tY = m.get(5),
+ strokeW = attrs['stroke-width'];
+
+ o._bbox = {
+ x: tX + ((attrs.cx - attrs.r) * sX) - strokeW,
+ y: tY + ((attrs.cy - attrs.r) * sY) - strokeW,
+ width: 2 * (strokeW + (attrs.r * sX)),
+ height: 2 * ((attrs.r * sY) + strokeW)
+ };
+
+ o._bbox.x2 = o._bbox.x + o._bbox.width;
+ o._bbox.y2 = o._bbox.y + o._bbox.height;
+
+ o.X = o._bbox.x;
+ o.Y = o._bbox.y;
+ o.W = o._bbox.width;
+ o.H = o._bbox.height;
+ }
+ });
+
+ PathFauxNode.prototype = R.extend(new FauxNode(), {
+
+ constructor: PathFauxNode,
+
+ paint: function () {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attrs,
+ path = el.attr('path'),
+ m = el.matrix,
+ ctx = o.context;
+
+ // 1. Get the path from the path attribute
+ // 2. Accept paths in different array formats.
+ // 3. Optimize as this can potentially be a huge pain-point.
+ // 4. Draw path mapping M,L,H,V etc to canvas APIs
+ o.drawPath(path);
+ o._transformPath = R.transformPath(path, m.toTransformString());
+
+ var strokeAlpha = attrs['stroke-opacity'] === undefined ? attrs['opacity'] : attrs['stroke-opacity'];
+ if (strokeAlpha !== undefined) {
+ ctx.globalAlpha = strokeAlpha;
+ }
+ ctx.stroke();
+ var fillAlpha = attrs['fill-opacity'] === undefined ? attrs['opacity'] : attrs['fill-opacity'];
+ if (fillAlpha !== undefined) {
+ ctx.globalAlpha = fillAlpha;
+ }
+ ctx.fill();
+
+ return;
+ }
+ });
+
+ TextFauxNode.prototype = R.extend(new FauxNode(), {
+
+ constructor: TextFauxNode,
+
+ paint: function () {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attr(),
+ text = attrs['text'],
+ stroke = attrs['stroke'],
+ valign = attrs['vertical-align'],
+ halign = attrs['text-anchor'],
+ x = attrs['x'],
+ y = attrs['y'],
+ m = el.matrix,
+ ctx = o.context,
+ path;
+
+ // apply the font styles, if any
+
+ // find the dimensions of the text using the given styles.
+ // All the dimensions should be present in the attrs user provided OR default.
+ var fontSize = attrs['font-size'] || 10,
+ lineHeight = attrs['line-height'] || toInt(fontSize, 10) * 1.2,
+ fontArr = ["normal", fontSize, attrs['font']];
+
+ // draw the text.
+ ctx.fillStyle = stroke;
+ ctx.font = fontArr.join(" ");
+
+ if (text) {
+ var texts = Str(text).split(/\n| /ig),
+ totalHeight = texts.length * lineHeight,
+ totalWidth = -Infinity,
+ startX = Infinity,
+ startY,
+ width,
+ textX,
+ textY;
+
+ if (valign === "top") {
+ startY = y + lineHeight;
+ }
+ else if (valign === "middle") {
+ startY = y - (totalHeight / 2) + (lineHeight / 2);
+ }
+ else { // valign is bottom.
+ startY = y - totalHeight + lineHeight;
+ }
+
+ for (var i = 0, ii = texts.length; i < ii; i += 1) {
+
+ text = texts[i];
+ textY = startY + (lineHeight * i);
+ width = ctx.measureText(text).width;
+
+ if (halign === "start") {
+ textX = x;
+ }
+ else if (halign === "middle") {
+ textX = x - (width / 2);
+ }
+ else {
+ textX = x - width;
+ }
+
+ totalWidth = mmax(totalWidth, width);
+ startX = mmin(startX, textX);
+
+ ctx.fillText(text, textX, textY);
+ }
+
+ el._textdirty = false;
+ }
+
+ o.outlinePath = [
+ "M",
+ startX,
+ startY - (lineHeight / 1.4),
+ "H",
+ startX + totalWidth,
+ "V",
+ startY - lineHeight + totalHeight,
+ "H",
+ startX,
+ "V",
+ startY - (lineHeight / 1.4)
+ ];
+
+ return;
+ }
+ });
+
+ GroupFauxNode.prototype = R.extend(R.extend(new FauxNode, NodeCollection.prototype), {
+
+ constructor: GroupFauxNode,
+
+ draw: function () {
+
+ // Clear the group canvas first.
+ this.layerItems.each(function () {
+ this.element.width = this.element.width;
+ });
+
+ FauxNode.prototype.draw.apply(this, arguments);
+ },
+
+ /**
+ * This method does the complete rendering of the element, including
+ * (re)setting the bbox and image map.
+ *
+ * @returns {_L10.FauxNode.prototype}
+ */
+ render: function () {
+
+ var o = this;
+
+ o.draw();
+ o.setBBox();
+
+ return o;
+ },
+
+ paint: function () {
+ var o = this,
+ list = o.nodeList,
+ el = o._rElement,
+ canvas = o.canvas,
+ attrs = el.attrs,
+ childNode = list.bottom;
+
+ /* @todo: Clean this up */
+ if (attrs.opacity !== undefined) {
+ this.layerItems.each(function () {
+ this.getContext().globalAlpha = attrs.opacity;
+ });
+ }
+
+ while (childNode) {
+ childNode.render();
+ childNode = childNode.next;
+ }
+ },
+
+ setBBox: function () {
+
+ },
+
+ addMouseInteractivity: function () {
+ },
+
+ applyTransform: function (m) {
+ var o = this,
+ parent = o.parent,
+ parentMatrix = parent.getTransformMatrix && parent.getTransformMatrix();
+
+ // Parent is a group element with a transformation applied to it.
+ if (parentMatrix) {
+ o.matrixApplied = parentMatrix.clone();
+ o.matrixApplied.add(m.a, m.b, m.c, m.d, m.e, m.f);
+ }
+ else {
+ o.matrixApplied = m;
+ }
+
+ this.layerItems.each(function () {
+ FauxNode.prototype.applyTransform.apply(this, [o.matrixApplied]);
+ })
+ },
+
+ getTransformMatrix: function () {
+ return this.matrixApplied;
+ }
+ });
+
+ Element = function (node, paper, group) {
+ var o = this,
+ parent = group || paper;
+
+ o.node = o[0] = node;
+ node.raphael = true;
+ node.raphaelid = o.id = R._oid++;
+ node._rElement = o;
+
+ o.X = 0;
+ o.Y = 0;
+
+ o.attrs = o.attrs || {};
+ o.styles = o.styles || {};
+ o.followers = o.followers || [];
+
+ o.paper = paper;
+ o.com = parent.com;
+
+ o.ca = o.customAttributes = o.customAttributes ||
+ new paper._CustomAttributes();
+
+ o.matrix = R.matrix();
+ o._ = {
+ transform: [],
+ sx: 1,
+ sy: 1,
+ dx: 0,
+ dy: 0,
+ deg: 0
+ };
+
+ o.parent = parent;
+ !parent.bottom && (parent.bottom = o);
+
+ o.prev = parent.top || null;
+ parent.top && (parent.top.next = o);
+ parent.top = o;
+ o.next = null;
+ };
+
+ Element.prototype = elproto;
+ elproto.constructor = Element;
+
+ var
+ repaint = function (el, finalAttrs, positionChanged, dimensionChanged) {
+
+ var node = getNode(el),
+ preC = R._getConnectedNodes(node),
+ attrs = el.attrs,
+ elAbove = preC.above,
+ elBelow = preC.below,
+ len,
+ attr,
+ i;
+
+ for (attr in finalAttrs) {
+ attrs[attr] = finalAttrs[attr];
+ }
+
+ node.redraw();
+ },
+
+ setFillAndStroke = function (el, params) {
+
+ var attrs = el.attrs,
+ node = el.node,
+ finalAttrs = {},
+ att,
+ val,
+ needsRepaint = false,
+ positionChanged = false,
+ dimensionChanged = false,
+ i;
+
+ for (att in params) {
+ if (params[has](att)) {
+ if (!R._availableAttrs[has](att)) {
+ continue;
+ }
+ val = params[att];
+
+ switch (att) {
+
+ case 'fill-opacity':
+ case 'opacity':
+ case 'stroke-opcaity':
+ case 'stroke':
+ case 'fill':
+ finalAttrs[att] = val;
+ needsRepaint = true;
+ break;
+
+ case 'stroke-width':
+ case "cx":
+ case "cy":
+ case "x":
+ case "y":
+ finalAttrs[att] = val;
+ positionChanged = true;
+ break;
+
+ case "width":
+ case "height":
+ finalAttrs[att] = val;
+ dimensionChanged = true;
+ break;
+
+ case "clip-rect":
+ finalAttrs[att] = val;
+ needsRepaint = true;
+ break;
+
+ case "font-size":
+ case "font":
+ case "vertical-align":
+ case "text-anchor":
+ finalAttrs[att] = val;
+ needsRepaint = true;
+
+ default:
+ continue;
+ }
+ }
+ }
+
+ tuneText(el, params, finalAttrs);
+
+ finalAttrs = node.validateAttrs(finalAttrs);
+
+ if (needsRepaint || positionChanged || dimensionChanged) {
+ repaint(el, finalAttrs, positionChanged, dimensionChanged);
+ }
+ },
+
+ leading = 1.2,
+
+ tuneText = function(el, params, finalAttrs) {
+ if (el.type != "text" || !(params[has]("text") || params[has]("font") ||
+ params[has]("font-size") || params[has]("x") || params[has]("y") ||
+ params[has]("line-height") || params[has]("vertical-align"))) {
+ return;
+ }
+
+ var a = el.attr(),
+ fontSize = params['font-size'] || a['font-size'] || 10,
+ lineHeight = toFloat(params['line-height'] || a['line-height']) || toInt(fontSize, 10) * leading,
+ valign = params["vertical-align"] || a["vertical-align"] || "middle";
+
+ if (isNaN(lineHeight)) {
+ lineHeight = fontSize * leading;
+ }
+
+ finalAttrs['font-size'] = toInt(fontSize, 10) + "px";
+ finalAttrs['font'] = params['font'] || a['font'] || 'Verdana';
+ finalAttrs['vertical-align'] = valign;
+ finalAttrs['x'] = params['x'] || a['x'] || 0;
+ finalAttrs['y'] = params['y'] || a['y'] || 0;
+ finalAttrs['line-height'] = toInt(lineHeight, 10);
+ finalAttrs['text-anchor'] = params['text-anchor'] || a['text-anchor'] || 'middle';
+ };
+
+ R._engine.initWin = function (win) {
+ win = win;
+ doc = win.document;
+ };
+
+ R._engine.setSize = function (w, h) {
+ var paper = this,
+ cs = paper.canvas.style;
+
+ cs.width = (paper.width = (+w || paper.width)) + PX;
+ cs.height = (paper.height = (+h || paper.height)) + PX;
+ /* @todo call setViewBox from setSize() */
+ return paper;
+ };
+ /* @todo implement setViewBox() */
+ R._engine.create = function () {
+ var con = R._getContainer.apply(0, arguments) || {},
+ container = con.container,
+ x = con.x,
+ y = con.y,
+ width = con.width,
+ height = con.height,
+ //handler = R._containerEventHandler,
+ wrapper,
+ cssText,
+ image,
+ mmap,
+ i,
+ paper,
+ canvas;
+
+ if (!container) {
+ throw new Error("Canvas container not found.");
+ }
+
+ paper = new R._Paper();
+ paper.canvas = wrapper = $("div");
+
+ x = (x || 0);
+ y = (y || 0);
+ paper.width = width = (width || 512);
+ paper.height = height = (height || 342);
+ paper.left = paper.top = 0;
+
+ if (container == 1) {
+ wrapper.style.cssText = cssText +
+ R.format(";width:100%;height:100%;position:absolute;left:{0}px;top:{1}px;", [x, y]);
+ doc.body.appendChild(wrapper);
+ }
+ else {
+ wrapper.style.cssText = cssText + ";width:100%;height:100%;position:absolute";
+ if (container.firstChild) {
+ container.insertBefore(wrapper, container.firstChild);
+ }
+ else {
+ container.appendChild(wrapper);
+ }
+ }
+
+ cssText = "overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);" +
+ "-webkit-user-select:none;-moz-user-select:-moz-none;" +
+ "-khtml-user-select:none;-ms-user-select:none;user-select:none;" +
+ "-o-user-select:none;cursor:default;" +
+ R.format("width:{0}px;height:{1}px;", [width, height]);
+
+ // Create the canvas element and set it to occupy full space. Retain a
+ // reference to its context.
+ canvas = $("canvas");
+ canvas.style.cssText = "position:absolute;left:0;top:0";
+ canvas.setAttribute('width', paper.width);
+ canvas.setAttribute('height', paper.height);
+
+ paper.com = new CanvasObjectModel(canvas, wrapper, paper.width, paper.height);
+
+ wrapper.appendChild(canvas);
+
+ image = $("img");
+
+ // Easter egg idea! :)
+ image.src = "image1.png";
+
+ image.style.cssText = "opacity: 0;z-index: 100;background: transparent;position: absolute;left: 0;top: 0;width: "+width+"px;height: "+height+"px";
+ wrapper.appendChild(image);
+
+ mmap = $("map");
+
+ mmap.setAttribute("name", "mousemap");
+ mmap.setAttribute("id", "mousemap"); // Needed for FF.
+
+ wrapper.appendChild(mmap);
+
+ image.setAttribute("usemap","#mousemap");
+ wrapper._image = image;
+ wrapper._map = mmap;
+
+ return paper;
+ };
+
+ var getNode = R._engine.getNode = function (el) {
+ return el.node || el[0].node;
+ };
+
+ R._engine.getLastNode = function (el) {
+ return el.node || el[el.length - 1].node;
+ };
+
+ R._engine.rect = function(paper, x, y, w, h, r, group) {
+
+ var node = paper.com.createNode('rect', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.x = x;
+ attrs.y = y;
+ attrs.width = w;
+ attrs.height = h;
+ attrs.fill = "#fff";
+ attrs.stroke = "#000";
+ attrs['stroke-width'] = 1;
+ attrs.r = r || 0;
+ attrs.rx = r || 0;
+ attrs.ry = r || 0;
+
+ el.type = "rect";
+
+ node.render();
+ return el;
+ };
+
+ R._engine.circle = function(paper, x, y, r, group) {
+ var node = paper.com.createNode('circle', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.cx = x;
+ attrs.cy = y;
+ attrs.r = r;
+ attrs.fill = 'none';
+ attrs.stroke = '#000';
+ attrs['stroke-width'] = 1;
+
+ el.type = "circle";
+
+ node.render();
+ return el;
+ };
+
+ R._engine.ellipse = function(paper, x, y, rx, ry, group) {
+ var node = new FauxNode(),
+ el = new Element(node, paper, group);
+
+ el.type = "ellipse";
+ return el;
+ };
+
+ R._engine.image = function(paper, src, x, y, w, h, group) {
+ var node = new FauxNode(),
+ el = new Element(node, paper, group);
+
+ el.type = "image";
+ return el;
+ };
+
+ R._engine.text = function(paper, x, y, text, group) {
+ var node = paper.com.createNode('text', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.x = x;
+ attrs.y = y;
+ attrs.text = text;
+ attrs.fill = 'none';
+ attrs.stroke = '#000';
+ attrs.font = 'Verdana';
+ attrs['font-size'] = '12px';
+ attrs['vertical-align'] = 'middle';
+ attrs['text-anchor'] = 'middle'
+
+ el.type = "text";
+ node.render();
+ return el;
+ };
+
+ R._engine.path = function(pathString, paper, group) {
+ var node = paper.com.createNode('path', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.path = pathString;
+ attrs.fill = "#fff";
+ attrs.stroke = "#000";
+ attrs['stroke-width'] = 1;
+
+ el.type = "path";
+
+ node.render();
+
+ return el;
+ };
+
+ R._engine.group = function (paper, id, group) {
+
+ var node = paper.com.createNode('group', group && group.node),
+ el = new Element(node, paper, group),
+ wrapper = node.wrapper;
+
+ id && wrapper.setAttribute('class', ['red', id].join('-'));
+
+ el.canvas = wrapper;
+
+ //(group && group.canvas.appendChild(wrapper)) || paper.canvas.appendChild(wrapper);
+
+ el.type = "group";
+ return el;
+ };
+
+ elproto._getBBox = function() {
+ if (this.removed) {
+ return {};
+ }
+
+ return {
+ x: this.X + (this.bbx || 0) - this.W / 2,
+ y: this.Y + (this.bby || 0) - this.H / 2,
+ width: this.W,
+ height: this.H
+ };
+ };
+
+ /***** ELEMENT REORDERING / RESTRUCTING *****/
+
+ elproto.toFront = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ parent = o.parent,
+ parentNode = thisNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._tofront(o, parent)) {
+ if (elproto.type === "group") {
+ parent.canvas.appendChild(thisNode);
+ }
+ else {
+ parentNode.nodeList.tofront(thisNode);
+ }
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+
+ return o;
+ };
+
+ elproto.toBack = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ parent = o.parent,
+ parentNode = thisNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._toback(o, parent)) {
+ if (elproto.type === "group") {
+ parent.canvas.appendChild(thisNode);
+ }
+ else {
+ parentNode.nodeList.toback(thisNode);
+ }
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+
+
+ return o;
+ };
+
+ elproto.insertAfter = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ thatNode = element.node,
+ parentNode = thatNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (thatNode.next) {
+ parentNode.nodeList.insertBefore(thisNode, thatNode.next);
+ }
+ else {
+ parentNode.appendChild(thisNode);
+ }
+
+ R._insertafter(o, element, o.parent, element.parent);
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+
+ return o;
+ };
+
+ elproto.insertBefore = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ thatNode = element.node,
+ parentNode = thatNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (thatNode) {
+ parentNode.nodeList.insertBefore(thisNode, thatNode);
+ }
+ else {
+ parentNode.appendChild(thisNode);
+ }
+
+ R._insertafter(o, element, o.parent, element.parent);
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+
+ return o;
+ };
+
+ elproto.appendChild = function(element) {
+ return this;
+ };
+
+ elproto.removeChild = function(element) {
+ return this;
+ };
+
+ /***** ELEMENT REORDERING / RESTRUCTING *****/
+
+
+
+ elproto.attr = function(name, value) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+
+ attrs = o.attrs,
+ ca = o.ca,
+ names,
+ params,
+ par,
+
+ res,
+ key,
+ out,
+
+ subkey,
+ delkeys,
+
+ follower,
+ ii,
+ i;
+
+ // fetch a copy of all attributes
+ if (name == null) {
+ res = {};
+ for (key in attrs) if (attrs.hasOwnProperty(key)) {
+ res[key] = attrs[key];
+ }
+ res.gradient && res.fill == "none" && (res.fill = res.gradient) && delete res.gradient;
+ res.transform = o._.transform;
+ /* @todo res.visibility = o.node.style.display === "none" ? "hidden" : "visible"; */
+ return res;
+ }
+
+ // fetch a single value
+ if (value == null && R.is(name, "string")) {
+ if (name == "fill" && attrs.fill == "none" && attrs.gradient) {
+ return attrs.gradient;
+ }
+ if (name == "transform") {
+ return o._.transform;
+ }
+ /* @todo if (name == "visibility") {
+ return this.node.style.display === "none" ? "hidden" : "visible";
+ }*/
+
+ names = name.split(separator),
+ out = {};
+
+ for (i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in attrs) {
+ out[name] = attrs[name];
+ }
+ else if (R.is(ca[name], "function")) {
+ out[name] = ca[name].def;
+ } else {
+ out[name] = R._availableAttrs[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+
+ // fetch specific attributes
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = o.attr(name[i]);
+ }
+ return out;
+ }
+
+ // prepare setter params
+ if (value != null) {
+ params = {};
+ params[name] = value;
+ }
+ else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+
+ for (key in params) {
+ eve("raphael.attr." + key + "." + o.id, o, params[key], key);
+ }
+
+ delkeys = {};
+ for (key in ca) {
+
+ if (ca[key] && params.hasOwnProperty(key) &&
+ R.is(ca[key], "function") && !ca['_invoked' + key]) {
+
+ ca['_invoked'+key] = true; // prevent recursion
+ par = ca[key].apply(o, [].concat(params[key]));
+ delete ca['_invoked'+key];
+
+ for (subkey in par) {
+ if (par.hasOwnProperty(subkey)) {
+ params[subkey] = par[subkey];
+ }
+ }
+ attrs[key] = params[key];
+ if (par === false) {
+ delkeys[key] = params[key];
+ delete params[key];
+ }
+ }
+ }
+
+ setFillAndStroke(this, params);
+
+ for (i = 0, ii = o.followers.length; i < ii; i++) {
+ follower = o.followers[i];
+ (follower.cb && !follower.cb.call(follower.el, params, o)) ||
+ follower.el.attr(params);
+ }
+
+ for (subkey in delkeys) {
+ params[subkey] = delkeys[subkey];
+ }
+ return this;
+ };
+
+ elproto.css = function (name, value) {
+ return this;
+ };
+
+
+
+
+ /**************** Drag *********************/
+ elproto.drag = function (onmove, onstart, onend, move_scope, start_scope, end_scope) {
+
+ function start(e) {
+ (e.originalEvent || e).preventDefault();
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft;
+ this._drag.x = e.clientX + scrollX;
+ this._drag.y = e.clientY + scrollY;
+ this._drag.id = e.identifier;
+ !drag.length && R.mousemove(dragMove).mouseup(dragUp);
+ drag.push({
+ el: this,
+ move_scope: move_scope,
+ start_scope: start_scope,
+ end_scope: end_scope
+ });
+ onstart && eve.on("raphael.drag.start." + this.id, onstart);
+ onmove && eve.on("raphael.drag.move." + this.id, onmove);
+ onend && eve.on("raphael.drag.end." + this.id, onend);
+ eve("raphael.drag.start." + this.id, start_scope || move_scope || this, e.clientX + scrollX, e.clientY + scrollY, e);
+ }
+ this._drag = {};
+ draggable.push({
+ el: this,
+ start: start
+ });
+ this.mousedown(start);
+ return this;
+ }
+
+ elproto.undrag = function() {
+ var i = draggable.length;
+ while (i--)
+ if (draggable[i].el == this) {
+ this.unmousedown(draggable[i].start);
+ draggable.splice(i, 1);
+ eve.unbind("raphael.drag.*." + this.id);
+ }
+ !draggable.length && R.unmousemove(dragMove).unmouseup(dragUp);
+ };
+
+
+ /***************** Drag *****************/
+ /************ TRANSFORMATIONS *************/
+
+
+
+ elproto.rotate = function(deg, cx, cy) {
+
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ deg = Str(deg).split(separator);
+ if (deg.length - 1) {
+ cx = toFloat(deg[1]);
+ cy = toFloat(deg[2]);
+ }
+ deg = toFloat(deg[0]);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ cx = bbox.x + bbox.width / 2;
+ cy = bbox.y + bbox.height / 2;
+ }
+ o.transform(o._.transform.concat([["r", deg, cx, cy]]));
+ return o;
+ };
+
+ elproto.scale = function(sx, sy, cx, cy) {
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ sx = Str(sx).split(separator);
+ if (sx.length - 1) {
+ sy = toFloat(sx[1]);
+ cx = toFloat(sx[2]);
+ cy = toFloat(sx[3]);
+ }
+ sx = toFloat(sx[0]);
+ (sy == null) && (sy = sx);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ }
+ cx = cx == null ? bbox.x + bbox.width / 2 : cx;
+ cy = cy == null ? bbox.y + bbox.height / 2 : cy;
+ o.transform(o._.transform.concat([["s", sx, sy, cx, cy]]));
+
+ return o;
+ };
+
+ elproto.translate = function(dx, dy) {
+ var o = this;
+ if (o.removed) {
+ return o;
+ }
+ dx = Str(dx).split(separator);
+ if (dx.length - 1) {
+ dy = toFloat(dx[1]);
+ }
+ dx = toFloat(dx[0]) || 0;
+ dy = +dy || 0;
+ o.transform(o._.transform.concat([["t", dx, dy]]));
+
+ return o;
+ };
+
+ elproto.transform = function(tstr) {
+
+ var o = this,
+ _ = o._,
+ sw;
+
+ if (tstr === null) {
+ return _.transform;
+ }
+
+ R._extractTransform(o, tstr);
+
+ /* @todo: what changes to be made here in the context of canvas */
+ /*o.clip && !_.clipispath && $(o.clip, {
+ transform: o.matrix.invert()
+ });
+ o.pattern && updatePosition(o); */
+ if (_.sx != 1 || _.sy != 1) {
+ sw = o.attrs[has]("stroke-width") ? o.attrs["stroke-width"] : 1;
+ o.attr({
+ "stroke-width": sw
+ });
+ }
+
+ o.node && o.node.redraw();
+
+ return o;
+ };
+
+ /************ TRANSFORMATIONS *************/
+
+
+ elproto.hide = function() {
+ return this;
+ };
+
+ elproto.show = function() {
+ return this;
+ };
+
+ elproto.blur = function(size) {
+ return this;
+ };
+
+ elproto.on = function(eventType, handler) {
+ var el = this,
+ listeners = el.listeners;
+
+ if (!listeners) {
+ listeners = el.listeners = {};
+ }
+
+ if (!listeners[eventType]) {
+ listeners[eventType] = [];
+ }
+
+ listeners[eventType].push(handler);
+ };
+
+ elproto.remove = function() {
+ return this;
+ };
+
+
+
+
+ paperproto.clear = function () {
+ eve("raphael.clear", this);
+ return this;
+ };
+
+ paperproto.remove = function () {
+ if (this.removed) {
+ return;
+ }
+
+ var paper = this,
+ canvas = paper.canvas,
+ pn = canvas.parentNode,
+ i;
+
+ eve("raphael.remove", paper);
+ pn.removeChild(canvas);
+
+ for (i in paper) {
+ paper[i] = typeof paper[i] == "function" ? R._removedFactory(i) : null;
+ }
+
+ this.removed = true;
+ };
+
+ R.toString = function () {
+ return "Your browser supports canvas.\nYou are running RedRaphael " +
+ R.version;
+ };
+
+ for (var method in elproto) {
+ if (elproto.hasOwnProperty(method) &&
+ !setproto.hasOwnProperty(method)) {
+ setproto[method] = (function (methodname) {
+ return function () {
+ var arg = arguments;
+ return this.forEach(function (el) {
+ el[methodname].apply(el, arg);
+ });
+ };
+ })(method);
+ }
+ };
+
+})();
+
+
+
+ // EXPOSE
+ // SVG and VML are appended just before the EXPOSE line
+ // Even with AMD, Raphael should be defined globally
+ oldRaphael.was ? (g.win.Raphael = R) : (Raphael = R);
+
+ return R;
+}));
\ No newline at end of file
diff --git a/source/Gruntfile.js b/source/Gruntfile.js
new file mode 100644
index 0000000..804f984
--- /dev/null
+++ b/source/Gruntfile.js
@@ -0,0 +1,109 @@
+
+"use strict";
+
+module.exports = function(grunt) {
+ var pkg = grunt.file.readJSON('package.json');
+
+ // Project configuration.
+ grunt.initConfig({
+ // Metadata.
+ pkg: pkg,
+ banner: grunt.file.read("copy.js").replace(/@VERSION/, pkg.version).replace(/@RVERSION/, pkg.rversion),
+ // Task configuration.
+ uglify: {
+ options: {
+ banner: "<%= banner %>"
+ },
+ dist: {
+ src: "<%= build.dist.dest %>",
+ dest: "package/<%= pkg.filename %>-min.js"
+ }
+ },
+ build: {
+ options: {
+ banner: "<%= banner %>"
+ },
+ dist: {
+ dest: "package/raphael.js",
+ src: [
+ "source/eve/eve.js",
+ "source/raphael.core.js",
+ "source/raphael.svg.js",
+ "source/raphael.vml.js",
+ "source/raphael.canvas.js"
+ ]
+ }
+ },
+ jasmine: {
+ pivotal: {
+ src: 'package/raphael-min.js',
+ options: {
+ specs: 'tests/*Spec.js',
+ helpers: 'tests/*Helper.js'
+ }
+ }
+ }
+ });
+
+
+ // These plugins provide necessary tasks.
+ grunt.loadNpmTasks("grunt-contrib-uglify");
+ grunt.loadNpmTasks("grunt-contrib-jasmine");
+
+ // Special concat/build task to handle RedRaphael's build requirements
+ grunt.registerMultiTask(
+ "build",
+ "Concatenate source, remove individual closures, embed version",
+ function() {
+ var data = this.data,
+ name = data.dest,
+ src = data.src,
+ options = this.options({
+ banner: ""
+ }),
+ // Start with banner
+ compiled = options.banner,
+ svgorvmlRegex = /\.(svg|vml|canvas)\.js/,
+ closureRegex = /window\.Raphael.*\(R\)\s*\{/,
+ closureEndRegex = /\}\(window\.Raphael\);\s*$/,
+ exposeRegex = /(\r?\n\s*\/\/\s*EXPOSE(?:\r|\n|.)*\}\)\);)/;
+
+ // Concatenate src
+ src.forEach(function(path) {
+ var source = grunt.file.read(path);
+ var match = svgorvmlRegex.exec(path);
+
+ // If either SVG or VML,
+ // remove the closure and add an early return if not required
+ if (match) {
+ source = "\n\n" +
+ source.replace(closureRegex,
+ "(function(){\n" +
+ " if (!R." + match[1] + ") {\n" +
+ " return;\n" +
+ " }"
+ )
+ .replace( closureEndRegex, "})();" );
+
+ // Add source before EXPOSE line
+ compiled.replace(exposeRegex, function () {
+ // Using this method instead of the raphael way as it makes it difficult to
+ // to have the string '$1' in the target file like raphael.svg.js.
+ var text = arguments[3],
+ index = arguments[2];
+
+ compiled = (text.slice(0, index) + source + text.slice(index));
+ });
+
+ } else {
+ compiled += source;
+ }
+ });
+
+ grunt.file.write( name, compiled );
+ }
+ );
+
+ // Default task.
+ grunt.registerTask("default", ["build", "uglify", "jasmine"]);
+};
\ No newline at end of file
diff --git a/source/copy.js b/source/copy.js
new file mode 100644
index 0000000..456357a
--- /dev/null
+++ b/source/copy.js
@@ -0,0 +1,10 @@
+/**!
+ * RedRaphael @VERSION - JavaScript Vector Library
+ * Copyright (c) 2012-2013 FusionCharts Technologies
+ *
+ * Raphael @RVERSION
+ * Copyright (c) 2008-2012 Dmitry Baranovskiy
+ * Copyright © 2008-2012 Sencha Labs
+ *
+ * Licensed under the MIT license.
+ */
diff --git a/source/eve/LICENSE b/source/eve/LICENSE
new file mode 100644
index 0000000..86a96fa
--- /dev/null
+++ b/source/eve/LICENSE
@@ -0,0 +1,202 @@
+
+ Apache License
+ Version 2.0, January 2004
+ http://www.apache.org/licenses/
+
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
+
+ 1. Definitions.
+
+ "License" shall mean the terms and conditions for use, reproduction,
+ and distribution as defined by Sections 1 through 9 of this document.
+
+ "Licensor" shall mean the copyright owner or entity authorized by
+ the copyright owner that is granting the License.
+
+ "Legal Entity" shall mean the union of the acting entity and all
+ other entities that control, are controlled by, or are under common
+ control with that entity. For the purposes of this definition,
+ "control" means (i) the power, direct or indirect, to cause the
+ direction or management of such entity, whether by contract or
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
+ outstanding shares, or (iii) beneficial ownership of such entity.
+
+ "You" (or "Your") shall mean an individual or Legal Entity
+ exercising permissions granted by this License.
+
+ "Source" form shall mean the preferred form for making modifications,
+ including but not limited to software source code, documentation
+ source, and configuration files.
+
+ "Object" form shall mean any form resulting from mechanical
+ transformation or translation of a Source form, including but
+ not limited to compiled object code, generated documentation,
+ and conversions to other media types.
+
+ "Work" shall mean the work of authorship, whether in Source or
+ Object form, made available under the License, as indicated by a
+ copyright notice that is included in or attached to the work
+ (an example is provided in the Appendix below).
+
+ "Derivative Works" shall mean any work, whether in Source or Object
+ form, that is based on (or derived from) the Work and for which the
+ editorial revisions, annotations, elaborations, or other modifications
+ represent, as a whole, an original work of authorship. For the purposes
+ of this License, Derivative Works shall not include works that remain
+ separable from, or merely link (or bind by name) to the interfaces of,
+ the Work and Derivative Works thereof.
+
+ "Contribution" shall mean any work of authorship, including
+ the original version of the Work and any modifications or additions
+ to that Work or Derivative Works thereof, that is intentionally
+ submitted to Licensor for inclusion in the Work by the copyright owner
+ or by an individual or Legal Entity authorized to submit on behalf of
+ the copyright owner. For the purposes of this definition, "submitted"
+ means any form of electronic, verbal, or written communication sent
+ to the Licensor or its representatives, including but not limited to
+ communication on electronic mailing lists, source code control systems,
+ and issue tracking systems that are managed by, or on behalf of, the
+ Licensor for the purpose of discussing and improving the Work, but
+ excluding communication that is conspicuously marked or otherwise
+ designated in writing by the copyright owner as "Not a Contribution."
+
+ "Contributor" shall mean Licensor and any individual or Legal Entity
+ on behalf of whom a Contribution has been received by Licensor and
+ subsequently incorporated within the Work.
+
+ 2. Grant of Copyright License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ copyright license to reproduce, prepare Derivative Works of,
+ publicly display, publicly perform, sublicense, and distribute the
+ Work and such Derivative Works in Source or Object form.
+
+ 3. Grant of Patent License. Subject to the terms and conditions of
+ this License, each Contributor hereby grants to You a perpetual,
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
+ (except as stated in this section) patent license to make, have made,
+ use, offer to sell, sell, import, and otherwise transfer the Work,
+ where such license applies only to those patent claims licensable
+ by such Contributor that are necessarily infringed by their
+ Contribution(s) alone or by combination of their Contribution(s)
+ with the Work to which such Contribution(s) was submitted. If You
+ institute patent litigation against any entity (including a
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
+ or a Contribution incorporated within the Work constitutes direct
+ or contributory patent infringement, then any patent licenses
+ granted to You under this License for that Work shall terminate
+ as of the date such litigation is filed.
+
+ 4. Redistribution. You may reproduce and distribute copies of the
+ Work or Derivative Works thereof in any medium, with or without
+ modifications, and in Source or Object form, provided that You
+ meet the following conditions:
+
+ (a) You must give any other recipients of the Work or
+ Derivative Works a copy of this License; and
+
+ (b) You must cause any modified files to carry prominent notices
+ stating that You changed the files; and
+
+ (c) You must retain, in the Source form of any Derivative Works
+ that You distribute, all copyright, patent, trademark, and
+ attribution notices from the Source form of the Work,
+ excluding those notices that do not pertain to any part of
+ the Derivative Works; and
+
+ (d) If the Work includes a "NOTICE" text file as part of its
+ distribution, then any Derivative Works that You distribute must
+ include a readable copy of the attribution notices contained
+ within such NOTICE file, excluding those notices that do not
+ pertain to any part of the Derivative Works, in at least one
+ of the following places: within a NOTICE text file distributed
+ as part of the Derivative Works; within the Source form or
+ documentation, if provided along with the Derivative Works; or,
+ within a display generated by the Derivative Works, if and
+ wherever such third-party notices normally appear. The contents
+ of the NOTICE file are for informational purposes only and
+ do not modify the License. You may add Your own attribution
+ notices within Derivative Works that You distribute, alongside
+ or as an addendum to the NOTICE text from the Work, provided
+ that such additional attribution notices cannot be construed
+ as modifying the License.
+
+ You may add Your own copyright statement to Your modifications and
+ may provide additional or different license terms and conditions
+ for use, reproduction, or distribution of Your modifications, or
+ for any such Derivative Works as a whole, provided Your use,
+ reproduction, and distribution of the Work otherwise complies with
+ the conditions stated in this License.
+
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
+ any Contribution intentionally submitted for inclusion in the Work
+ by You to the Licensor shall be under the terms and conditions of
+ this License, without any additional terms or conditions.
+ Notwithstanding the above, nothing herein shall supersede or modify
+ the terms of any separate license agreement you may have executed
+ with Licensor regarding such Contributions.
+
+ 6. Trademarks. This License does not grant permission to use the trade
+ names, trademarks, service marks, or product names of the Licensor,
+ except as required for reasonable and customary use in describing the
+ origin of the Work and reproducing the content of the NOTICE file.
+
+ 7. Disclaimer of Warranty. Unless required by applicable law or
+ agreed to in writing, Licensor provides the Work (and each
+ Contributor provides its Contributions) on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
+ implied, including, without limitation, any warranties or conditions
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
+ PARTICULAR PURPOSE. You are solely responsible for determining the
+ appropriateness of using or redistributing the Work and assume any
+ risks associated with Your exercise of permissions under this License.
+
+ 8. Limitation of Liability. In no event and under no legal theory,
+ whether in tort (including negligence), contract, or otherwise,
+ unless required by applicable law (such as deliberate and grossly
+ negligent acts) or agreed to in writing, shall any Contributor be
+ liable to You for damages, including any direct, indirect, special,
+ incidental, or consequential damages of any character arising as a
+ result of this License or out of the use or inability to use the
+ Work (including but not limited to damages for loss of goodwill,
+ work stoppage, computer failure or malfunction, or any and all
+ other commercial damages or losses), even if such Contributor
+ has been advised of the possibility of such damages.
+
+ 9. Accepting Warranty or Additional Liability. While redistributing
+ the Work or Derivative Works thereof, You may choose to offer,
+ and charge a fee for, acceptance of support, warranty, indemnity,
+ or other liability obligations and/or rights consistent with this
+ License. However, in accepting such obligations, You may act only
+ on Your own behalf and on Your sole responsibility, not on behalf
+ of any other Contributor, and only if You agree to indemnify,
+ defend, and hold each Contributor harmless for any liability
+ incurred by, or claims asserted against, such Contributor by reason
+ of your accepting any such warranty or additional liability.
+
+ END OF TERMS AND CONDITIONS
+
+ APPENDIX: How to apply the Apache License to your work.
+
+ To apply the Apache License to your work, attach the following
+ boilerplate notice, with the fields enclosed by brackets "[]"
+ replaced with your own identifying information. (Don't include
+ the brackets!) The text should be enclosed in the appropriate
+ comment syntax for the file format. We also recommend that a
+ file or class name and description of purpose be included on the
+ same "printed page" as the copyright notice for easier
+ identification within third-party archives.
+
+ Copyright 2013 Adobe Systems Incorporated
+
+ Licensed under the Apache License, Version 2.0 (the "License");
+ you may not use this file except in compliance with the License.
+ You may obtain a copy of the License at
+
+ http://www.apache.org/licenses/LICENSE-2.0
+
+ Unless required by applicable law or agreed to in writing, software
+ distributed under the License is distributed on an "AS IS" BASIS,
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ See the License for the specific language governing permissions and
+ limitations under the License.
\ No newline at end of file
diff --git a/source/eve/README.md b/source/eve/README.md
new file mode 100644
index 0000000..c129612
--- /dev/null
+++ b/source/eve/README.md
@@ -0,0 +1,7 @@
+# Eve
+
+Tiny event helping JavaScript library.
+
+MIT Licence
+
+For use case look at e.html
diff --git a/source/eve/component.json b/source/eve/component.json
new file mode 100644
index 0000000..651815f
--- /dev/null
+++ b/source/eve/component.json
@@ -0,0 +1,13 @@
+{
+ "name": "eve",
+ "repo": "DmitryBaranovskiy/eve",
+ "description": "Custom Events",
+ "version": "0.4.1",
+ "keywords": ["events"],
+ "dependencies": {},
+ "development": {},
+ "main": "eve.js",
+ "scripts": [
+ "eve.js"
+ ]
+}
\ No newline at end of file
diff --git a/source/eve/e.html b/source/eve/e.html
new file mode 100644
index 0000000..6687278
--- /dev/null
+++ b/source/eve/e.html
@@ -0,0 +1,66 @@
+
+
+
+ Eve Use Case
+
+
+
+
+
diff --git a/source/eve/eve.js b/source/eve/eve.js
new file mode 100644
index 0000000..23f71b9
--- /dev/null
+++ b/source/eve/eve.js
@@ -0,0 +1,371 @@
+// Copyright (c) 2013 Adobe Systems Incorporated. All rights reserved.
+//
+// Licensed under the Apache License, Version 2.0 (the "License");
+// you may not use this file except in compliance with the License.
+// You may obtain a copy of the License at
+//
+// http://www.apache.org/licenses/LICENSE-2.0
+//
+// Unless required by applicable law or agreed to in writing, software
+// distributed under the License is distributed on an "AS IS" BASIS,
+// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+// See the License for the specific language governing permissions and
+// limitations under the License.
+// ┌────────────────────────────────────────────────────────────┐ \\
+// │ Eve 0.4.2 - JavaScript Events Library │ \\
+// ├────────────────────────────────────────────────────────────┤ \\
+// │ Author Dmitry Baranovskiy (http://dmitry.baranovskiy.com/) │ \\
+// └────────────────────────────────────────────────────────────┘ \\
+
+(function (glob) {
+ var version = "0.4.2",
+ has = "hasOwnProperty",
+ separator = /[\.\/]/,
+ wildcard = "*",
+ fun = function () {},
+ numsort = function (a, b) {
+ return a - b;
+ },
+ current_event,
+ stop,
+ events = {n: {}},
+ /*\
+ * eve
+ [ method ]
+
+ * Fires event with given `name`, given scope and other parameters.
+
+ > Arguments
+
+ - name (string) name of the *event*, dot (`.`) or slash (`/`) separated
+ - scope (object) context for the event handlers
+ - varargs (...) the rest of arguments will be sent to event handlers
+
+ = (object) array of returned values from the listeners
+ \*/
+ eve = function (name, scope) {
+ name = String(name);
+ var e = events,
+ oldstop = stop,
+ args = Array.prototype.slice.call(arguments, 2),
+ listeners = eve.listeners(name),
+ z = 0,
+ f = false,
+ l,
+ indexed = [],
+ queue = {},
+ out = [],
+ ce = current_event,
+ errors = [];
+ current_event = name;
+ stop = 0;
+ for (var i = 0, ii = listeners.length; i < ii; i++) if ("zIndex" in listeners[i]) {
+ indexed.push(listeners[i].zIndex);
+ if (listeners[i].zIndex < 0) {
+ queue[listeners[i].zIndex] = listeners[i];
+ }
+ }
+ indexed.sort(numsort);
+ while (indexed[z] < 0) {
+ l = queue[indexed[z++]];
+ out.push(l.apply(scope, args));
+ if (stop) {
+ stop = oldstop;
+ return out;
+ }
+ }
+ for (i = 0; i < ii; i++) {
+ l = listeners[i];
+ if ("zIndex" in l) {
+ if (l.zIndex == indexed[z]) {
+ out.push(l.apply(scope, args));
+ if (stop) {
+ break;
+ }
+ do {
+ z++;
+ l = queue[indexed[z]];
+ l && out.push(l.apply(scope, args));
+ if (stop) {
+ break;
+ }
+ } while (l)
+ } else {
+ queue[l.zIndex] = l;
+ }
+ } else {
+ out.push(l.apply(scope, args));
+ if (stop) {
+ break;
+ }
+ }
+ }
+ stop = oldstop;
+ current_event = ce;
+ return out.length ? out : null;
+ };
+ // Undocumented. Debug only.
+ eve._events = events;
+ /*\
+ * eve.listeners
+ [ method ]
+
+ * Internal method which gives you array of all event handlers that will be triggered by the given `name`.
+
+ > Arguments
+
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated
+
+ = (array) array of event handlers
+ \*/
+ eve.listeners = function (name) {
+ var names = name.split(separator),
+ e = events,
+ item,
+ items,
+ k,
+ i,
+ ii,
+ j,
+ jj,
+ nes,
+ es = [e],
+ out = [];
+ for (i = 0, ii = names.length; i < ii; i++) {
+ nes = [];
+ for (j = 0, jj = es.length; j < jj; j++) {
+ e = es[j].n;
+ items = [e[names[i]], e[wildcard]];
+ k = 2;
+ while (k--) {
+ item = items[k];
+ if (item) {
+ nes.push(item);
+ out = out.concat(item.f || []);
+ }
+ }
+ }
+ es = nes;
+ }
+ return out;
+ };
+
+ /*\
+ * eve.on
+ [ method ]
+ **
+ * Binds given event handler with a given name. You can use wildcards “`*`” for the names:
+ | eve.on("*.under.*", f);
+ | eve("mouse.under.floor"); // triggers f
+ * Use @eve to trigger the listener.
+ **
+ > Arguments
+ **
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated, with optional wildcards
+ - f (function) event handler function
+ **
+ = (function) returned function accepts a single numeric parameter that represents z-index of the handler. It is an optional feature and only used when you need to ensure that some subset of handlers will be invoked in a given order, despite of the order of assignment.
+ > Example:
+ | eve.on("mouse", eatIt)(2);
+ | eve.on("mouse", scream);
+ | eve.on("mouse", catchIt)(1);
+ * This will ensure that `catchIt()` function will be called before `eatIt()`.
+ *
+ * If you want to put your handler before non-indexed handlers, specify a negative value.
+ * Note: I assume most of the time you don’t need to worry about z-index, but it’s nice to have this feature “just in case”.
+ \*/
+ eve.on = function (name, f) {
+ name = String(name);
+ if (typeof f != "function") {
+ return function () {};
+ }
+ var names = name.split(separator),
+ e = events;
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ e = e.n;
+ e = e.hasOwnProperty(names[i]) && e[names[i]] || (e[names[i]] = {n: {}});
+ }
+ e.f = e.f || [];
+ for (i = 0, ii = e.f.length; i < ii; i++) if (e.f[i] == f) {
+ return fun;
+ }
+ e.f.push(f);
+ return function (zIndex) {
+ if (+zIndex == +zIndex) {
+ f.zIndex = +zIndex;
+ }
+ };
+ };
+ /*\
+ * eve.f
+ [ method ]
+ **
+ * Returns function that will fire given event with optional arguments.
+ * Arguments that will be passed to the result function will be also
+ * concated to the list of final arguments.
+ | el.onclick = eve.f("click", 1, 2);
+ | eve.on("click", function (a, b, c) {
+ | console.log(a, b, c); // 1, 2, [event object]
+ | });
+ > Arguments
+ - event (string) event name
+ - varargs (…) and any other arguments
+ = (function) possible event handler function
+ \*/
+ eve.f = function (event) {
+ var attrs = [].slice.call(arguments, 1);
+ return function () {
+ eve.apply(null, [event, null].concat(attrs).concat([].slice.call(arguments, 0)));
+ };
+ };
+ /*\
+ * eve.stop
+ [ method ]
+ **
+ * Is used inside an event handler to stop the event, preventing any subsequent listeners from firing.
+ \*/
+ eve.stop = function () {
+ stop = 1;
+ };
+ /*\
+ * eve.nt
+ [ method ]
+ **
+ * Could be used inside event handler to figure out actual name of the event.
+ **
+ > Arguments
+ **
+ - subname (string) #optional subname of the event
+ **
+ = (string) name of the event, if `subname` is not specified
+ * or
+ = (boolean) `true`, if current event’s name contains `subname`
+ \*/
+ eve.nt = function (subname) {
+ if (subname) {
+ return new RegExp("(?:\\.|\\/|^)" + subname + "(?:\\.|\\/|$)").test(current_event);
+ }
+ return current_event;
+ };
+ /*\
+ * eve.nts
+ [ method ]
+ **
+ * Could be used inside event handler to figure out actual name of the event.
+ **
+ **
+ = (array) names of the event
+ \*/
+ eve.nts = function () {
+ return current_event.split(separator);
+ };
+ /*\
+ * eve.off
+ [ method ]
+ **
+ * Removes given function from the list of event listeners assigned to given name.
+ * If no arguments specified all the events will be cleared.
+ **
+ > Arguments
+ **
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated, with optional wildcards
+ - f (function) event handler function
+ \*/
+ /*\
+ * eve.unbind
+ [ method ]
+ **
+ * See @eve.off
+ \*/
+ eve.off = eve.unbind = function (name, f) {
+ if (!name) {
+ eve._events = events = {n: {}};
+ return;
+ }
+ var names = name.split(separator),
+ e,
+ key,
+ splice,
+ i, ii, j, jj,
+ cur = [events];
+ for (i = 0, ii = names.length; i < ii; i++) {
+ for (j = 0; j < cur.length; j += splice.length - 2) {
+ splice = [j, 1];
+ e = cur[j].n;
+ if (names[i] != wildcard) {
+ if (e[names[i]]) {
+ splice.push(e[names[i]]);
+ }
+ } else {
+ for (key in e) if (e[has](key)) {
+ splice.push(e[key]);
+ }
+ }
+ cur.splice.apply(cur, splice);
+ }
+ }
+ for (i = 0, ii = cur.length; i < ii; i++) {
+ e = cur[i];
+ while (e.n) {
+ if (f) {
+ if (e.f) {
+ for (j = 0, jj = e.f.length; j < jj; j++) if (e.f[j] == f) {
+ e.f.splice(j, 1);
+ break;
+ }
+ !e.f.length && delete e.f;
+ }
+ for (key in e.n) if (e.n[has](key) && e.n[key].f) {
+ var funcs = e.n[key].f;
+ for (j = 0, jj = funcs.length; j < jj; j++) if (funcs[j] == f) {
+ funcs.splice(j, 1);
+ break;
+ }
+ !funcs.length && delete e.n[key].f;
+ }
+ } else {
+ delete e.f;
+ for (key in e.n) if (e.n[has](key) && e.n[key].f) {
+ delete e.n[key].f;
+ }
+ }
+ e = e.n;
+ }
+ }
+ };
+ /*\
+ * eve.once
+ [ method ]
+ **
+ * Binds given event handler with a given name to only run once then unbind itself.
+ | eve.once("login", f);
+ | eve("login"); // triggers f
+ | eve("login"); // no listeners
+ * Use @eve to trigger the listener.
+ **
+ > Arguments
+ **
+ - name (string) name of the event, dot (`.`) or slash (`/`) separated, with optional wildcards
+ - f (function) event handler function
+ **
+ = (function) same return function as @eve.on
+ \*/
+ eve.once = function (name, f) {
+ var f2 = function () {
+ eve.unbind(name, f2);
+ return f.apply(this, arguments);
+ };
+ return eve.on(name, f2);
+ };
+ /*\
+ * eve.version
+ [ property (string) ]
+ **
+ * Current version of the library.
+ \*/
+ eve.version = version;
+ eve.toString = function () {
+ return "You are running Eve " + version;
+ };
+ (typeof module != "undefined" && module.exports) ? (module.exports = eve) : (typeof define != "undefined" ? (define("eve", [], function() { return eve; })) : (glob.eve = eve));
+})(this);
diff --git a/source/eve/package.json b/source/eve/package.json
new file mode 100644
index 0000000..8fb1bb9
--- /dev/null
+++ b/source/eve/package.json
@@ -0,0 +1,18 @@
+{
+ "name" : "eve",
+
+ "author" : {
+ "name" : "Dmitry Baranovskiy",
+ "email" : "dmitry@baranovskiy.com",
+ "url" : "http://dmitry.baranovskiy.com"
+ },
+ "description" : "Simple custom events",
+ "version" : "0.4.1",
+
+ "main" : "./eve.js",
+
+ "repository": {
+ "type": "git",
+ "url": "git@github.com:adobe/eve.git"
+ }
+}
diff --git a/source/make b/source/make
new file mode 100755
index 0000000..8c20e5f
--- /dev/null
+++ b/source/make
@@ -0,0 +1,59 @@
+#!/usr/bin/env node
+var ujs = require('uglify-js'),
+ fs = require('fs'),
+
+ noticeText = function() {
+/**!
+ * RedRaphael 1.0.0 - JavaScript Vector Library
+ * Copyright (c) 2012-2013 FusionCharts Technologies
+ *
+ * Raphael 2.1.0
+ * Copyright (c) 2008-2012 Dmitry Baranovskiy
+ * Copyright © 2008-2012 Sencha Labs
+ *
+ * Licensed under the MIT license.
+ */
+ }.toString().slice(14,-3),
+
+ input = {
+ core : 'raphael.core.js',
+ svg : 'raphael.svg.js',
+ vml : 'raphael.vml.js',
+ canvas: 'raphael.canvas.js',
+ eve : 'eve/eve.js'
+ },
+ outPath = "../package/",
+ output = {
+ 'raphael-min.js' : ['eve', 'core', 'svg', 'vml', 'canvas'],
+ 'raphael.js' : ['eve', 'core', 'svg', 'vml', 'canvas']
+ };
+
+for (var file in input) {
+ input[file] = fs.readFileSync(input[file], 'utf8');
+}
+for (file in output) {
+ var out = '';
+ if (file.indexOf('min') !== -1) {
+ console.log('Compressing ' + file);
+ for (var i = 0, l = output[file].length; i < l; i++) {
+ var o = ujs.minify(input[output[file][i]], {
+ fromString : true
+ });
+ out += o.code;
+ }
+ } else {
+ console.log('Concatinating ' + file);
+ for (i = 0, l = output[file].length; i < l; i++) {
+ out += input[output[file][i]] + '\n';
+ }
+ }
+ (function(f, code){
+ fs.writeFile(f, noticeText + '\n' + code, function(err) {
+ if (err) {
+ console.log(err);
+ } else {
+ console.log('Saved to \033[32m' + f + '\033[0m\n');
+ }
+ });
+ })(outPath + file, out);
+}
diff --git a/source/raphael.canvas.js b/source/raphael.canvas.js
new file mode 100644
index 0000000..0a0fc8c
--- /dev/null
+++ b/source/raphael.canvas.js
@@ -0,0 +1,2734 @@
+/*jslint forin: true, regexp: true, todo: true, white: false, browser: true,
+ sloppy: true, white: true, eqeq: false, newcap: true, nomen: true */
+
+/*global FusionCharts */
+
+/**
+ * Raphael Canvas Extension
+ */
+
+window.Raphael && window.Raphael.canvas && function (R) {
+ var win = R._g.win,
+ doc = R._g.doc,
+ g = R._g,
+
+ STRING = 'string',
+ PX = 'px',
+
+ separator = /[, ]+/,
+
+ Str = win.String,
+ toInt = win.parseInt,
+ toFloat = win.parseFloat,
+
+ math = win.Math,
+ mmax = math.max,
+ mmin = math.min,
+ pi = math.PI,
+ mathFloor = math.floor,
+
+ eve = R.eve,
+ paperproto = R.fn,
+ elproto = R.el,
+ setproto = R.st,
+
+ clone = R.clone,
+ deg2rad = pi / 180,
+ rad2deg = 180 / pi,
+
+ DEFAULT_FILL = "#fff",
+ DEFAULT_STROKE = "#000",
+ has = "hasOwnProperty",
+ S = " ",
+ /* @todo: detect touch */
+ supportsTouch = (('ontouchstart' in win) || (navigator.msMaxTouchPoints > 0)),
+ events = ("click dblclick mousedown mousemove mouseout mouseover mouseup touchstart touchmove touchend touchcancel").split(S),
+ noHandle = false,
+
+ $,
+ FauxNode,
+ Element,
+ draggable = [],
+ drag = [],
+ dragMove = function(e) {
+ var x = e.clientX,
+ y = e.clientY,
+ scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft,
+ dragi,
+ j = drag.length;
+ while (j--) {
+ dragi = drag[j];
+ if (supportsTouch) {
+ var i = e.touches.length,
+ touch;
+ while (i--) {
+ touch = e.touches[i];
+ if (touch.identifier == dragi.el._drag.id) {
+ x = touch.clientX;
+ y = touch.clientY;
+ (e.originalEvent ? e.originalEvent : e).preventDefault();
+ break;
+ }
+ }
+ } else {
+ e.preventDefault();
+ }
+
+ //var node = dragi.el.node;
+ //o,
+ //next = node.nextSibling,
+ //parent = node.parentNode,
+ //display = node.style.display;
+
+ /* @todo: implement raphael.drag.over */
+
+ //g.win.opera && parent.removeChild(node);
+ //node.style.display = "none";
+ //o = dragi.el.paper.getElementByPoint(x, y);
+ //node.style.display = display;
+ //g.win.opera && (next ? parent.insertBefore(node, next) : parent.appendChild(node));
+ //o && eve("raphael.drag.over." + dragi.el.id, dragi.el, o);
+ x += scrollX;
+ y += scrollY;
+ eve("raphael.drag.move." + dragi.el.id, dragi.move_scope || dragi.el, x - dragi.el._drag.x, y - dragi.el._drag.y, x, y, e);
+ }
+ },
+ dragUp = function(e) {
+ R.unmousemove(dragMove).unmouseup(dragUp);
+ var i = drag.length,
+ dragi;
+ while (i--) {
+ dragi = drag[i];
+ dragi.el._drag = {};
+ eve("raphael.drag.end." + dragi.el.id, dragi.end_scope || dragi.start_scope || dragi.move_scope || dragi.el, e);
+ }
+ drag = [];
+ };
+
+
+ if (!R.canvas) {
+ return;
+ }
+
+ $ = R._createNode = function(el, attr) {
+ if (attr) {
+ if (typeof el === STRING) {
+ el = $(el);
+ }
+ for (var key in attr)
+ if (attr.hasOwnProperty(key)) {
+ el.setAttribute(key, Str(attr[key]));
+ }
+ } else {
+ el = doc.createElement(el);
+ }
+ return el;
+ };
+
+ R._getConnectedNodes = function (node) {
+ return {
+ above: [],
+ below: []
+ };
+ };
+
+ R._getTargetNode = function (coords) {
+ var x = coords[0],
+ y = coords[1];
+
+
+ };
+
+ R._containerEventHandler = function (event) {
+ event = event || win.event;
+
+ if (noHandle) {
+ return;
+ }
+
+ /* @todo: do not use offsetX and offsetY */
+ var x = event.offsetX,//mathFloor(event.pageX || (event.clientX + doc.body.scrollLeft + doc.documentElement.scrollLeft) || 0),
+ y = event.offsetY,//mathFloor(event.pageY || (event.clientY + doc.body.scrollTop + doc.documentElement.scrollTop) || 0),
+ type = event.type,
+ node = R._getTargetNode([x, y]);
+
+ //console.log(type + ":: x: " + x + " y: " + y);
+
+ /*
+ if (typeof node.listeners[type] === 'function') {
+ node.listeners[type].call(node, e);
+ }*/
+
+ };
+
+
+
+
+
+ FauxNode = function (parent) {
+ this.type = "basic";
+
+ this.owner = parent;
+
+ // Element reference needed to get attrs so that there is no need to
+ // maintain a redundant copy.
+ this._rElement = null;
+
+ // False if this element does not need mouse interactivity.
+ this.mouseInteractions = false;
+ this.matrix = null;
+
+ // An outline path that will be the outline of the shape or in case of
+ // text the rect that bounds the text.
+ this.outlinePath = null;
+
+ this.conf = {};
+ };
+
+ FauxNode.prototype = {
+
+ constructor: FauxNode,
+
+ /**
+ * This method does the complete rendering of the element, including
+ * (re)setting the bbox and image map.
+ *
+ * @returns {_L10.FauxNode.prototype}
+ */
+ render: function () {
+
+ var o = this;
+
+ o.draw();
+ o.setBBox();
+
+ return o;
+ },
+
+ /**
+ * Applies the transforms and clipping to the context and draws the
+ * element.
+ *
+ * @returns {undefined}
+ */
+ draw: function () {
+ var o = this,
+ ctx = o.context,
+ el = o._rElement,
+ m = el.matrix,
+ isClipped = o.isClipped,
+ attrs = o.validateAttrs(),
+ cr,
+ end;
+
+
+ // After the validation the attributes need to be set to the element
+ // as validateAttrs works on a copy of the attrs object.
+ el.attrs = attrs;
+
+ ctx.save();
+
+ ctx.fillStyle = attrs.fill;
+ ctx.strokeStyle = attrs.stroke;
+ ctx.lineWidth = attrs['stroke-width'];
+
+ // Applying the clip before the context is transformed as is the case
+ // in case of SVG.
+ if (cr = attrs['clip-rect']) {
+ cr = cr.split(" ");
+ ctx.rect(cr[0], cr[1], cr[2], cr[3]);
+ ctx.clip();
+ isClipped = o.isClipped = true;
+ }
+
+ o.applyTransform(m);
+
+ o.paint();
+
+ ctx.restore();
+ },
+
+ /**
+ * Parse the attributes provided to paint the element shape using the
+ * canvas context.
+ *
+ * Overridden by individual derived FauxNodes
+ */
+ paint: function () {
+ },
+
+ /**
+ * The redraw of the FauxNodes is to be handled by the CanvasObjectModel
+ * instance as it involves redrawing all the elements corresponsing to
+ * that canvas in the proper order.
+ */
+ redraw: function () {
+ this.COMInstance.redraw(this);
+ },
+
+ /**
+ * Clears the rectangle corresponding to the bounding box of the node.
+ */
+ clear: function () {
+ var o = this,
+ ctx = o.context,
+ bbox = o._bbox;
+
+ /* @todo: while clearing the stroke-width also needs to be accounted for. */
+ if (bbox) {
+ ctx.clearRect(bbox.x, bbox.y, bbox.width, bbox.height);
+ }
+ },
+
+ /**
+ * Creates the area node in the image map so that mouse interactivity
+ * can be emulated using it.
+ */
+ addMouseInteractivity: function () {
+ var o = this,
+ attrs = o._rElement.attrs,
+ bbox = o._bbox,
+ map = o.owner.wrapper._map,
+ shape = (o.type === 'circle') ? 'circle' : 'rect',
+ coords = (shape === 'circle') ? [attrs.cx, attrs.cy, attrs.r].join(",")
+ : [bbox.x, bbox.y, bbox.x2, bbox.y2].join(","),
+ area,
+ end;
+
+ area = $('area', {shape: shape, coords: coords});
+ if (map.firstChild) {
+ map.insertBefore(area, map.firstChild);
+ }
+ else {
+ map.appendChild(area);
+ }
+
+ /* @todo: should be put this check here */
+ o._mouseArea = area;
+
+ // Needed for paths.
+ o.eventListeners = {};
+ },
+
+
+ /**
+ * Update the coords of the area node (image map) corresponding to the FauxNode
+ * after it has been modified by changing attributes.
+ */
+ updateMapAreaCoords: function () {
+ var o = this,
+ oArea = o._mouseArea,
+ bbox = o._bbox;
+
+ if (oArea) {
+ //o.context.strokeRect(bbox.x, bbox.y, bbox.width, bbox.height);
+ if (o instanceof CircleFauxNode) {
+ var r = bbox.width / 2;
+ oArea.setAttribute('coords',
+ [bbox.x + r, bbox.y + r, r].join(","));
+ }
+ else {
+ oArea.setAttribute('coords',
+ [bbox.x, bbox.y, bbox.x2, bbox.y2].join(","));
+ }
+ }
+ },
+
+ /**
+ * Applies the transform given the transformation matrix to the context.
+ *
+ * @param {type} matrix
+ * @param {type} bbox
+ * @param {type} dontsetbbox
+ *
+ * @returns {_L10.FauxNode.prototype.transformBBox.tbox}
+ */
+ applyTransform: function (m) {
+ var o = this,
+ ctx = o.context,
+ split;
+
+ if (m) {
+ split = m.split();
+ ctx.translate(split.dx, split.dy);
+ !split.noRotation && ctx.rotate(deg2rad * split.rotate);
+ ctx.scale(split.scalex, split.scaley);
+ }
+ },
+
+ /**
+ *
+ */
+ setBBox: function () {
+ var o = this,
+ el = o._rElement,
+ m = el.matrix,
+ parent = o.owner,
+ pm = parent.getTransformMatrix && parent.getTransformMatrix();
+
+ if (pm) {
+ pm = pm.clone();
+ pm.add(m);
+ m = pm;
+ }
+
+ if (o.outlinePath) {
+ o._bbox = R.pathBBox(R.transformPath(o.outlinePath, m.toTransformString()).toString());
+ }
+ else {
+ o.setShapeBBox(m);
+ }
+
+ o._mouseArea && o.updateMapAreaCoords();
+ },
+
+ getBBox: function () {
+ return this._bbox;
+ },
+
+ drawPath: function (path) {
+ var o = this,
+ ctx = o.context,
+ len = (path && path.length) || 0,
+ pp = PathParser,
+ i = 0,
+ command,
+ x,
+ y,
+ end;
+
+ // The PathParser object has been borrowed from canvg. All 3rd party attributions implied.
+ pp.reset();
+ pp.setTokens(path);
+
+ //var bb = new BoundingBox();
+ if (ctx != null) {
+ ctx.beginPath();
+ }
+
+ while (!pp.isEnd()) {
+ pp.nextCommand();
+ switch (pp.command) {
+ case 'M':
+ case 'm':
+ var p = pp.getAsCurrentPoint();
+ pp.addMarker(p);
+ //bb.addPoint(p.x, p.y);
+ if (ctx != null) {
+ ctx.moveTo(p.x, p.y);
+ }
+ pp.start = pp.current;
+ while (!pp.isCommandOrEnd()) {
+ var p = pp.getAsCurrentPoint();
+ pp.addMarker(p, pp.start);
+ //bb.addPoint(p.x, p.y);
+ if (ctx != null) {
+ ctx.lineTo(p.x, p.y);
+ }
+ }
+ break;
+ case 'L':
+ case 'l':
+ while (!pp.isCommandOrEnd()) {
+ var c = pp.current;
+ var p = pp.getAsCurrentPoint();
+ pp.addMarker(p, c);
+ //bb.addPoint(p.x, p.y);
+ if (ctx != null) {
+ ctx.lineTo(p.x, p.y);
+ }
+ }
+ break;
+ case 'H':
+ case 'h':
+ while (!pp.isCommandOrEnd()) {
+ var newP = new Point((pp.isRelativeCommand() ? pp.current.x : 0) + pp.getScalar(), pp.current.y);
+ pp.addMarker(newP, pp.current);
+ pp.current = newP;
+ //bb.addPoint(pp.current.x, pp.current.y);
+ if (ctx != null) {
+ ctx.lineTo(pp.current.x, pp.current.y);
+ }
+ }
+ break;
+ case 'V':
+ case 'v':
+ while (!pp.isCommandOrEnd()) {
+ var newP = new Point(pp.current.x, (pp.isRelativeCommand() ? pp.current.y : 0) + pp.getScalar());
+ pp.addMarker(newP, pp.current);
+ pp.current = newP;
+ //bb.addPoint(pp.current.x, pp.current.y);
+ if (ctx != null) {
+ ctx.lineTo(pp.current.x, pp.current.y);
+ }
+ }
+ break;
+ case 'C':
+ case 'c':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var p1 = pp.getPoint();
+ var cntrl = pp.getAsControlPoint();
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, p1);
+ //bb.addBezierCurve(curr.x, curr.y, p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) {
+ ctx.bezierCurveTo(p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ }
+ break;
+ case 'S':
+ case 's':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var p1 = pp.getReflectedControlPoint();
+ var cntrl = pp.getAsControlPoint();
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, p1);
+ //bb.addBezierCurve(curr.x, curr.y, p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) {
+ ctx.bezierCurveTo(p1.x, p1.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ }
+ break;
+ case 'Q':
+ case 'q':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var cntrl = pp.getAsControlPoint();
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, cntrl);
+ //bb.addQuadraticCurve(curr.x, curr.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) ctx.quadraticCurveTo(cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ break;
+ case 'T':
+ case 't':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var cntrl = pp.getReflectedControlPoint();
+ pp.control = cntrl;
+ var cp = pp.getAsCurrentPoint();
+ pp.addMarker(cp, cntrl, cntrl);
+ //bb.addQuadraticCurve(curr.x, curr.y, cntrl.x, cntrl.y, cp.x, cp.y);
+ if (ctx != null) ctx.quadraticCurveTo(cntrl.x, cntrl.y, cp.x, cp.y);
+ }
+ break;
+ case 'A':
+ case 'a':
+ while (!pp.isCommandOrEnd()) {
+ var curr = pp.current;
+ var rx = pp.getScalar();
+ var ry = pp.getScalar();
+ var xAxisRotation = pp.getScalar() * (Math.PI / 180.0);
+ var largeArcFlag = pp.getScalar();
+ var sweepFlag = pp.getScalar();
+ var cp = pp.getAsCurrentPoint();
+
+ // Conversion from endpoint to center parameterization
+ // http://www.w3.org/TR/SVG11/implnote.html#ArcImplementationNotes
+ // x1', y1'
+ var currp = new Point(
+ Math.cos(xAxisRotation) * (curr.x - cp.x) / 2.0 + Math.sin(xAxisRotation) * (curr.y - cp.y) / 2.0,
+ -Math.sin(xAxisRotation) * (curr.x - cp.x) / 2.0 + Math.cos(xAxisRotation) * (curr.y - cp.y) / 2.0
+ );
+ // adjust radii
+ var l = Math.pow(currp.x,2)/Math.pow(rx,2)+Math.pow(currp.y,2)/Math.pow(ry,2);
+ if (l > 1) {
+ rx *= Math.sqrt(l);
+ ry *= Math.sqrt(l);
+ }
+ // cx', cy'
+ var s = (largeArcFlag == sweepFlag ? -1 : 1) * Math.sqrt(
+ ((Math.pow(rx,2)*Math.pow(ry,2))-(Math.pow(rx,2)*Math.pow(currp.y,2))-(Math.pow(ry,2)*Math.pow(currp.x,2))) /
+ (Math.pow(rx,2)*Math.pow(currp.y,2)+Math.pow(ry,2)*Math.pow(currp.x,2))
+ );
+ if (isNaN(s)) s = 0;
+ var cpp = new Point(s * rx * currp.y / ry, s * -ry * currp.x / rx);
+ // cx, cy
+ var centp = new Point(
+ (curr.x + cp.x) / 2.0 + Math.cos(xAxisRotation) * cpp.x - Math.sin(xAxisRotation) * cpp.y,
+ (curr.y + cp.y) / 2.0 + Math.sin(xAxisRotation) * cpp.x + Math.cos(xAxisRotation) * cpp.y
+ );
+ // vector magnitude
+ var m = function(v) { return Math.sqrt(Math.pow(v[0],2) + Math.pow(v[1],2)); }
+ // ratio between two vectors
+ var r = function(u, v) { return (u[0]*v[0]+u[1]*v[1]) / (m(u)*m(v)) }
+ // angle between two vectors
+ var a = function(u, v) { return (u[0]*v[1] < u[1]*v[0] ? -1 : 1) * Math.acos(r(u,v)); }
+ // initial angle
+ var a1 = a([1,0], [(currp.x-cpp.x)/rx,(currp.y-cpp.y)/ry]);
+ // angle delta
+ var u = [(currp.x-cpp.x)/rx,(currp.y-cpp.y)/ry];
+ var v = [(-currp.x-cpp.x)/rx,(-currp.y-cpp.y)/ry];
+ var ad = a(u, v);
+ if (r(u,v) <= -1) ad = Math.PI;
+ if (r(u,v) >= 1) ad = 0;
+
+ // for markers
+ var dir = 1 - sweepFlag ? 1.0 : -1.0;
+ var ah = a1 + dir * (ad / 2.0);
+ var halfWay = new Point(
+ centp.x + rx * Math.cos(ah),
+ centp.y + ry * Math.sin(ah)
+ );
+ pp.addMarkerAngle(halfWay, ah - dir * Math.PI / 2);
+ pp.addMarkerAngle(cp, ah - dir * Math.PI);
+
+ //bb.addPoint(cp.x, cp.y); // TODO: this is too naive, make it better
+ if (ctx != null) {
+ var r = rx > ry ? rx : ry;
+ var sx = rx > ry ? 1 : rx / ry;
+ var sy = rx > ry ? ry / rx : 1;
+
+ ctx.translate(centp.x, centp.y);
+ ctx.rotate(xAxisRotation);
+ ctx.scale(sx, sy);
+ ctx.arc(0, 0, r, a1, a1 + ad, 1 - sweepFlag);
+ ctx.scale(1/sx, 1/sy);
+ ctx.rotate(-xAxisRotation);
+ ctx.translate(-centp.x, -centp.y);
+ }
+ }
+ break;
+ case 'Z':
+ case 'z':
+ if (ctx != null) ctx.closePath();
+ pp.current = pp.start;
+ }
+ }
+
+ o.outlinePath = path;
+
+ return o;
+ },
+
+ /**
+ * Methods to add and remove event listeners emulating the DOM of
+ * standard browsers (and also the non-standard one).
+ */
+ addEventListener: function () {
+ var o = this,
+ args = arguments,
+ eventName = args && args[0],
+ handler = args && args[1],
+ area,
+ checkPathHandler,
+ end;
+
+ if (!o._mouseArea) {
+ o.addMouseInteractivity();
+ }
+
+ area = o._mouseArea;
+
+ if (typeof eventName === 'string' && typeof handler === 'function') {
+ /*
+ * If the shape has an associated path then we need to check if
+ * the mouse is within the co-ordinates of the path.
+ */
+ if (o._path) {
+
+ /*
+ * If the event being listened to is mouseover, mouseout or
+ * mousemove then the mouse position has to be constantly
+ * monitored and the event handler called explicitly when
+ * appropriate.
+ */
+ if (eventName === 'mouseover' || eventName === 'mouseout' || eventName === 'mousemove') {
+
+ if (!o._mousemoveAdded) {
+ var startListening = (function (node) {
+
+ var isInside = false,
+ isOutside = true,
+ transition = false;
+
+ return function (event) {
+ /* @todo: replace layerX and layerY with
+ * standard ways of determining mouse position.
+ */
+ var x = event.layerX,
+ y = event.layerY;
+
+ transition = false;
+ /*
+ * Check if the current position of the mouse
+ * pointer lies within the path or not.
+ */
+ if (R.isPointInsidePath(node._transformPath, x, y)) {
+ /*
+ * @todo: fix isPointInsidePath to return the
+ * proper result when the mouse pointer is on
+ * the same horizontal/vertical lines as one
+ * of the vertices of the path.
+ */
+ isInside = true;
+ if (isOutside) {
+ isOutside = false;
+ transition = true;
+ }
+ }
+ else {
+ isOutside = true;
+ if (isInside) {
+ isInside = false;
+ transition = true;
+ }
+ }
+
+ // Based on the state of the flags, fire the
+ // appropriate event handlers.
+ if (isOutside && transition && node.eventListeners['mouseout']) {
+ node.eventListeners['mouseout'].apply(this, arguments);
+ }
+ if (isInside) {
+ if (transition && node.eventListeners['mouseover']) {
+ node.eventListeners['mouseover'].apply(this, arguments);
+ }
+ if (node.eventListeners['mousemove']) {
+ node.eventListeners['mousemove'].apply(this, arguments);
+ }
+ }
+ };
+ })(o);
+
+ area.addEventListener('mousemove', startListening, false);
+ o._mousemoveAdded = true;
+ }
+
+ o.eventListeners[eventName] = handler;
+ }
+ else {
+ var checkPathHandler = (function (node, handler) {
+ return function (event) {
+ if (R.isPointInsidePath(node._path, event.layerX, event.layerY)) {
+ handler.apply(this, arguments);
+ }
+ };
+ }(o, handler));
+
+ area.addEventListener(eventName, checkPathHandler, false);
+ }
+ }
+ else {
+ area.addEventListener(eventName, handler, false);
+ }
+ }
+ },
+
+ removeEventListener: function () {
+ var o = this,
+ args = arguments,
+ eventName = args && args[0],
+ handler = args && args[1],
+ area,
+ end;
+
+ if (!o._mouseArea) {
+ return;
+ }
+ area = o._mouseArea;
+
+ if (typeof eventName === 'string' && typeof handler === 'function') {
+ area.removeEventListener(eventName, handler);
+ }
+ },
+
+ attachEvent: function () {
+ },
+
+ detachEvent: function () {
+ },
+
+ validateAttrs: function (attrs) {
+
+ var o = this,
+ elAttrs = clone(o._rElement.attrs),
+ attr,
+ val;
+
+ if (attrs === null) {
+ if (o._isValid) {
+ return elAttrs;
+ }
+ else {
+ o._isValid = true;
+ }
+ }
+
+ attrs = attrs || elAttrs;
+
+ for (attr in attrs) {
+ val = attrs[attr];
+
+ switch (attr) {
+
+ default:
+ continue;
+ }
+ }
+
+ return attrs;
+ },
+
+ attrs: function () {
+ }
+ };
+
+
+
+
+ var NodeListItem = function (node) {
+ this.node = node;
+ this.next = null;
+ this.prev = null;
+ },
+
+ NodeList = function () {
+ this.top = null;
+ this.bottom = null;
+ };
+
+ NodeList.prototype = {
+
+ constructor: NodeList,
+
+ add: function (node) {
+
+ node = new NodeListItem(node);
+
+ if (!this.bottom) {
+ this.bottom = node;
+ }
+ if (this.top) {
+ this.top.next = node;
+ }
+ node.next = null;
+ node.prev = this.top;
+
+ this.top = node;
+ },
+
+ addList: function (list) {
+ if (!this.bottom) {
+ this.bottom = list.bottom;
+ }
+
+ if (this.top) {
+ this.top.next = list.bottom;
+ list.bottom.prev = this.top;
+ }
+
+ this.top = list.top;
+ },
+
+ toFront: function (node) {
+ if (this.top === node) {
+ return false;
+ }
+
+ if (this.bottom === node) {
+ this.bottom = node.next;
+ }
+
+ //var map = node.node.canvas._map,
+ // area = node.node._mouseArea;
+
+ node.prev && (node.prev.next = node.next);
+ node.next && (node.next.prev = node.prev);
+
+ this.top.next = node;
+ node.prev = this.top;
+ node.next = null;
+
+ this.top = node;
+
+ /*if (map.firstChild) {
+ map.insertBefore(area, map.firstChild);
+ }
+ else {
+ map.appendChild(area);
+ }
+
+ node.redraw();*/
+ },
+
+ toBack: function (node) {
+ if (this.bottom === node) {
+ return false;
+ }
+
+ if (this.top === node) {
+ this.top = node.prev;
+ }
+
+ //var map = node.canvas._map,
+ // area = node._mouseArea;
+
+ node.prev && (node.prev.next = node.next);
+ node.next && (node.next.prev = node.prev);
+
+ this.bottom.prev = node;
+ node.prev = null;
+ node.next = this.bottom;
+
+ this.bottom = node;
+
+ /*map.appendChild(area);
+
+ node.redraw();*/
+ },
+
+ insertBefore: function () {
+ },
+
+ insertAfter: function () {
+ },
+
+ each: function (fn, args) {
+ var item = this.bottom;
+
+ while (item) {
+ fn.apply(item.node, args);
+ item = item.next;
+ }
+ },
+
+ iterate: function (fn, args) {
+ var item = this.bottom,
+ retVal = true;
+
+ while (item) {
+ retVal = fn.apply(item.node, args);
+
+ if (retVal === false) {
+ break;
+ }
+
+ item = item.next;
+ }
+ },
+
+ dispose: function () {
+
+ this.each(function () {
+ this.node.dispose && this.node.dispose();
+ });
+
+ this.top = null;
+ this.bottom = null;
+ },
+ };
+
+ /**
+ *
+ ncowner, a NodeCollection that corresponds to the collection of which
+ the layer neing created is a part
+
+ above, a NodeCollection iterator, that indicated the collection the layer has to be rendered. If
+ not provided then this is the first layer of ncowner.
+ */
+ var CanvasLayer = function (ncowner, canvas) {
+ this.items = new NodeList();
+
+ this.owner = ncowner;
+ //this.above = above;
+ this.element = null;
+
+ if (canvas) {
+ this.element = canvas;
+ }
+ else {
+ this.init();
+ }
+ };
+
+ CanvasLayer.prototype = {
+ constructor: CanvasLayer,
+
+ appendChild: function () {
+ var o = this,
+ ownerWrapper = o.owner.wrapper,
+ ele = this.element;
+
+ if (ownerWrapper._image) {
+ ownerWrapper.insertBefore(ele, ownerWrapper._image);
+ }
+ else {
+ ownerWrapper.appendChild(ele);
+ }
+ },
+
+ insertBefore: function () {
+
+ },
+
+ insertAfter: function () {
+
+ },
+
+ init: function () {
+ this.element = $("canvas");
+ // CHECKPOINT: width and height in %?
+ $(this.element, {
+ width: this.owner.wrapper.offsetWidth,
+ height: this.owner.wrapper.offsetHeight
+ });
+
+ this.element.style.cssText = "position:absolute;left:0;top:0;";
+
+ this.appendChild();
+ },
+
+ getCanvas: function () {
+ return this.element;
+ },
+
+ getContext: function () {
+ return this.element.getContext('2d');
+ },
+
+ addToLayer: function (node) {
+ this.items.add(node);
+ },
+
+ mergeWithLayerOnTop: function (layerObj) {
+ this.items.addList(layerObj.items);
+ layerObj.dispose(true);
+ },
+
+ mergeWithLayerOnBottom: function (layerObj) {
+ layerObj.items.addList(this.items);
+ this.items = layerObj.items;
+ layerObj.dispose(true);
+ },
+
+ dispose: function (softDispose) {
+
+ if (!softDispose) {
+ this.items.each(function () {
+ this.dispose();
+ });
+ }
+
+ this.items = null;
+ this.owner = null;
+
+ this.element.parentNode.removeChild(this.element);
+ this.element = null;
+ }
+ };
+
+ var NodeCollection = function (parent, wrapper, canvas) {
+
+ this.nodeItems = new NodeList();
+ this.collectionItems = new NodeList();
+ this.layerItems = new NodeList();
+
+ this.owner = this.parent = parent;
+ this.layerOnTop = null;
+
+ this.currentLayer = null;
+ this.baseLayer = null;
+
+ if (wrapper) {
+ this.wrapper = wrapper;
+ this.currentLayer = this.baseLayer = new CanvasLayer(this, canvas);
+ }
+ else {
+ this.init();
+ }
+ };
+
+ NodeCollection.prototype = {
+ constructor: NodeCollection,
+
+ init: function () {
+
+ var o = this,
+ parent = o.parent,
+ imageMap = parent.wrapper._image,
+ wrapper = $("div");
+
+ // Hacky but need a refernce to the image map to addEventListeners.
+ wrapper.style.cssText = "width:100%;height:100%;position:absolute;left:0;top:0;";
+
+ wrapper._map = parent.wrapper._map;
+
+ if (imageMap) {
+ parent.wrapper.insertBefore(wrapper, imageMap);
+ }
+ else {
+ parent.wrapper.appendChild(wrapper);
+ }
+
+ o.wrapper = wrapper;
+ o.currentLayer = o.baseLayer = new CanvasLayer(o);
+ },
+
+ getCurrentContext: function () {
+ return this.currentLayer.getContext();
+ },
+
+ setLayerOnTop: function (layerObj) {
+ this.layerOnTop = layerObj;
+ },
+
+ getCurrentCanvas: function () {
+ return this.currentLayer.getCanvas();
+ },
+
+ addNode: function (node) {
+ this.nodeItems.add(node);
+
+ if (node.type === "group") {
+ this.addCollection(node);
+ }
+ else {
+ this.currentLayer.addToLayer(node);
+ }
+ },
+
+ addCollection: function (collectionNode) {
+
+ collectionNode = collectionNode || new NodeCollection(this);
+
+ this.collectionItems.add(collectionNode);
+ this.currentLayer = new CanvasLayer(this);
+ this.layerItems.add(this.currentLayer);
+ collectionNode.setLayerOnTop(this.currentLayer);
+ },
+
+ dispose: function () {
+ this.nodeItems.dispose();
+ this.collectionItems.dispose();
+ this.layerItems.dispose();
+ this.owner = this.parent = null;
+
+ this.ownerLayer = null;
+ this.currentLayer = null;
+ this.baseLayer = null;
+ }
+ };
+
+ /**
+ * The CanvasObjectModel will be a layer of abstraction above the individual
+ * FauxNodes created to emulate the DOM in case of canvas rendering.
+ * The engine will be the point of contact for Raphael._engine that will be
+ * the direct consumer of the FauxNodes.
+ */
+ var CanvasObjectModel = function (cnvs, wrpr, width, height) {
+
+ var com = this,
+ root = new NodeCollection(null, wrpr, cnvs);
+
+ //root.set
+
+ com.width = width;
+ com.height = height;
+
+ com.createNode = function (type, parent) {
+
+ parent = parent || root;
+
+ var node,
+ nodeItems = parent.nodeItems,
+ layer = parent.currentLayer,
+ canvasEle = layer.getCanvas();
+
+ switch (type) {
+ case 'rect':
+ node = new RectFauxNode(parent);
+ break;
+
+ case 'circle':
+ node = new CircleFauxNode(parent);
+ break;
+
+ case 'path':
+ node = new PathFauxNode(parent);
+ break;
+
+ case 'text':
+ node = new TextFauxNode(parent);
+ break;
+
+ case 'group':
+ node = new GroupFauxNode(parent);
+ parent.addCollection(node);
+ break;
+
+ default:
+ node = new FauxNode(canvasEle);
+ }
+
+ node.COMInstance = this;
+ nodeItems.add(node);
+ layer.addToLayer(node);
+
+ return node;
+ };
+
+ /**
+ * This method is needed to redraw a node. Redraw is to be handled at the
+ * COM level as redrawing one node needs all the (connected) nodes to be
+ * redrawn in the right order.
+ *
+ * @param {type} node The node that needs to be redrawn.
+ *
+ * @returns {undefined}
+ */
+ com.redraw = function (node) {
+
+ // Check if node is a group or a shape.
+ var nodeList,
+ childNode,
+ layer;
+
+ if (node.type === "group") {
+ nodeList = node.nodeItems;
+ node.render();
+ }
+ else {
+
+ layer = node.layer;
+ nodeList = layer.items;
+ childNode = nodeList.bottom;
+
+ // Clear the canvas.
+ layer.element.width = layer.element.width;
+
+ while (childNode) {
+ fNode = childNode.node;
+ (fNode.type !== "group") && fNode.render();
+ childNode = childNode.next;
+ }
+
+ }
+
+ };
+
+ com.insertBefore = function (node) {
+
+ };
+
+ com.insertAfter = function (node) {
+
+ };
+
+ com.removeNode = function (node) {
+
+ };
+
+ com.refreshNode = function (node) {
+
+ };
+
+ com.refreshAll = function () {
+
+ };
+ };
+
+
+
+ Point = function(x, y) {
+ this.x = x;
+ this.y = y;
+ }
+
+ Point.prototype.angleTo = function(p) {
+ return Math.atan2(p.y - this.y, p.x - this.x);
+ }
+
+ Point.prototype.applyTransform = function(v) {
+ var xp = this.x * v[0] + this.y * v[2] + v[4];
+ var yp = this.x * v[1] + this.y * v[3] + v[5];
+ this.x = xp;
+ this.y = yp;
+ }
+
+ PathParser = new (function() {
+
+ this.tokens = null;
+
+ this.setTokens = function (d) {
+ if (typeof d === 'string') {
+ this.tokens = d.split(' ');
+ }
+ else {
+ this.tokens = d;
+ }
+ };
+
+ this.reset = function() {
+ this.i = -1;
+ this.command = '';
+ this.previousCommand = '';
+ this.start = new Point(0, 0);
+ this.control = new Point(0, 0);
+ this.current = new Point(0, 0);
+ this.points = [];
+ this.angles = [];
+ };
+
+ this.isEnd = function() {
+ return this.i >= this.tokens.length - 1;
+ }
+
+ this.isCommandOrEnd = function() {
+ if (this.isEnd()) {
+ return true;
+ }
+
+ return this.tokens[this.i + 1].toString().match(/^[A-Za-z]$/) != null;
+ }
+
+ this.isRelativeCommand = function() {
+ switch(this.command)
+ {
+ case 'm':
+ case 'l':
+ case 'h':
+ case 'v':
+ case 'c':
+ case 's':
+ case 'q':
+ case 't':
+ case 'a':
+ case 'z':
+ return true;
+ break;
+ }
+ return false;
+ }
+
+ this.getToken = function() {
+ this.i++;
+ return this.tokens[this.i];
+ }
+
+ this.getScalar = function() {
+ return parseFloat(this.getToken());
+ }
+
+ this.nextCommand = function() {
+ this.previousCommand = this.command;
+ this.command = this.getToken();
+ }
+
+ this.getPoint = function() {
+ var p = new Point(this.getScalar(), this.getScalar());
+ return this.makeAbsolute(p);
+ }
+
+ this.getAsControlPoint = function() {
+ var p = this.getPoint();
+ this.control = p;
+ return p;
+ }
+
+ this.getAsCurrentPoint = function() {
+ var p = this.getPoint();
+ this.current = p;
+ return p;
+ }
+
+ this.getReflectedControlPoint = function() {
+ if (this.previousCommand.toLowerCase() != 'c' &&
+ this.previousCommand.toLowerCase() != 's' &&
+ this.previousCommand.toLowerCase() != 'q' &&
+ this.previousCommand.toLowerCase() != 't' ){
+ return this.current;
+ }
+
+ // reflect point
+ var p = new Point(2 * this.current.x - this.control.x, 2 * this.current.y - this.control.y);
+ return p;
+ }
+
+ this.makeAbsolute = function(p) {
+ if (this.isRelativeCommand()) {
+ p.x += this.current.x;
+ p.y += this.current.y;
+ }
+ return p;
+ }
+
+ this.addMarker = function(p, from, priorTo) {
+ // if the last angle isn't filled in because we didn't have this point yet ...
+ if (priorTo != null && this.angles.length > 0 && this.angles[this.angles.length-1] == null) {
+ this.angles[this.angles.length-1] = this.points[this.points.length-1].angleTo(priorTo);
+ }
+ this.addMarkerAngle(p, from == null ? null : from.angleTo(p));
+ }
+
+ this.addMarkerAngle = function(p, a) {
+ this.points.push(p);
+ this.angles.push(a);
+ }
+
+ this.getMarkerPoints = function() {
+ return this.points;
+ }
+
+ this.getMarkerAngles = function() {
+ for (var i=0; i maxR) {
+ attrs.r = maxR;
+ }
+
+ if (val < 0) {
+ attrs.r = 0;
+ }
+ break;
+
+ case "width":
+ case "height":
+ if (val < 0) {
+ attrs[attr] = 0;
+ };
+ break;
+
+ default:
+ continue;
+ }
+ }
+
+ return attrs;
+ },
+
+ setShapeBBox: function (m) {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attrs,
+ sX = m.get(0),
+ sY = m.get(3),
+ tX = m.get(4),
+ tY = m.get(5),
+ strokeW = attrs['stroke-width'];
+
+ o._bbox = {
+ x: ((attrs.x * sX) + tX) - strokeW,
+ y: ((attrs.y * sY) + tY) - strokeW,
+ width: (attrs.width * sX) + (2 * strokeW),
+ height: (attrs.height * sY) + (2 * strokeW),
+ };
+
+ o._bbox.x2 = o._bbox.x + o._bbox.width;
+ o._bbox.y2 = o._bbox.y + o._bbox.height;
+
+ o.X = o._bbox.x;
+ o.Y = o._bbox.y;
+ o.W = o._bbox.width;
+ o.H = o._bbox.height;
+ }
+ });
+
+ var CircleFauxNode = function (parentObj) {
+ this.type = "circle";
+ this._isValid = false;
+
+ this.parent = this.owner = parentObj;
+ this.context = parentObj.getCurrentContext();
+ this.layer = parentObj.currentLayer;
+ },
+
+ PathFauxNode = function (parentObj) {
+ this.type = "path";
+ this._isValid = false;
+
+ this.parent = this.owner = parentObj;
+ this.context = parentObj.getCurrentContext();
+ this.layer = parentObj.currentLayer;
+ },
+
+ TextFauxNode = function (parentObj) {
+ this.type = "text";
+ this._isValid = false;
+
+ this.parent = this.owner = parentObj;
+ this.context = parentObj.getCurrentContext();
+ this.layer = parentObj.currentLayer;
+ },
+
+ GroupFauxNode = function (parent, width, height) {
+
+ this.type = "group";
+
+ this.nodeItems = new NodeList();
+ this.collectionItems = new NodeList();
+ this.layerItems = new NodeList();
+
+ this.owner = this.parent = parent;
+ this.layerOnTop = parent.currentLayer;
+
+ this.currentLayer = null;
+ this.baseLayer = null;
+
+ this.init();
+ };
+
+ CircleFauxNode.prototype = R.extend(new FauxNode(), {
+
+ constructor: CircleFauxNode,
+
+ paint: function () {
+ var o = this,
+ ctx = o.context,
+ attrs = o.validateAttrs(),
+ x = attrs.cx,
+ y = attrs.cy,
+ r = attrs.r,
+ /* @todo: provide support for rx, ry */
+ rx = r || attrs.rx,
+ ry = r || attrs.ry;
+
+ if (attrs.r) {
+
+ o.drawPath(["M", x + r, y, "A", rx, ry, 0, 1, 0, x - r, y, "A", rx, ry, 0, 1, 0, x + r, y, "Z"]);
+
+ if (attrs['stroke-width']) {
+ var strokeAlpha = attrs['stroke-opacity'] === undefined ? attrs['opacity'] : attrs['stroke-opacity'];
+ if (strokeAlpha !== undefined) {
+ ctx.globalAlpha = strokeAlpha;
+ }
+ ctx.stroke();
+ }
+ var fillAlpha = attrs['fill-opacity'] === undefined ? attrs['opacity'] : attrs['fill-opacity'];
+ if (fillAlpha !== undefined) {
+ ctx.globalAlpha = fillAlpha;
+ }
+ ctx.fill();
+ }
+
+ return;
+ },
+
+ setShapeBBox: function (m) {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attrs,
+ sX = m.get(0),
+ sY = m.get(3),
+ tX = m.get(4),
+ tY = m.get(5),
+ strokeW = attrs['stroke-width'];
+
+ o._bbox = {
+ x: tX + ((attrs.cx - attrs.r) * sX) - strokeW,
+ y: tY + ((attrs.cy - attrs.r) * sY) - strokeW,
+ width: 2 * (strokeW + (attrs.r * sX)),
+ height: 2 * ((attrs.r * sY) + strokeW)
+ };
+
+ o._bbox.x2 = o._bbox.x + o._bbox.width;
+ o._bbox.y2 = o._bbox.y + o._bbox.height;
+
+ o.X = o._bbox.x;
+ o.Y = o._bbox.y;
+ o.W = o._bbox.width;
+ o.H = o._bbox.height;
+ }
+ });
+
+ PathFauxNode.prototype = R.extend(new FauxNode(), {
+
+ constructor: PathFauxNode,
+
+ paint: function () {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attrs,
+ path = el.attr('path'),
+ m = el.matrix,
+ ctx = o.context;
+
+ // 1. Get the path from the path attribute
+ // 2. Accept paths in different array formats.
+ // 3. Optimize as this can potentially be a huge pain-point.
+ // 4. Draw path mapping M,L,H,V etc to canvas APIs
+ o.drawPath(path);
+ o._transformPath = R.transformPath(path, m.toTransformString());
+
+ var strokeAlpha = attrs['stroke-opacity'] === undefined ? attrs['opacity'] : attrs['stroke-opacity'];
+ if (strokeAlpha !== undefined) {
+ ctx.globalAlpha = strokeAlpha;
+ }
+ ctx.stroke();
+ var fillAlpha = attrs['fill-opacity'] === undefined ? attrs['opacity'] : attrs['fill-opacity'];
+ if (fillAlpha !== undefined) {
+ ctx.globalAlpha = fillAlpha;
+ }
+ ctx.fill();
+
+ return;
+ }
+ });
+
+ TextFauxNode.prototype = R.extend(new FauxNode(), {
+
+ constructor: TextFauxNode,
+
+ paint: function () {
+ var o = this,
+ el = o._rElement,
+ attrs = el.attr(),
+ text = attrs['text'],
+ stroke = attrs['stroke'],
+ valign = attrs['vertical-align'],
+ halign = attrs['text-anchor'],
+ x = attrs['x'],
+ y = attrs['y'],
+ m = el.matrix,
+ ctx = o.context,
+ path;
+
+ // apply the font styles, if any
+
+ // find the dimensions of the text using the given styles.
+ // All the dimensions should be present in the attrs user provided OR default.
+ var fontSize = attrs['font-size'] || 10,
+ lineHeight = attrs['line-height'] || toInt(fontSize, 10) * 1.2,
+ fontArr = ["normal", fontSize, attrs['font']];
+
+ // draw the text.
+ ctx.fillStyle = stroke;
+ ctx.font = fontArr.join(" ");
+
+ if (text) {
+ var texts = Str(text).split(/\n| /ig),
+ totalHeight = texts.length * lineHeight,
+ totalWidth = -Infinity,
+ startX = Infinity,
+ startY,
+ width,
+ textX,
+ textY;
+
+ if (valign === "top") {
+ startY = y + lineHeight;
+ }
+ else if (valign === "middle") {
+ startY = y - (totalHeight / 2) + (lineHeight / 2);
+ }
+ else { // valign is bottom.
+ startY = y - totalHeight + lineHeight;
+ }
+
+ for (var i = 0, ii = texts.length; i < ii; i += 1) {
+
+ text = texts[i];
+ textY = startY + (lineHeight * i);
+ width = ctx.measureText(text).width;
+
+ if (halign === "start") {
+ textX = x;
+ }
+ else if (halign === "middle") {
+ textX = x - (width / 2);
+ }
+ else {
+ textX = x - width;
+ }
+
+ totalWidth = mmax(totalWidth, width);
+ startX = mmin(startX, textX);
+
+ ctx.fillText(text, textX, textY);
+ }
+
+ el._textdirty = false;
+ }
+
+ o.outlinePath = [
+ "M",
+ startX,
+ startY - (lineHeight / 1.4),
+ "H",
+ startX + totalWidth,
+ "V",
+ startY - lineHeight + totalHeight,
+ "H",
+ startX,
+ "V",
+ startY - (lineHeight / 1.4)
+ ];
+
+ return;
+ }
+ });
+
+ GroupFauxNode.prototype = R.extend(R.extend(new FauxNode, NodeCollection.prototype), {
+
+ constructor: GroupFauxNode,
+
+ draw: function () {
+
+ // Clear the group canvas first.
+ this.layerItems.each(function () {
+ this.element.width = this.element.width;
+ });
+
+ FauxNode.prototype.draw.apply(this, arguments);
+ },
+
+ /**
+ * This method does the complete rendering of the element, including
+ * (re)setting the bbox and image map.
+ *
+ * @returns {_L10.FauxNode.prototype}
+ */
+ render: function () {
+
+ var o = this;
+
+ o.draw();
+ o.setBBox();
+
+ return o;
+ },
+
+ paint: function () {
+ var o = this,
+ list = o.nodeList,
+ el = o._rElement,
+ canvas = o.canvas,
+ attrs = el.attrs,
+ childNode = list.bottom;
+
+ /* @todo: Clean this up */
+ if (attrs.opacity !== undefined) {
+ this.layerItems.each(function () {
+ this.getContext().globalAlpha = attrs.opacity;
+ });
+ }
+
+ while (childNode) {
+ childNode.render();
+ childNode = childNode.next;
+ }
+ },
+
+ setBBox: function () {
+
+ },
+
+ addMouseInteractivity: function () {
+ },
+
+ applyTransform: function (m) {
+ var o = this,
+ parent = o.parent,
+ parentMatrix = parent.getTransformMatrix && parent.getTransformMatrix();
+
+ // Parent is a group element with a transformation applied to it.
+ if (parentMatrix) {
+ o.matrixApplied = parentMatrix.clone();
+ o.matrixApplied.add(m.a, m.b, m.c, m.d, m.e, m.f);
+ }
+ else {
+ o.matrixApplied = m;
+ }
+
+ this.layerItems.each(function () {
+ FauxNode.prototype.applyTransform.apply(this, [o.matrixApplied]);
+ })
+ },
+
+ getTransformMatrix: function () {
+ return this.matrixApplied;
+ }
+ });
+
+ Element = function (node, paper, group) {
+ var o = this,
+ parent = group || paper;
+
+ o.node = o[0] = node;
+ node.raphael = true;
+ node.raphaelid = o.id = R._oid++;
+ node._rElement = o;
+
+ o.X = 0;
+ o.Y = 0;
+
+ o.attrs = o.attrs || {};
+ o.styles = o.styles || {};
+ o.followers = o.followers || [];
+
+ o.paper = paper;
+ o.com = parent.com;
+
+ o.ca = o.customAttributes = o.customAttributes ||
+ new paper._CustomAttributes();
+
+ o.matrix = R.matrix();
+ o._ = {
+ transform: [],
+ sx: 1,
+ sy: 1,
+ dx: 0,
+ dy: 0,
+ deg: 0
+ };
+
+ o.parent = parent;
+ !parent.bottom && (parent.bottom = o);
+
+ o.prev = parent.top || null;
+ parent.top && (parent.top.next = o);
+ parent.top = o;
+ o.next = null;
+ };
+
+ Element.prototype = elproto;
+ elproto.constructor = Element;
+
+ var
+ repaint = function (el, finalAttrs, positionChanged, dimensionChanged) {
+
+ var node = getNode(el),
+ preC = R._getConnectedNodes(node),
+ attrs = el.attrs,
+ elAbove = preC.above,
+ elBelow = preC.below,
+ len,
+ attr,
+ i;
+
+ for (attr in finalAttrs) {
+ attrs[attr] = finalAttrs[attr];
+ }
+
+ node.redraw();
+ },
+
+ setFillAndStroke = function (el, params) {
+
+ var attrs = el.attrs,
+ node = el.node,
+ finalAttrs = {},
+ att,
+ val,
+ needsRepaint = false,
+ positionChanged = false,
+ dimensionChanged = false,
+ i;
+
+ for (att in params) {
+ if (params[has](att)) {
+ if (!R._availableAttrs[has](att)) {
+ continue;
+ }
+ val = params[att];
+
+ switch (att) {
+
+ case 'fill-opacity':
+ case 'opacity':
+ case 'stroke-opcaity':
+ case 'stroke':
+ case 'fill':
+ finalAttrs[att] = val;
+ needsRepaint = true;
+ break;
+
+ case 'stroke-width':
+ case "cx":
+ case "cy":
+ case "x":
+ case "y":
+ finalAttrs[att] = val;
+ positionChanged = true;
+ break;
+
+ case "width":
+ case "height":
+ finalAttrs[att] = val;
+ dimensionChanged = true;
+ break;
+
+ case "clip-rect":
+ finalAttrs[att] = val;
+ needsRepaint = true;
+ break;
+
+ case "font-size":
+ case "font":
+ case "vertical-align":
+ case "text-anchor":
+ finalAttrs[att] = val;
+ needsRepaint = true;
+
+ default:
+ continue;
+ }
+ }
+ }
+
+ tuneText(el, params, finalAttrs);
+
+ finalAttrs = node.validateAttrs(finalAttrs);
+
+ if (needsRepaint || positionChanged || dimensionChanged) {
+ repaint(el, finalAttrs, positionChanged, dimensionChanged);
+ }
+ },
+
+ leading = 1.2,
+
+ tuneText = function(el, params, finalAttrs) {
+ if (el.type != "text" || !(params[has]("text") || params[has]("font") ||
+ params[has]("font-size") || params[has]("x") || params[has]("y") ||
+ params[has]("line-height") || params[has]("vertical-align"))) {
+ return;
+ }
+
+ var a = el.attr(),
+ fontSize = params['font-size'] || a['font-size'] || 10,
+ lineHeight = toFloat(params['line-height'] || a['line-height']) || toInt(fontSize, 10) * leading,
+ valign = params["vertical-align"] || a["vertical-align"] || "middle";
+
+ if (isNaN(lineHeight)) {
+ lineHeight = fontSize * leading;
+ }
+
+ finalAttrs['font-size'] = toInt(fontSize, 10) + "px";
+ finalAttrs['font'] = params['font'] || a['font'] || 'Verdana';
+ finalAttrs['vertical-align'] = valign;
+ finalAttrs['x'] = params['x'] || a['x'] || 0;
+ finalAttrs['y'] = params['y'] || a['y'] || 0;
+ finalAttrs['line-height'] = toInt(lineHeight, 10);
+ finalAttrs['text-anchor'] = params['text-anchor'] || a['text-anchor'] || 'middle';
+ };
+
+ R._engine.initWin = function (win) {
+ win = win;
+ doc = win.document;
+ };
+
+ R._engine.setSize = function (w, h) {
+ var paper = this,
+ cs = paper.canvas.style;
+
+ cs.width = (paper.width = (+w || paper.width)) + PX;
+ cs.height = (paper.height = (+h || paper.height)) + PX;
+ /* @todo call setViewBox from setSize() */
+ return paper;
+ };
+ /* @todo implement setViewBox() */
+ R._engine.create = function () {
+ var con = R._getContainer.apply(0, arguments) || {},
+ container = con.container,
+ x = con.x,
+ y = con.y,
+ width = con.width,
+ height = con.height,
+ //handler = R._containerEventHandler,
+ wrapper,
+ cssText,
+ image,
+ mmap,
+ i,
+ paper,
+ canvas;
+
+ if (!container) {
+ throw new Error("Canvas container not found.");
+ }
+
+ paper = new R._Paper();
+ paper.canvas = wrapper = $("div");
+
+ x = (x || 0);
+ y = (y || 0);
+ paper.width = width = (width || 512);
+ paper.height = height = (height || 342);
+ paper.left = paper.top = 0;
+
+ if (container == 1) {
+ wrapper.style.cssText = cssText +
+ R.format(";width:100%;height:100%;position:absolute;left:{0}px;top:{1}px;", [x, y]);
+ doc.body.appendChild(wrapper);
+ }
+ else {
+ wrapper.style.cssText = cssText + ";width:100%;height:100%;position:absolute";
+ if (container.firstChild) {
+ container.insertBefore(wrapper, container.firstChild);
+ }
+ else {
+ container.appendChild(wrapper);
+ }
+ }
+
+ cssText = "overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);" +
+ "-webkit-user-select:none;-moz-user-select:-moz-none;" +
+ "-khtml-user-select:none;-ms-user-select:none;user-select:none;" +
+ "-o-user-select:none;cursor:default;" +
+ R.format("width:{0}px;height:{1}px;", [width, height]);
+
+ // Create the canvas element and set it to occupy full space. Retain a
+ // reference to its context.
+ canvas = $("canvas");
+ canvas.style.cssText = "position:absolute;left:0;top:0";
+ canvas.setAttribute('width', paper.width);
+ canvas.setAttribute('height', paper.height);
+
+ paper.com = new CanvasObjectModel(canvas, wrapper, paper.width, paper.height);
+
+ wrapper.appendChild(canvas);
+
+ image = $("img");
+
+ // Easter egg idea! :)
+ image.src = "image1.png";
+
+ image.style.cssText = "opacity: 0;z-index: 100;background: transparent;position: absolute;left: 0;top: 0;width: "+width+"px;height: "+height+"px";
+ wrapper.appendChild(image);
+
+ mmap = $("map");
+
+ mmap.setAttribute("name", "mousemap");
+ mmap.setAttribute("id", "mousemap"); // Needed for FF.
+
+ wrapper.appendChild(mmap);
+
+ image.setAttribute("usemap","#mousemap");
+ wrapper._image = image;
+ wrapper._map = mmap;
+
+ return paper;
+ };
+
+ var getNode = R._engine.getNode = function (el) {
+ return el.node || el[0].node;
+ };
+
+ R._engine.getLastNode = function (el) {
+ return el.node || el[el.length - 1].node;
+ };
+
+ R._engine.rect = function(paper, x, y, w, h, r, group) {
+
+ var node = paper.com.createNode('rect', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.x = x;
+ attrs.y = y;
+ attrs.width = w;
+ attrs.height = h;
+ attrs.fill = "#fff";
+ attrs.stroke = "#000";
+ attrs['stroke-width'] = 1;
+ attrs.r = r || 0;
+ attrs.rx = r || 0;
+ attrs.ry = r || 0;
+
+ el.type = "rect";
+
+ node.render();
+ return el;
+ };
+
+ R._engine.circle = function(paper, x, y, r, group) {
+ var node = paper.com.createNode('circle', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.cx = x;
+ attrs.cy = y;
+ attrs.r = r;
+ attrs.fill = 'none';
+ attrs.stroke = '#000';
+ attrs['stroke-width'] = 1;
+
+ el.type = "circle";
+
+ node.render();
+ return el;
+ };
+
+ R._engine.ellipse = function(paper, x, y, rx, ry, group) {
+ var node = new FauxNode(),
+ el = new Element(node, paper, group);
+
+ el.type = "ellipse";
+ return el;
+ };
+
+ R._engine.image = function(paper, src, x, y, w, h, group) {
+ var node = new FauxNode(),
+ el = new Element(node, paper, group);
+
+ el.type = "image";
+ return el;
+ };
+
+ R._engine.text = function(paper, x, y, text, group) {
+ var node = paper.com.createNode('text', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.x = x;
+ attrs.y = y;
+ attrs.text = text;
+ attrs.fill = 'none';
+ attrs.stroke = '#000';
+ attrs.font = 'Verdana';
+ attrs['font-size'] = '12px';
+ attrs['vertical-align'] = 'middle';
+ attrs['text-anchor'] = 'middle'
+
+ el.type = "text";
+ node.render();
+ return el;
+ };
+
+ R._engine.path = function(pathString, paper, group) {
+ var node = paper.com.createNode('path', group && group.node),
+ el = new Element(node, paper, group),
+ attrs = el.attrs;
+
+ attrs.path = pathString;
+ attrs.fill = "#fff";
+ attrs.stroke = "#000";
+ attrs['stroke-width'] = 1;
+
+ el.type = "path";
+
+ node.render();
+
+ return el;
+ };
+
+ R._engine.group = function (paper, id, group) {
+
+ var node = paper.com.createNode('group', group && group.node),
+ el = new Element(node, paper, group),
+ wrapper = node.wrapper;
+
+ id && wrapper.setAttribute('class', ['red', id].join('-'));
+
+ el.canvas = wrapper;
+
+ //(group && group.canvas.appendChild(wrapper)) || paper.canvas.appendChild(wrapper);
+
+ el.type = "group";
+ return el;
+ };
+
+ elproto._getBBox = function() {
+ if (this.removed) {
+ return {};
+ }
+
+ return {
+ x: this.X + (this.bbx || 0) - this.W / 2,
+ y: this.Y + (this.bby || 0) - this.H / 2,
+ width: this.W,
+ height: this.H
+ };
+ };
+
+ /***** ELEMENT REORDERING / RESTRUCTING *****/
+
+ elproto.toFront = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ parent = o.parent,
+ parentNode = thisNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._tofront(o, parent)) {
+ if (elproto.type === "group") {
+ parent.canvas.appendChild(thisNode);
+ }
+ else {
+ parentNode.nodeList.tofront(thisNode);
+ }
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+
+ return o;
+ };
+
+ elproto.toBack = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ parent = o.parent,
+ parentNode = thisNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._toback(o, parent)) {
+ if (elproto.type === "group") {
+ parent.canvas.appendChild(thisNode);
+ }
+ else {
+ parentNode.nodeList.toback(thisNode);
+ }
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+
+
+ return o;
+ };
+
+ elproto.insertAfter = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ thatNode = element.node,
+ parentNode = thatNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (thatNode.next) {
+ parentNode.nodeList.insertBefore(thisNode, thatNode.next);
+ }
+ else {
+ parentNode.appendChild(thisNode);
+ }
+
+ R._insertafter(o, element, o.parent, element.parent);
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+
+ return o;
+ };
+
+ elproto.insertBefore = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = o.node,
+ thatNode = element.node,
+ parentNode = thatNode.owner,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (thatNode) {
+ parentNode.nodeList.insertBefore(thisNode, thatNode);
+ }
+ else {
+ parentNode.appendChild(thisNode);
+ }
+
+ R._insertafter(o, element, o.parent, element.parent);
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+
+ return o;
+ };
+
+ elproto.appendChild = function(element) {
+ return this;
+ };
+
+ elproto.removeChild = function(element) {
+ return this;
+ };
+
+ /***** ELEMENT REORDERING / RESTRUCTING *****/
+
+
+
+ elproto.attr = function(name, value) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+
+ attrs = o.attrs,
+ ca = o.ca,
+ names,
+ params,
+ par,
+
+ res,
+ key,
+ out,
+
+ subkey,
+ delkeys,
+
+ follower,
+ ii,
+ i;
+
+ // fetch a copy of all attributes
+ if (name == null) {
+ res = {};
+ for (key in attrs) if (attrs.hasOwnProperty(key)) {
+ res[key] = attrs[key];
+ }
+ res.gradient && res.fill == "none" && (res.fill = res.gradient) && delete res.gradient;
+ res.transform = o._.transform;
+ /* @todo res.visibility = o.node.style.display === "none" ? "hidden" : "visible"; */
+ return res;
+ }
+
+ // fetch a single value
+ if (value == null && R.is(name, "string")) {
+ if (name == "fill" && attrs.fill == "none" && attrs.gradient) {
+ return attrs.gradient;
+ }
+ if (name == "transform") {
+ return o._.transform;
+ }
+ /* @todo if (name == "visibility") {
+ return this.node.style.display === "none" ? "hidden" : "visible";
+ }*/
+
+ names = name.split(separator),
+ out = {};
+
+ for (i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in attrs) {
+ out[name] = attrs[name];
+ }
+ else if (R.is(ca[name], "function")) {
+ out[name] = ca[name].def;
+ } else {
+ out[name] = R._availableAttrs[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+
+ // fetch specific attributes
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = o.attr(name[i]);
+ }
+ return out;
+ }
+
+ // prepare setter params
+ if (value != null) {
+ params = {};
+ params[name] = value;
+ }
+ else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+
+ for (key in params) {
+ eve("raphael.attr." + key + "." + o.id, o, params[key], key);
+ }
+
+ delkeys = {};
+ for (key in ca) {
+
+ if (ca[key] && params.hasOwnProperty(key) &&
+ R.is(ca[key], "function") && !ca['_invoked' + key]) {
+
+ ca['_invoked'+key] = true; // prevent recursion
+ par = ca[key].apply(o, [].concat(params[key]));
+ delete ca['_invoked'+key];
+
+ for (subkey in par) {
+ if (par.hasOwnProperty(subkey)) {
+ params[subkey] = par[subkey];
+ }
+ }
+ attrs[key] = params[key];
+ if (par === false) {
+ delkeys[key] = params[key];
+ delete params[key];
+ }
+ }
+ }
+
+ setFillAndStroke(this, params);
+
+ for (i = 0, ii = o.followers.length; i < ii; i++) {
+ follower = o.followers[i];
+ (follower.cb && !follower.cb.call(follower.el, params, o)) ||
+ follower.el.attr(params);
+ }
+
+ for (subkey in delkeys) {
+ params[subkey] = delkeys[subkey];
+ }
+ return this;
+ };
+
+ elproto.css = function (name, value) {
+ return this;
+ };
+
+
+
+
+ /**************** Drag *********************/
+ elproto.drag = function (onmove, onstart, onend, move_scope, start_scope, end_scope) {
+
+ function start(e) {
+ (e.originalEvent || e).preventDefault();
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft;
+ this._drag.x = e.clientX + scrollX;
+ this._drag.y = e.clientY + scrollY;
+ this._drag.id = e.identifier;
+ !drag.length && R.mousemove(dragMove).mouseup(dragUp);
+ drag.push({
+ el: this,
+ move_scope: move_scope,
+ start_scope: start_scope,
+ end_scope: end_scope
+ });
+ onstart && eve.on("raphael.drag.start." + this.id, onstart);
+ onmove && eve.on("raphael.drag.move." + this.id, onmove);
+ onend && eve.on("raphael.drag.end." + this.id, onend);
+ eve("raphael.drag.start." + this.id, start_scope || move_scope || this, e.clientX + scrollX, e.clientY + scrollY, e);
+ }
+ this._drag = {};
+ draggable.push({
+ el: this,
+ start: start
+ });
+ this.mousedown(start);
+ return this;
+ }
+
+ elproto.undrag = function() {
+ var i = draggable.length;
+ while (i--)
+ if (draggable[i].el == this) {
+ this.unmousedown(draggable[i].start);
+ draggable.splice(i, 1);
+ eve.unbind("raphael.drag.*." + this.id);
+ }
+ !draggable.length && R.unmousemove(dragMove).unmouseup(dragUp);
+ };
+
+
+ /***************** Drag *****************/
+ /************ TRANSFORMATIONS *************/
+
+
+
+ elproto.rotate = function(deg, cx, cy) {
+
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ deg = Str(deg).split(separator);
+ if (deg.length - 1) {
+ cx = toFloat(deg[1]);
+ cy = toFloat(deg[2]);
+ }
+ deg = toFloat(deg[0]);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ cx = bbox.x + bbox.width / 2;
+ cy = bbox.y + bbox.height / 2;
+ }
+ o.transform(o._.transform.concat([["r", deg, cx, cy]]));
+ return o;
+ };
+
+ elproto.scale = function(sx, sy, cx, cy) {
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ sx = Str(sx).split(separator);
+ if (sx.length - 1) {
+ sy = toFloat(sx[1]);
+ cx = toFloat(sx[2]);
+ cy = toFloat(sx[3]);
+ }
+ sx = toFloat(sx[0]);
+ (sy == null) && (sy = sx);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ }
+ cx = cx == null ? bbox.x + bbox.width / 2 : cx;
+ cy = cy == null ? bbox.y + bbox.height / 2 : cy;
+ o.transform(o._.transform.concat([["s", sx, sy, cx, cy]]));
+
+ return o;
+ };
+
+ elproto.translate = function(dx, dy) {
+ var o = this;
+ if (o.removed) {
+ return o;
+ }
+ dx = Str(dx).split(separator);
+ if (dx.length - 1) {
+ dy = toFloat(dx[1]);
+ }
+ dx = toFloat(dx[0]) || 0;
+ dy = +dy || 0;
+ o.transform(o._.transform.concat([["t", dx, dy]]));
+
+ return o;
+ };
+
+ elproto.transform = function(tstr) {
+
+ var o = this,
+ _ = o._,
+ sw;
+
+ if (tstr === null) {
+ return _.transform;
+ }
+
+ R._extractTransform(o, tstr);
+
+ /* @todo: what changes to be made here in the context of canvas */
+ /*o.clip && !_.clipispath && $(o.clip, {
+ transform: o.matrix.invert()
+ });
+ o.pattern && updatePosition(o); */
+ if (_.sx != 1 || _.sy != 1) {
+ sw = o.attrs[has]("stroke-width") ? o.attrs["stroke-width"] : 1;
+ o.attr({
+ "stroke-width": sw
+ });
+ }
+
+ o.node && o.node.redraw();
+
+ return o;
+ };
+
+ /************ TRANSFORMATIONS *************/
+
+
+ elproto.hide = function() {
+ return this;
+ };
+
+ elproto.show = function() {
+ return this;
+ };
+
+ elproto.blur = function(size) {
+ return this;
+ };
+
+ elproto.on = function(eventType, handler) {
+ var el = this,
+ listeners = el.listeners;
+
+ if (!listeners) {
+ listeners = el.listeners = {};
+ }
+
+ if (!listeners[eventType]) {
+ listeners[eventType] = [];
+ }
+
+ listeners[eventType].push(handler);
+ };
+
+ elproto.remove = function() {
+ return this;
+ };
+
+
+
+
+ paperproto.clear = function () {
+ eve("raphael.clear", this);
+ return this;
+ };
+
+ paperproto.remove = function () {
+ if (this.removed) {
+ return;
+ }
+
+ var paper = this,
+ canvas = paper.canvas,
+ pn = canvas.parentNode,
+ i;
+
+ eve("raphael.remove", paper);
+ pn.removeChild(canvas);
+
+ for (i in paper) {
+ paper[i] = typeof paper[i] == "function" ? R._removedFactory(i) : null;
+ }
+
+ this.removed = true;
+ };
+
+ R.toString = function () {
+ return "Your browser supports canvas.\nYou are running RedRaphael " +
+ R.version;
+ };
+
+ for (var method in elproto) {
+ if (elproto.hasOwnProperty(method) &&
+ !setproto.hasOwnProperty(method)) {
+ setproto[method] = (function (methodname) {
+ return function () {
+ var arg = arguments;
+ return this.forEach(function (el) {
+ el[methodname].apply(el, arg);
+ });
+ };
+ })(method);
+ }
+ };
+
+}(window.Raphael);
diff --git a/source/raphael.core.js b/source/raphael.core.js
new file mode 100644
index 0000000..70edfa2
--- /dev/null
+++ b/source/raphael.core.js
@@ -0,0 +1,6328 @@
+/**!
+ * RedRaphael 1.0.0 - JavaScript Vector Library
+ * Copyright (c) 2012-2013 FusionCharts Technologies
+ *
+ * Raphael 2.1.0
+ * Copyright (c) 2008-2012 Dmitry Baranovskiy
+ * Copyright © 2008-2012 Sencha Labs
+ *
+ * Licensed under the MIT license.
+ */
+(function (glob, factory) {
+ // AMD support
+ if (typeof define === "function" && define.amd) {
+ // Define as an anonymous module
+ define(["eve"], function( eve ) {
+ return factory(glob, eve);
+ });
+ } else {
+ // Browser globals (glob is window)
+ // Raphael adds itself to window
+ factory(glob, glob.eve);
+ }
+}(this, function (window, eve) {
+ /*\
+ * Raphael
+ [ method ]
+ **
+ * Creates a canvas object on which to draw.
+ * You must do this first, as all future calls to drawing methods
+ * from this instance will be bound to this canvas.
+ > Parameters
+ **
+ - container (HTMLElement|string) DOM element or its ID which is going to be a parent for drawing surface
+ - width (number)
+ - height (number)
+ - callback (function) #optional callback function which is going to be executed in the context of newly created paper
+ * or
+ - x (number)
+ - y (number)
+ - width (number)
+ - height (number)
+ - callback (function) #optional callback function which is going to be executed in the context of newly created paper
+ * or
+ - all (array) (first 3 or 4 elements in the array are equal to [containerID, width, height] or [x, y, width, height]. The rest are element descriptions in format {type: type, }). See @Paper.add.
+ - callback (function) #optional callback function which is going to be executed in the context of newly created paper
+ * or
+ - onReadyCallback (function) function that is going to be called on DOM ready event. You can also subscribe to this event via Eve’s “DOMLoad” event. In this case method returns `undefined`.
+ = (object) @Paper
+ > Usage
+ | // Each of the following examples create a canvas
+ | // that is 320px wide by 200px high.
+ | // Canvas is created at the viewport’s 10,50 coordinate.
+ | var paper = Raphael(10, 50, 320, 200);
+ | // Canvas is created at the top left corner of the #notepad element
+ | // (or its top right corner in dir="rtl" elements)
+ | var paper = Raphael(document.getElementById("notepad"), 320, 200);
+ | // Same as above
+ | var paper = Raphael("notepad", 320, 200);
+ | // Image dump
+ | var set = Raphael(["notepad", 320, 200, {
+ | type: "rect",
+ | x: 10,
+ | y: 10,
+ | width: 25,
+ | height: 25,
+ | stroke: "#f00"
+ | }, {
+ | type: "text",
+ | x: 30,
+ | y: 40,
+ | text: "Dump"
+ | }]);
+ \*/
+ function R(first) {
+ var args,
+ f;
+
+
+ if (R._url) { // reinitialize URL to be safe from popstate event
+ R._url = (R._g && R._g.win || window).location.href.replace(/#.*?$/, "");
+ }
+ if (R.is(first, "function")) {
+ return loaded ? first() : eve.on("raphael.DOMload", first);
+ }
+ else if (R.is(first, array)) {
+ return R._engine.create[apply](R, first.splice(0, 3 + R.is(first[0], nu))).add(first);
+ }
+ else {
+ args = Array.prototype.slice.call(arguments, 0);
+ if (R.is(args[args.length - 1], "function")) {
+ f = args.pop();
+ return loaded ? f.call(R._engine.create[apply](R, args)) : eve.on("raphael.DOMload", function() {
+ f.call(R._engine.create[apply](R, args));
+ });
+ } else {
+ return R._engine.create[apply](R, arguments);
+ }
+ }
+ }
+
+ R.upgrade = "1.0.0";
+ R.version = "2.1.0";
+ R.eve = eve;
+
+ var loaded,
+
+ undef,
+ E = "",
+ S = " ",
+ proto = "prototype",
+ has = "hasOwnProperty",
+ appendChild = "appendChild",
+ apply = "apply",
+ concat = "concat",
+ nu = "number",
+ string = "string",
+ array = "array",
+ object = "object",
+ finite = "finite",
+ toString = "toString",
+ fillString = "fill",
+ push = "push",
+ setAttribute = "setAttribute",
+ split = "split",
+ none = "none",
+ OBJECTSTRING = "object",
+ arrayToStr = "[object Array]",
+ objectToStr = "[object Object]",
+ arraySlice = Array.prototype.slice,
+ arraySplice = Array.prototype.splice,
+ g = {
+ doc: document,
+ win: window
+ },
+ oldRaphael = {
+ was: Object.prototype[has].call(g.win, "Raphael"),
+ is: g.win.Raphael
+ },
+ doc = g.doc,
+ win = g.win,
+
+ supportsTouch = R.supportsTouch = "createTouch" in doc,
+
+ CustomAttributes = function () {
+ /*\
+ * Raphael.ca
+ [ property (object) ]
+ **
+ * Shortcut for @Raphael.customAttributes
+ \*/
+ /*\
+ * Raphael.customAttributes
+ [ property (object) ]
+ **
+ * If you have a set of attributes that you would like to represent
+ * as a function of some number across all papers you can do it
+ * easily with custom attributes:
+ > Usage
+ | Raphael.customAttributes.hue = function (num) {
+ | num = num % 1;
+ | return {fill: "hsb(" + num + ", 0.75, 1)"};
+ | };
+ | // Custom attribute “hue” will change fill
+ | // to be given hue with fixed saturation and brightness.
+ | // Now you can use it like this:
+ | var c = paper.circle(10, 10, 10).attr({hue: .45});
+ | // or even like this:
+ | c.animate({hue: 1}, 1e3);
+ |
+ | // You could also create custom attribute
+ | // with multiple parameters:
+ | Raphael.customAttributes.hsb = function (h, s, b) {
+ | return {fill: "hsb(" + [h, s, b].join(",") + ")"};
+ | };
+ | c.attr({hsb: "0.5 .8 1"});
+ | c.animate({hsb: [1, 0, 0.5]}, 1e3);
+ \*/
+ },
+ caproto = R.ca = R.customAttributes = CustomAttributes.prototype,
+
+ Paper = function () {
+ /*\
+ * Paper.ca
+ [ property (object) ]
+ **
+ * Shortcut for @Paper.customAttributes
+ \*/
+ /*\
+ * Paper.customAttributes
+ [ property (object) ]
+ **
+ * If you have a set of attributes that you would like to represent
+ * as a function of some number you can do it easily with custom attributes:
+ > Usage
+ | paper.customAttributes.hue = function (num) {
+ | num = num % 1;
+ | return {fill: "hsb(" + num + ", 0.75, 1)"};
+ | };
+ | // Custom attribute “hue” will change fill
+ | // to be given hue with fixed saturation and brightness.
+ | // Now you can use it like this:
+ | var c = paper.circle(10, 10, 10).attr({hue: .45});
+ | // or even like this:
+ | c.animate({hue: 1}, 1e3);
+ |
+ | // You could also create custom attribute
+ | // with multiple parameters:
+ | paper.customAttributes.hsb = function (h, s, b) {
+ | return {fill: "hsb(" + [h, s, b].join(",") + ")"};
+ | };
+ | c.attr({hsb: "0.5 .8 1"});
+ | c.animate({hsb: [1, 0, 0.5]}, 1e3);
+ \*/
+ this.ca = this.customAttributes = new CustomAttributes();
+ this._CustomAttributes = function () {};
+ this._CustomAttributes.prototype = this.ca;
+ },
+
+ /*\
+ * Raphael.fn
+ [ property (object) ]
+ **
+ * You can add your own method to the canvas. For example if you want to draw a pie chart,
+ * you can create your own pie chart function and ship it as a Raphaël plugin. To do this
+ * you need to extend the `Raphael.fn` object. You should modify the `fn` object before a
+ * Raphaël instance is created, otherwise it will take no effect. Please note that the
+ * ability for namespaced plugins was removed in Raphael 2.0. It is up to the plugin to
+ * ensure any namespacing ensures proper context.
+ > Usage
+ | Raphael.fn.arrow = function (x1, y1, x2, y2, size) {
+ | return this.path( ... );
+ | };
+ | // or create namespace
+ | Raphael.fn.mystuff = {
+ | arrow: function () {…},
+ | star: function () {…},
+ | // etc…
+ | };
+ | var paper = Raphael(10, 10, 630, 480);
+ | // then use it
+ | paper.arrow(10, 10, 30, 30, 5).attr({fill: "#f00"});
+ | paper.mystuff.arrow();
+ | paper.mystuff.star();
+ \*/
+ paperproto = R.fn = Paper.prototype = R.prototype,
+
+ elements = {
+ circle: 1,
+ rect: 1,
+ path: 1,
+ ellipse: 1,
+ text: 1,
+ image: 1,
+ group: 1
+ },
+ events = "click dblclick mousedown mousemove mouseout mouseover mouseup touchstart touchmove touchend touchcancel"[split](S),
+ touchMap = R._touchMap = {
+ mousedown: "touchstart",
+ mousemove: "touchmove",
+ mouseup: "touchend"
+ },
+
+ Str = win.String,
+ toFloat = win.parseFloat,
+ toInt = win.parseInt,
+ math = win.Math,
+ mmax = math.max,
+ mmin = math.min,
+ abs = math.abs,
+ pow = math.pow,
+ mathCos = math.cos,
+ mathSin = math.sin,
+ mathSqrt = math.sqrt,
+ round = math.round,
+ PI = math.PI,
+ deg2rad = PI / 180,
+ rad2deg = 180 / PI,
+
+ lowerCase = Str.prototype.toLowerCase,
+ upperCase = Str.prototype.toUpperCase,
+ objectToString = win.Object.prototype.toString,
+ paper = {},
+
+ separator = /[, ]+/,
+ formatrg = /\{(\d+)\}/g,
+ ISURL = R._ISURL = /^url\(['"]?([^\)]+?)['"]?\)$/i,
+ colourRegExp = /^\s*((#[a-f\d]{6})|(#[a-f\d]{3})|rgba?\(\s*([\d\.]+%?\s*,\s*[\d\.]+%?\s*,\s*[\d\.]+%?(?:\s*,\s*[\d\.]+%?)?)\s*\)|hsba?\(\s*([\d\.]+(?:deg|\xb0|%)?\s*,\s*[\d\.]+%?\s*,\s*[\d\.]+(?:%?\s*,\s*[\d\.]+)?)%?\s*\)|hsla?\(\s*([\d\.]+(?:deg|\xb0|%)?\s*,\s*[\d\.]+%?\s*,\s*[\d\.]+(?:%?\s*,\s*[\d\.]+)?)%?\s*\))\s*$/i,
+ bezierrg = /^(?:cubic-)?bezier\(([^,]+),([^,]+),([^,]+),([^\)]+)\)/,
+ whitespace = /[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]/g,
+ commaSpaces = /[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*/,
+ p2s = /,?([achlmqrstvxz]),?/gi,
+ pathCommand = /([achlmrqstvz])[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029,]*((-?\d*\.?\d*(?:e[\-+]?\d+)?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*)+)/ig,
+ tCommand = /([rstm])[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029,]*((-?\d*\.?\d*(?:e[\-+]?\d+)?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*)+)/ig,
+ pathValues = /(-?\d*\.?\d*(?:e[\-+]?\d+)?)[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*,?[\x09\x0a\x0b\x0c\x0d\x20\xa0\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000\u2028\u2029]*/ig,
+ radial_gradient = R._radial_gradient = /^x?r(?:\(([^\)]*?)\))?/,
+
+ isnan = {
+ "NaN": 1,
+ "Infinity": 1,
+ "-Infinity": 1
+ },
+ hsrg = {
+ hs: 1,
+ rg: 1
+ },
+ availableAttrs = R._availableAttrs = {
+ "arrow-end": none,
+ "arrow-start": none,
+ blur: 0,
+ "clip-rect": "0 0 1e9 1e9",
+ "clip-path": E,
+ cursor: "default",
+ cx: 0,
+ cy: 0,
+ fill: "#fff",
+ "fill-opacity": 1,
+ font: '10px "Arial"',
+ "font-family": '"Arial"',
+ "font-size": "10",
+ "font-style": "normal",
+ "font-weight": 400,
+ gradient: 0,
+ height: 0,
+ href: "about:blank",
+ "letter-spacing": 0,
+ "line-height": 12,
+ "vertical-align": "middle",
+ opacity: 1,
+ path: "M0,0",
+ r: 0,
+ rx: 0,
+ ry: 0,
+ src: E,
+ stroke: "#000",
+ "stroke-dasharray": E,
+ "stroke-linecap": "butt",
+ "stroke-linejoin": "butt",
+ "stroke-miterlimit": 0,
+ "stroke-opacity": 1,
+ "stroke-width": 1,
+ target: "_blank",
+ "text-anchor": "middle",
+ "visibility": E,
+ title: E,
+ transform: E,
+ rotation: 0,
+ width: 0,
+ x: 0,
+ y: 0
+ },
+ availableAnimAttrs = R._availableAnimAttrs = {
+ blur: nu,
+ "clip-rect": "csv",
+ "clip-path": "path",
+ cx: nu,
+ cy: nu,
+ fill: "colour",
+ "fill-opacity": nu,
+ "font-size": nu,
+ height: nu,
+ opacity: nu,
+ path: "path",
+ r: nu,
+ rx: nu,
+ ry: nu,
+ stroke: "colour",
+ "stroke-opacity": nu,
+ "stroke-width": nu,
+ transform: "transform",
+ width: nu,
+ x: nu,
+ y: nu
+ },
+ eldata = {},
+
+ sortByKey = function(a, b) {
+ return a.key - b.key;
+ },
+ sortByNumber = function(a, b) {
+ return toFloat(a) - toFloat(b);
+ },
+ fun = function() {
+ },
+ pipe = function(x) {
+ return x;
+ },
+
+ rectPath = R._rectPath = function(x, y, w, h, r) {
+ if (r) {
+ return [["M", x + r, y], ["l", w - r * 2, 0], ["a", r, r, 0, 0, 1, r, r], ["l", 0, h - r * 2], ["a", r, r, 0, 0, 1, -r, r], ["l", r * 2 - w, 0], ["a", r, r, 0, 0, 1, -r, -r], ["l", 0, r * 2 - h], ["a", r, r, 0, 0, 1, r, -r], ["z"]];
+ }
+ return [["M", x, y], ["l", w, 0], ["l", 0, h], ["l", -w, 0], ["z"]];
+ },
+
+ ellipsePath = function(x, y, rx, ry) {
+ if (ry == null) {
+ ry = rx;
+ }
+ return [["M", x, y], ["m", 0, -ry], ["a", rx, ry, 0, 1, 1, 0, 2 * ry], ["a", rx, ry, 0, 1, 1, 0, -2 * ry], ["z"]];
+ },
+
+ getPath = R._getPath = {
+ group: function() {
+ return false;
+ },
+ path: function(el) {
+ return el.attr("path");
+ },
+ circle: function(el) {
+ var a = el.attrs;
+ return ellipsePath(a.cx, a.cy, a.r);
+ },
+ ellipse: function(el) {
+ var a = el.attrs;
+ return ellipsePath(a.cx, a.cy, a.rx, a.ry);
+ },
+ rect: function(el) {
+ var a = el.attrs;
+ return rectPath(a.x, a.y, a.width, a.height, a.r);
+ },
+ image: function(el) {
+ var a = el.attrs;
+ return rectPath(a.x, a.y, a.width, a.height);
+ },
+ text: function(el) {
+ var bbox = el._getBBox();
+ return rectPath(bbox.x, bbox.y, bbox.width, bbox.height);
+ }
+ },
+
+ /*\
+ * Raphael.mapPath
+ [ method ]
+ **
+ * Transform the path string with given matrix.
+ > Parameters
+ - path (string) path string
+ - matrix (object) see @Matrix
+ = (string) transformed path string
+ \*/
+ mapPath = R.mapPath = function(path, matrix) {
+ if (!matrix) {
+ return path;
+ }
+ var x,
+ y,
+ i,
+ j,
+ ii,
+ jj,
+ pathi;
+
+ path = path2curve(path);
+ for (i = 0, ii = path.length; i < ii; i++) {
+ pathi = path[i];
+ for (j = 1, jj = pathi.length; j < jj; j += 2) {
+ x = matrix.x(pathi[j], pathi[j + 1]);
+ y = matrix.y(pathi[j], pathi[j + 1]);
+ pathi[j] = x;
+ pathi[j + 1] = y;
+ }
+ }
+ return path;
+ },
+
+ /*\
+ * Raphael.pick
+ [ method ]
+ **
+ * Returns the first truthy argument.
+ \*/
+ pick = R.pick = function() {
+ for (var arg, i = 0, ii = arguments.length; i < ii; i += 1) {
+ arg = arguments[i];
+ if (!arg && arg !== false && arg !== 0) {
+ continue;
+ }
+ return arg;
+ }
+ return undef;
+ },
+
+ lastArgIfGroup = R._lastArgIfGroup = function (args, clear) {
+ var last = args.length - 1,
+ arg = args[last];
+
+ if (arg && (arg.constructor === R.el.constructor) && arg.type === 'group') {
+ if (clear) {
+ arraySplice.call(args, last, 1);
+ }
+ return arg;
+ }
+ },
+
+ merge = R.merge = function (obj1, obj2, skipUndef, tgtArr, srcArr) {
+ var item,
+ srcVal,
+ tgtVal,
+ str,
+ cRef;
+ //check whether obj2 is an array
+ //if array then iterate through it's index
+ //**** MOOTOOLS precution
+
+ if (!srcArr) {
+ tgtArr = [obj1];
+ srcArr = [obj2];
+ }
+ else {
+ tgtArr.push(obj1);
+ srcArr.push(obj2);
+ }
+
+ if (obj2 instanceof Array) {
+ for (item = 0; item < obj2.length; item += 1) {
+ try {
+ srcVal = obj1[item];
+ tgtVal = obj2[item];
+ }
+ catch (e) {
+ continue;
+ }
+
+ if (typeof tgtVal !== OBJECTSTRING) {
+ if (!(skipUndef && tgtVal === undefined)) {
+ obj1[item] = tgtVal;
+ }
+ }
+ else {
+ if (srcVal === null || typeof srcVal !== OBJECTSTRING) {
+ srcVal = obj1[item] = tgtVal instanceof Array ? [] : {};
+ }
+ cRef = checkCyclicRef(tgtVal, srcArr);
+ if (cRef !== -1) {
+ srcVal = obj1[item] = tgtArr[cRef];
+ }
+ else {
+ merge(srcVal, tgtVal, skipUndef, tgtArr, srcArr);
+ }
+ }
+ }
+ }
+ else {
+ for (item in obj2) {
+ try {
+ srcVal = obj1[item];
+ tgtVal = obj2[item];
+ }
+ catch (e) {
+ continue;
+ }
+
+ if (tgtVal !== null && typeof tgtVal === OBJECTSTRING) {
+ // Fix for issue BUG: FWXT-602
+ // IE < 9 Object.prototype.toString.call(null) gives
+ // "[object Object]" instead of "[object Null]"
+ // that's why null value becomes Object in IE < 9
+ str = objectToString.call(tgtVal);
+ if (str === objectToStr) {
+ if (srcVal === null || typeof srcVal !== OBJECTSTRING) {
+ srcVal = obj1[item] = {};
+ }
+ cRef = checkCyclicRef(tgtVal, srcArr);
+ if (cRef !== -1) {
+ srcVal = obj1[item] = tgtArr[cRef];
+ }
+ else {
+ merge(srcVal, tgtVal, skipUndef, tgtArr, srcArr);
+ }
+ }
+ else if (str === arrayToStr) {
+ if (srcVal === null || !(srcVal instanceof Array)) {
+ srcVal = obj1[item] = [];
+ }
+ cRef = checkCyclicRef(tgtVal, srcArr);
+ if (cRef !== -1) {
+ srcVal = obj1[item] = tgtArr[cRef];
+ }
+ else {
+ merge(srcVal, tgtVal, skipUndef, tgtArr, srcArr);
+ }
+ }
+ else {
+ obj1[item] = tgtVal;
+ }
+ }
+ else {
+ obj1[item] = tgtVal;
+ }
+ }
+ }
+ return obj1;
+ },
+
+ extend = R.extend = function (obj1, obj2, skipUndef) {
+ if (typeof obj1 !== OBJECTSTRING && typeof obj2 !== OBJECTSTRING) {//if none of the arguments are object then return back
+ return null;
+ }
+
+ if (typeof obj2 !== OBJECTSTRING || obj2 === null) {
+ return obj1;
+ }
+
+ if (typeof obj1 !== OBJECTSTRING) {
+ obj1 = obj2 instanceof Array ? [] : {};
+ }
+ merge(obj1, obj2, skipUndef);
+ return obj1;
+
+ },
+
+ /*\
+ * Raphael.is
+ [ method ]
+ **
+ * Handfull replacement for `typeof` operator.
+ > Parameters
+ - o (…) any object or primitive
+ - type (string) name of the type, i.e. “string”, “function”, “number”, etc.
+ = (boolean) is given value is of given type
+ \*/
+ is = R.is = function(o, type) {
+ type = lowerCase.call(type);
+
+ if (type == finite) {
+ return !isnan[has](+o);
+ }
+ if (type == array) {
+ return o instanceof Array;
+ }
+ if (type === 'object' && (o === undef || o === null)) {
+ return false;
+ }
+ return (type == "null" && o === null) ||
+ (type == typeof o && o !== null) ||
+ (type == object && o === Object(o)) ||
+ (type == "array" && Array.isArray && Array.isArray(o)) ||
+ objectToString.call(o).slice(8, -1).toLowerCase() == type;
+ },
+
+ /*\
+ * Raphael.clone
+ [ method ]
+ **
+ * Returns a recursively cloned version of an object.
+ \*/
+ clone = R.clone = function (obj) {
+ if (Object(obj) !== obj) {
+ return obj;
+ }
+ var res = new obj.constructor;
+ for (var key in obj)
+ if (obj[has](key)) {
+ res[key] = clone(obj[key]);
+ }
+ return res;
+ },
+
+ /*\
+ * Raphael.createUUID
+ [ method ]
+ **
+ * Returns RFC4122, version 4 ID
+ \*/
+ createUUID = R.createUUID = (function(uuidRegEx, uuidReplacer) {
+ return function() {
+ return "xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx".replace(uuidRegEx, uuidReplacer).toUpperCase();
+ };
+ })(/[xy]/g, function(c) {
+ var r = math.random() * 16 | 0,
+ v = c == "x" ? r : (r & 3 | 8);
+ return v.toString(16);
+ });
+
+ R._g = g;
+
+ /*\
+ * Raphael.type
+ [ property (string) ]
+ **
+ * Can be “SVG”, “VML” or empty, depending on browser support.
+ \*/
+ R.type = (win.ENABLE_RED_CANVAS && (win.CanvasRenderingContext2D || doc.createElement('canvas').getContext)) ? "CANVAS" :
+ (win.SVGAngle || doc.implementation.hasFeature("http://www.w3.org/TR/SVG11/feature#BasicStructure", "1.1") ? "SVG" : "VML");
+
+ if (R.type == "VML") {
+ var d = doc.createElement("div"),
+ b;
+
+ d.innerHTML = ' ';
+ b = d.firstChild;
+ b.style.behavior = "url(#default#VML)";
+ if (!(b && typeof b.adj == object)) {
+ return (R.type = E);
+ }
+ d = null;
+ }
+
+ /*\
+ * Raphael.svg
+ [ property (boolean) ]
+ **
+ * `true` if browser supports SVG.
+ \*/
+ /*\
+ * Raphael.vml
+ [ property (boolean) ]
+ **
+ * `true` if browser supports VML.
+ \*/
+ R.svg = !((R.vml = R.type == "VML") || (R.canvas = R.type == "CANVAS"));
+
+ R._Paper = Paper;
+ R._id = 0;
+ R._oid = 0;
+
+ /*\
+ * Raphael.angle
+ [ method ]
+ **
+ * Returns angle between two or three points
+ > Parameters
+ - x1 (number) x coord of first point
+ - y1 (number) y coord of first point
+ - x2 (number) x coord of second point
+ - y2 (number) y coord of second point
+ - x3 (number) #optional x coord of third point
+ - y3 (number) #optional y coord of third point
+ = (number) angle in degrees.
+ \*/
+ R.angle = function (x1, y1, x2, y2, x3, y3) {
+ if (x3 == null) {
+ var x = x1 - x2,
+ y = y1 - y2;
+ if (!x && !y) {
+ return 0;
+ }
+ return (180 + math.atan2(-y, -x) * rad2deg + 360) % 360;
+ }
+ else {
+ return R.angle(x1, y1, x3, y3) - R.angle(x2, y2, x3, y3);
+ }
+ };
+
+ /*\
+ * Raphael.rad
+ [ method ]
+ **
+ * Transform angle to radians
+ > Parameters
+ - deg (number) angle in degrees
+ = (number) angle in radians.
+ \*/
+ R.rad = function (deg) {
+ return deg % 360 * deg2rad;
+ };
+
+ /*\
+ * Raphael.deg
+ [ method ]
+ **
+ * Transform angle to degrees
+ > Parameters
+ - deg (number) angle in radians
+ = (number) angle in degrees.
+ \*/
+ R.deg = function (rad) {
+ return rad * rad2deg % 360;
+ };
+
+ /*\
+ * Raphael.snapTo
+ [ method ]
+ **
+ * Snaps given value to given grid.
+ > Parameters
+ - values (array|number) given array of values or step of the grid
+ - value (number) value to adjust
+ - tolerance (number) #optional tolerance for snapping. Default is `10`.
+ = (number) adjusted value.
+ \*/
+ R.snapTo = function (values, value, tolerance) {
+ var rem,
+ i;
+
+ if (!is(tolerance, finite)) {
+ tolerance = 10;
+ }
+
+ if (is(values, array)) {
+ i = values.length;
+ while (i--) {
+ if (abs(values[i] - value) <= tolerance) {
+ return values[i];
+ }
+ }
+ }
+ else {
+ values = +values;
+ rem = value % values;
+
+ if (rem < tolerance) {
+ return value - rem;
+ }
+ if (rem > values - tolerance) {
+ return value - rem + values;
+ }
+ }
+ return value;
+ };
+
+ /*\
+ * Raphael.setWindow
+ [ method ]
+ **
+ * Used when you need to draw in `<iframe>`. Switched window to the iframe one.
+ > Parameters
+ - newwin (window) new window object
+ \*/
+ R.setWindow = function (newwin) {
+ eve("raphael.setWindow", R, g.win, newwin);
+ win = g.win = newwin;
+ doc = g.doc = g.win.document;
+ if (R._engine.initWin) {
+ R._engine.initWin(g.win);
+ }
+ };
+
+ var toHex = function (color) {
+ if (R.vml) {
+ // http://dean.edwards.name/weblog/2009/10/convert-any-colour-value-to-hex-in-msie/
+ var trim = /^\s+|\s+$/g;
+ var bod;
+ try {
+ var docum = new ActiveXObject("htmlfile");
+ docum.write("");
+ docum.close();
+ bod = docum.body;
+ } catch (e) {
+ bod = createPopup().document.body;
+ }
+ var range = bod.createTextRange();
+ toHex = cacher(function(color) {
+ try {
+ bod.style.color = Str(color).replace(trim, E);
+ var value = range.queryCommandValue("ForeColor");
+ value = ((value & 255) << 16) | (value & 65280) | ((value & 16711680) >>> 16);
+ return "#" + ("000000" + value.toString(16)).slice(-6);
+ } catch (e) {
+ return none;
+ }
+ });
+ } else {
+ var i = g.doc.createElement("i");
+ i.title = "Rapha\xebl Colour Picker";
+ i.style.display = none;
+ g.doc.body.appendChild(i);
+ toHex = cacher(function(color) {
+ i.style.color = color;
+ return g.doc.defaultView.getComputedStyle(i, E).getPropertyValue("color");
+ });
+ }
+ return toHex(color);
+ },
+ hsbtoString = function() {
+ return "hsb(" + [this.h, this.s, this.b] + ")";
+ },
+ hsltoString = function() {
+ return "hsl(" + [this.h, this.s, this.l] + ")";
+ },
+ rgbtoString = function() {
+ return this.hex;
+ },
+ prepareRGB = function(r, g, b) {
+ if (g == null && is(r, object) && "r" in r && "g" in r && "b" in r) {
+ b = r.b;
+ g = r.g;
+ r = r.r;
+ }
+ if (g == null && is(r, string)) {
+ var clr = R.getRGB(r);
+ r = clr.r;
+ g = clr.g;
+ b = clr.b;
+ }
+ if (r > 1 || g > 1 || b > 1) {
+ r /= 255;
+ g /= 255;
+ b /= 255;
+ }
+
+ return [r, g, b];
+ },
+ packageRGB = function(r, g, b, o) {
+ var rgb = {
+ r: (r *= 255),
+ g: (g *= 255),
+ b: (b *= 255),
+ hex: R.rgb(r, g, b),
+ toString: rgbtoString
+ };
+ is(o, "finite") && (rgb.opacity = o);
+ return rgb;
+ };
+
+ /*\
+ * Raphael.color
+ [ method ]
+ **
+ * Parses the color string and returns object with all values for the given color.
+ > Parameters
+ - clr (string) color string in one of the supported formats (see @Raphael.getRGB)
+ = (object) Combined RGB & HSB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue,
+ o hex (string) color in HTML/CSS format: #••••••,
+ o error (boolean) `true` if string can’t be parsed,
+ o h (number) hue,
+ o s (number) saturation,
+ o v (number) value (brightness),
+ o l (number) lightness
+ o }
+ \*/
+ R.color = function(clr) {
+ var rgb;
+ if (R.is(clr, object) && "h" in clr && "s" in clr && "b" in clr) {
+ rgb = R.hsb2rgb(clr);
+ clr.r = rgb.r;
+ clr.g = rgb.g;
+ clr.b = rgb.b;
+ clr.hex = rgb.hex;
+ } else if (R.is(clr, object) && "h" in clr && "s" in clr && "l" in clr) {
+ rgb = R.hsl2rgb(clr);
+ clr.r = rgb.r;
+ clr.g = rgb.g;
+ clr.b = rgb.b;
+ clr.hex = rgb.hex;
+ } else {
+ if (R.is(clr, "string")) {
+ clr = R.getRGB(clr);
+ }
+ if (R.is(clr, object) && "r" in clr && "g" in clr && "b" in clr) {
+ rgb = R.rgb2hsl(clr);
+ clr.h = rgb.h;
+ clr.s = rgb.s;
+ clr.l = rgb.l;
+ rgb = R.rgb2hsb(clr);
+ clr.v = rgb.b;
+ } else {
+ clr = {
+ hex: none
+ };
+ clr.r = clr.g = clr.b = clr.h = clr.s = clr.v = clr.l = -1;
+ }
+ }
+ clr.toString = rgbtoString;
+ return clr;
+ };
+
+ /*\
+ * Raphael.hsb2rgb
+ [ method ]
+ **
+ * Converts HSB values to RGB object.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - v (number) value or brightness
+ = (object) RGB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue,
+ o hex (string) color in HTML/CSS format: #••••••
+ o }
+ \*/
+ R.hsb2rgb = function(h, s, v, o) {
+ if (this.is(h, object) && "h" in h && "s" in h && "b" in h) {
+ v = h.b;
+ s = h.s;
+ h = h.h;
+ o = h.o;
+ }
+ h *= 360;
+ var R, G, B, X, C;
+ h = (h % 360) / 60;
+ C = v * s;
+ X = C * (1 - abs(h % 2 - 1));
+ R = G = B = v - C;
+
+ h = ~~h;
+ R += [C, X, 0, 0, X, C][h];
+ G += [X, C, C, X, 0, 0][h];
+ B += [0, 0, X, C, C, X][h];
+ return packageRGB(R, G, B, o);
+ };
+
+ /*\
+ * Raphael.hsl2rgb
+ [ method ]
+ **
+ * Converts HSL values to RGB object.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - l (number) luminosity
+ = (object) RGB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue,
+ o hex (string) color in HTML/CSS format: #••••••
+ o }
+ \*/
+ R.hsl2rgb = function(h, s, l, o) {
+ if (this.is(h, object) && "h" in h && "s" in h && "l" in h) {
+ l = h.l;
+ s = h.s;
+ h = h.h;
+ }
+ if (h > 1 || s > 1 || l > 1) {
+ h /= 360;
+ s /= 100;
+ l /= 100;
+ }
+ h *= 360;
+ var R, G, B, X, C;
+ h = (h % 360) / 60;
+ C = 2 * s * (l < .5 ? l : 1 - l);
+ X = C * (1 - abs(h % 2 - 1));
+ R = G = B = l - C / 2;
+
+ h = ~~h;
+ R += [C, X, 0, 0, X, C][h];
+ G += [X, C, C, X, 0, 0][h];
+ B += [0, 0, X, C, C, X][h];
+ return packageRGB(R, G, B, o);
+ };
+
+ /*\
+ * Raphael.rgb2hsb
+ [ method ]
+ **
+ * Converts RGB values to HSB object.
+ > Parameters
+ - r (number) red
+ - g (number) green
+ - b (number) blue
+ = (object) HSB object in format:
+ o {
+ o h (number) hue
+ o s (number) saturation
+ o b (number) brightness
+ o }
+ \*/
+ R.rgb2hsb = function(r, g, b) {
+ b = prepareRGB(r, g, b);
+ r = b[0];
+ g = b[1];
+ b = b[2];
+
+ var H, S, V, C;
+ V = mmax(r, g, b);
+ C = V - mmin(r, g, b);
+ H = (C == 0 ? null :
+ V == r ? (g - b) / C :
+ V == g ? (b - r) / C + 2 :
+ (r - g) / C + 4
+ );
+ H = ((H + 360) % 6) * 60 / 360;
+ S = C == 0 ? 0 : C / V;
+ return {
+ h: H,
+ s: S,
+ b: V,
+ toString: hsbtoString
+ };
+ };
+
+ /*\
+ * Raphael.rgb2hsl
+ [ method ]
+ **
+ * Converts RGB values to HSL object.
+ > Parameters
+ - r (number) red
+ - g (number) green
+ - b (number) blue
+ = (object) HSL object in format:
+ o {
+ o h (number) hue
+ o s (number) saturation
+ o l (number) luminosity
+ o }
+ \*/
+ R.rgb2hsl = function(r, g, b) {
+ b = prepareRGB(r, g, b);
+ r = b[0];
+ g = b[1];
+ b = b[2];
+
+ var H, S, L, M, m, C;
+ M = mmax(r, g, b);
+ m = mmin(r, g, b);
+ C = M - m;
+ H = (C == 0 ? null :
+ M == r ? (g - b) / C :
+ M == g ? (b - r) / C + 2 :
+ (r - g) / C + 4);
+ H = ((H + 360) % 6) * 60 / 360;
+ L = (M + m) / 2;
+ S = (C == 0 ? 0 :
+ L < .5 ? C / (2 * L) :
+ C / (2 - 2 * L));
+ return {
+ h: H,
+ s: S,
+ l: L,
+ toString: hsltoString
+ };
+ };
+
+ R._path2string = function() {
+ return this.join(",").replace(p2s, "$1");
+ };
+
+ function repush(array, item) {
+ for (var i = 0, ii = array.length; i < ii; i++) {
+ if (array[i] === item) {
+ return array.push(array.splice(i, 1)[0]);
+ }
+ }
+ }
+
+ var cacher = R._cacher = function (f, scope, postprocessor) {
+ function cachedfunction() {
+ var arg = arraySlice.call(arguments, 0),
+ args = arg.join("\u2400"),
+ cache = cachedfunction.cache = cachedfunction.cache || {},
+ count = cachedfunction.count = cachedfunction.count || [];
+ if (cache[has](args)) {
+ repush(count, args);
+ return postprocessor ? postprocessor(cache[args]) : cache[args];
+ }
+ count.length >= 1e3 && delete cache[count.shift()];
+ count.push(args);
+ cache[args] = f[apply](scope, arg);
+ return postprocessor ? postprocessor(cache[args]) : cache[args];
+ }
+ return cachedfunction;
+ };
+
+ var preload = R._preload = function(src, f) {
+ var img = doc.createElement("img");
+ img.style.cssText = "position:absolute;left:-9999em;top:-9999em";
+ img.onload = function() {
+ f.call(this);
+ this.onload = null;
+ doc.body.removeChild(this);
+ };
+ img.onerror = function() {
+ doc.body.removeChild(this);
+ };
+ doc.body.appendChild(img);
+ img.src = src;
+ };
+
+ function clrToString() {
+ return this.hex;
+ }
+
+ /*\
+ * Raphael.getRGB
+ [ method ]
+ **
+ * Parses colour string as RGB object
+ > Parameters
+ - colour (string) colour string in one of formats:
+ #
+ # Colour name (“red
”, “green
”, “cornflowerblue
”, etc)
+ # #••• — shortened HTML colour: (“#000
”, “#fc0
”, etc)
+ # #•••••• — full length HTML colour: (“#000000
”, “#bd2300
”)
+ # rgb(•••, •••, •••) — red, green and blue channels’ values: (“rgb(200, 100, 0)
”)
+ # rgb(•••%, •••%, •••%) — same as above, but in %: (“rgb(100%, 175%, 0%)
”)
+ # hsb(•••, •••, •••) — hue, saturation and brightness values: (“hsb(0.5, 0.25, 1)
”)
+ # hsb(•••%, •••%, •••%) — same as above, but in %
+ # hsl(•••, •••, •••) — same as hsb
+ # hsl(•••%, •••%, •••%) — same as hsb
+ #
+ = (object) RGB object in format:
+ o {
+ o r (number) red,
+ o g (number) green,
+ o b (number) blue
+ o hex (string) color in HTML/CSS format: #••••••,
+ o error (boolean) true if string can’t be parsed
+ o }
+ \*/
+ R.getRGB = cacher(function(colour) {
+ var opacity,
+ res,
+ red,
+ green,
+ blue,
+ t,
+ values,
+ rgb;
+
+ colour && is(colour, 'object') && "opacity" in colour &&
+ (opacity = colour.opacity);
+ if (!colour || !!((colour = Str(colour)).indexOf("-") + 1)) {
+ return {
+ r: -1,
+ g: -1,
+ b: -1,
+ hex: none,
+ error: 1,
+ toString: clrToString
+ };
+ }
+ if (colour == none) {
+ return {
+ r: -1,
+ g: -1,
+ b: -1,
+ hex: none,
+ toString: clrToString
+ };
+ }
+ !(hsrg[has](colour.toLowerCase().substring(0, 2)) ||
+ colour.charAt() === "#") && (colour = toHex(colour));
+
+
+ if ((rgb = colour.match(colourRegExp))) {
+ if (rgb[2]) {
+ blue = toInt(rgb[2].substring(5), 16);
+ green = toInt(rgb[2].substring(3, 5), 16);
+ red = toInt(rgb[2].substring(1, 3), 16);
+ }
+ if (rgb[3]) {
+ blue = toInt((t = rgb[3].charAt(3)) + t, 16);
+ green = toInt((t = rgb[3].charAt(2)) + t, 16);
+ red = toInt((t = rgb[3].charAt(1)) + t, 16);
+ }
+ if (rgb[4]) {
+ values = rgb[4][split](commaSpaces);
+ red = toFloat(values[0]);
+ values[0].slice(-1) == "%" && (red *= 2.55);
+ green = toFloat(values[1]);
+ values[1].slice(-1) == "%" && (green *= 2.55);
+ blue = toFloat(values[2]);
+ values[2].slice(-1) == "%" && (blue *= 2.55);
+ rgb[1].toLowerCase().slice(0, 4) == "rgba" && (opacity = toFloat(values[3]));
+ values[3] && values[3].slice(-1) == "%" && (opacity /= 100);
+ }
+ if (rgb[5]) {
+ values = rgb[5][split](commaSpaces);
+ red = toFloat(values[0]);
+ values[0].slice(-1) == "%" && (red *= 2.55);
+ green = toFloat(values[1]);
+ values[1].slice(-1) == "%" && (green *= 2.55);
+ blue = toFloat(values[2]);
+ values[2].slice(-1) == "%" && (blue *= 2.55);
+ (values[0].slice(-3) == "deg" || values[0].slice(-1) == "\xb0") && (red /= 360);
+ rgb[1].toLowerCase().slice(0, 4) == "hsba" && (opacity = toFloat(values[3]));
+ values[3] && values[3].slice(-1) == "%" && (opacity /= 100);
+ return R.hsb2rgb(red, green, blue, opacity);
+ }
+ if (rgb[6]) {
+ values = rgb[6][split](commaSpaces);
+ red = toFloat(values[0]);
+ values[0].slice(-1) == "%" && (red *= 2.55);
+ green = toFloat(values[1]);
+ values[1].slice(-1) == "%" && (green *= 2.55);
+ blue = toFloat(values[2]);
+ values[2].slice(-1) == "%" && (blue *= 2.55);
+ (values[0].slice(-3) == "deg" || values[0].slice(-1) == "\xb0") && (red /= 360);
+ rgb[1].toLowerCase().slice(0, 4) == "hsla" && (opacity = toFloat(values[3]));
+ values[3] && values[3].slice(-1) == "%" && (opacity /= 100);
+ return R.hsl2rgb(red, green, blue, opacity);
+ }
+ rgb = {
+ r: red,
+ g: green,
+ b: blue,
+ toString: clrToString
+ };
+ rgb.hex = "#" + (16777216 | blue | (green << 8) | (red << 16)).toString(16).slice(1);
+ R.is(opacity, "finite") && (rgb.opacity = opacity);
+ return rgb;
+ }
+ return {
+ r: -1,
+ g: -1,
+ b: -1,
+ hex: none,
+ error: 1,
+ toString: clrToString
+ };
+ }, R);
+
+ R.tintshade = cacher(function(colour, percent) {
+ var rgb = R.getRGB(colour),
+ tint,
+ offset = 255;
+
+ (percent < 0) && (percent *= -1, offset = 0);
+ (percent > 1) && (percent = 1);
+
+ tint = percent === 0 ? rgb : {
+ r: offset - (offset - rgb.r) * percent,
+ g: offset - (offset - rgb.g) * percent,
+ b: offset - (offset - rgb.b) * percent,
+ toString: clrToString
+ };
+ tint.hex = R.rgb(tint.r, tint.g, tint.b);
+ rgb.error && (tint.error = rgb.error);
+
+ if ("opacity" in rgb) {
+ tint.rgba = 'rgba(' + [tint.r, tint.g, tint.b, rgb.opacity].join(',') + ')';
+ tint.opacity = rgb.opacity;
+ }
+ else {
+ tint.rgba = 'rgb(' + [tint.r, tint.g, tint.b].join(',') + ')';
+ }
+ return tint;
+ }, R);
+
+ /*\
+ * Raphael.hsb
+ [ method ]
+ **
+ * Converts HSB values to hex representation of the colour.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - b (number) value or brightness
+ = (string) hex representation of the colour.
+ \*/
+ R.hsb = cacher(function(h, s, b) {
+ return R.hsb2rgb(h, s, b).hex;
+ });
+
+ /*\
+ * Raphael.hsl
+ [ method ]
+ **
+ * Converts HSL values to hex representation of the colour.
+ > Parameters
+ - h (number) hue
+ - s (number) saturation
+ - l (number) luminosity
+ = (string) hex representation of the colour.
+ \*/
+ R.hsl = cacher(function(h, s, l) {
+ return R.hsl2rgb(h, s, l).hex;
+ });
+
+ /*\
+ * Raphael.rgb
+ [ method ]
+ **
+ * Converts RGB values to hex representation of the colour.
+ > Parameters
+ - r (number) red
+ - g (number) green
+ - b (number) blue
+ = (string) hex representation of the colour.
+ \*/
+ R.rgb = cacher(function(r, g, b) {
+ return "#" + (16777216 | b | (g << 8) | (r << 16)).toString(16).slice(1);
+ });
+
+ /*\
+ * Raphael.getColor
+ [ method ]
+ **
+ * On each call returns next colour in the spectrum. To reset it back to red call @Raphael.getColor.reset
+ > Parameters
+ - value (number) #optional brightness, default is `0.75`
+ = (string) hex representation of the colour.
+ \*/
+ R.getColor = function(value) {
+ var start = this.getColor.start = this.getColor.start || {
+ h: 0,
+ s: 1,
+ b: value || .75
+ },
+ rgb = this.hsb2rgb(start.h, start.s, start.b);
+ start.h += .075;
+ if (start.h > 1) {
+ start.h = 0;
+ start.s -= .2;
+ start.s <= 0 && (this.getColor.start = {
+ h: 0,
+ s: 1,
+ b: start.b
+ });
+ }
+ return rgb.hex;
+ };
+
+ /*\
+ * Raphael.getColor.reset
+ [ method ]
+ **
+ * Resets spectrum position for @Raphael.getColor back to red.
+ \*/
+ R.getColor.reset = function() {
+ delete this.start;
+ };
+
+ // http://schepers.cc/getting-to-the-point
+ function catmullRom2bezier(crp, z) {
+ var d = [];
+ for (var i = 0, iLen = crp.length; iLen - 2 * !z > i; i += 2) {
+ var p = [
+ {
+ x: +crp[i - 2],
+ y: +crp[i - 1]
+ },
+ {
+ x: +crp[i],
+ y: +crp[i + 1]
+ },
+ {
+ x: +crp[i + 2],
+ y: +crp[i + 3]
+ },
+ {
+ x: +crp[i + 4],
+ y: +crp[i + 5]
+ }
+ ];
+ if (z) {
+ if (!i) {
+ p[0] = {
+ x: +crp[iLen - 2],
+ y: +crp[iLen - 1]
+ };
+ } else if (iLen - 4 == i) {
+ p[3] = {
+ x: +crp[0],
+ y: +crp[1]
+ };
+ } else if (iLen - 2 == i) {
+ p[2] = {
+ x: +crp[0],
+ y: +crp[1]
+ };
+ p[3] = {
+ x: +crp[2],
+ y: +crp[3]
+ };
+ }
+ } else {
+ if (iLen - 4 == i) {
+ p[3] = p[2];
+ } else if (!i) {
+ p[0] = {
+ x: +crp[i],
+ y: +crp[i + 1]
+ };
+ }
+ }
+ d.push(["C",
+ (-p[0].x + 6 * p[1].x + p[2].x) / 6,
+ (-p[0].y + 6 * p[1].y + p[2].y) / 6,
+ (p[1].x + 6 * p[2].x - p[3].x) / 6,
+ (p[1].y + 6 * p[2].y - p[3].y) / 6,
+ p[2].x,
+ p[2].y
+ ]);
+ }
+
+ return d;
+ }
+
+ /*\
+ * Raphael.parsePathString
+ [ method ]
+ **
+ * Utility method
+ **
+ * Parses given path string into an array of arrays of path segments.
+ > Parameters
+ - pathString (string|array) path string or array of segments (in the last case it will be returned straight away)
+ = (array) array of segments.
+ \*/
+ R.parsePathString = function(pathString) {
+ if (!pathString) {
+ return null;
+ }
+ var pth = paths(pathString);
+ if (pth.arr) {
+ return pathClone(pth.arr);
+ }
+
+ var paramCounts = {
+ a: 7,
+ c: 6,
+ h: 1,
+ l: 2,
+ m: 2,
+ r: 4,
+ q: 4,
+ s: 4,
+ t: 2,
+ v: 1,
+ z: 0
+ },
+ data = [];
+ if (R.is(pathString, array) && R.is(pathString[0], array)) { // rough assumption
+ data = pathClone(pathString);
+ }
+ if (!data.length) {
+ Str(pathString).replace(pathCommand, function(a, b, c) {
+ var params = [],
+ name = b.toLowerCase();
+ c.replace(pathValues, function(a, b) {
+ b && params.push(+b);
+ });
+ if (name == "m" && params.length > 2) {
+ data.push([b][concat](params.splice(0, 2)));
+ name = "l";
+ b = b == "m" ? "l" : "L";
+ }
+ if (name == "r") {
+ data.push([b][concat](params));
+ } else
+ while (params.length >= paramCounts[name]) {
+ data.push([b][concat](params.splice(0, paramCounts[name])));
+ if (!paramCounts[name]) {
+ break;
+ }
+ }
+ });
+ }
+ data.toString = R._path2string;
+ pth.arr = pathClone(data);
+ return data;
+ };
+
+ /*\
+ * Raphael.parseTransformString
+ [ method ]
+ **
+ * Utility method
+ **
+ * Parses given path string into an array of transformations.
+ > Parameters
+ - TString (string|array) transform string or array of transformations (in the last case it will be returned straight away)
+ = (array) array of transformations.
+ \*/
+ R.parseTransformString = cacher(function(TString) {
+ if (!TString) {
+ return null;
+ }
+ var paramCounts = {
+ r: 3,
+ s: 4,
+ t: 2,
+ m: 6
+ },
+ data = [];
+ if (R.is(TString, array) && R.is(TString[0], array)) { // rough assumption
+ data = pathClone(TString);
+ }
+ if (!data.length) {
+ Str(TString).replace(tCommand, function(a, b, c) {
+ var params = [],
+ name = lowerCase.call(b);
+ c.replace(pathValues, function(a, b) {
+ b && params.push(+b);
+ });
+ data.push([b][concat](params));
+ });
+ }
+ data.toString = R._path2string;
+ return data;
+ });
+ // PATHS
+ var paths = function(ps) {
+ var p = paths.ps = paths.ps || {};
+ if (p[ps]) {
+ p[ps].sleep = 100;
+ } else {
+ p[ps] = {
+ sleep: 100
+ };
+ }
+ setTimeout(function() {
+ for (var key in p)
+ if (p[has](key) && key != ps) {
+ p[key].sleep--;
+ !p[key].sleep && delete p[key];
+ }
+ });
+ return p[ps];
+ };
+
+ /*\
+ * Raphael.findDotsAtSegment
+ [ method ]
+ **
+ * Utility method
+ **
+ * Find dot coordinates on the given cubic bezier curve at the given t.
+ > Parameters
+ - p1x (number) x of the first point of the curve
+ - p1y (number) y of the first point of the curve
+ - c1x (number) x of the first anchor of the curve
+ - c1y (number) y of the first anchor of the curve
+ - c2x (number) x of the second anchor of the curve
+ - c2y (number) y of the second anchor of the curve
+ - p2x (number) x of the second point of the curve
+ - p2y (number) y of the second point of the curve
+ - t (number) position on the curve (0..1)
+ = (object) point information in format:
+ o {
+ o x: (number) x coordinate of the point
+ o y: (number) y coordinate of the point
+ o m: {
+ o x: (number) x coordinate of the left anchor
+ o y: (number) y coordinate of the left anchor
+ o }
+ o n: {
+ o x: (number) x coordinate of the right anchor
+ o y: (number) y coordinate of the right anchor
+ o }
+ o start: {
+ o x: (number) x coordinate of the start of the curve
+ o y: (number) y coordinate of the start of the curve
+ o }
+ o end: {
+ o x: (number) x coordinate of the end of the curve
+ o y: (number) y coordinate of the end of the curve
+ o }
+ o alpha: (number) angle of the curve derivative at the point
+ o }
+ \*/
+ R.findDotsAtSegment = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t) {
+ var t1 = 1 - t,
+ t13 = pow(t1, 3),
+ t12 = pow(t1, 2),
+ t2 = t * t,
+ t3 = t2 * t,
+ x = t13 * p1x + t12 * 3 * t * c1x + t1 * 3 * t * t * c2x + t3 * p2x,
+ y = t13 * p1y + t12 * 3 * t * c1y + t1 * 3 * t * t * c2y + t3 * p2y,
+ mx = p1x + 2 * t * (c1x - p1x) + t2 * (c2x - 2 * c1x + p1x),
+ my = p1y + 2 * t * (c1y - p1y) + t2 * (c2y - 2 * c1y + p1y),
+ nx = c1x + 2 * t * (c2x - c1x) + t2 * (p2x - 2 * c2x + c1x),
+ ny = c1y + 2 * t * (c2y - c1y) + t2 * (p2y - 2 * c2y + c1y),
+ ax = t1 * p1x + t * c1x,
+ ay = t1 * p1y + t * c1y,
+ cx = t1 * c2x + t * p2x,
+ cy = t1 * c2y + t * p2y,
+ alpha = (90 - math.atan2(mx - nx, my - ny) * 180 / PI);
+ (mx > nx || my < ny) && (alpha += 180);
+ return {
+ x: x,
+ y: y,
+ m: {
+ x: mx,
+ y: my
+ },
+ n: {
+ x: nx,
+ y: ny
+ },
+ start: {
+ x: ax,
+ y: ay
+ },
+ end: {
+ x: cx,
+ y: cy
+ },
+ alpha: alpha
+ };
+ };
+
+ /*\
+ * Raphael.bezierBBox
+ [ method ]
+ **
+ * Utility method
+ **
+ * Return bounding box of a given cubic bezier curve
+ > Parameters
+ - p1x (number) x of the first point of the curve
+ - p1y (number) y of the first point of the curve
+ - c1x (number) x of the first anchor of the curve
+ - c1y (number) y of the first anchor of the curve
+ - c2x (number) x of the second anchor of the curve
+ - c2y (number) y of the second anchor of the curve
+ - p2x (number) x of the second point of the curve
+ - p2y (number) y of the second point of the curve
+ * or
+ - bez (array) array of six points for bezier curve
+ = (object) point information in format:
+ o {
+ o min: {
+ o x: (number) x coordinate of the left point
+ o y: (number) y coordinate of the top point
+ o }
+ o max: {
+ o x: (number) x coordinate of the right point
+ o y: (number) y coordinate of the bottom point
+ o }
+ o }
+ \*/
+ R.bezierBBox = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y) {
+ if (!R.is(p1x, "array")) {
+ p1x = [p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y];
+ }
+ var bbox = curveDim.apply(null, p1x);
+ return {
+ x: bbox.min.x,
+ y: bbox.min.y,
+ x2: bbox.max.x,
+ y2: bbox.max.y,
+ width: bbox.max.x - bbox.min.x,
+ height: bbox.max.y - bbox.min.y
+ };
+ };
+
+ /*\
+ * Raphael.isPointInsideBBox
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns `true` if given point is inside bounding boxes.
+ > Parameters
+ - bbox (string) bounding box
+ - x (string) x coordinate of the point
+ - y (string) y coordinate of the point
+ = (boolean) `true` if point inside
+ \*/
+ R.isPointInsideBBox = function(bbox, x, y) {
+ return x >= bbox.x && x <= bbox.x2 && y >= bbox.y && y <= bbox.y2;
+ };
+
+ /*\
+ * Raphael.isBBoxIntersect
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns `true` if two bounding boxes intersect
+ > Parameters
+ - bbox1 (string) first bounding box
+ - bbox2 (string) second bounding box
+ = (boolean) `true` if they intersect
+ \*/
+ R.isBBoxIntersect = function(bbox1, bbox2) {
+ var i = R.isPointInsideBBox;
+ return i(bbox2, bbox1.x, bbox1.y) ||
+ i(bbox2, bbox1.x2, bbox1.y) ||
+ i(bbox2, bbox1.x, bbox1.y2) ||
+ i(bbox2, bbox1.x2, bbox1.y2) ||
+ i(bbox1, bbox2.x, bbox2.y) ||
+ i(bbox1, bbox2.x2, bbox2.y) ||
+ i(bbox1, bbox2.x, bbox2.y2) ||
+ i(bbox1, bbox2.x2, bbox2.y2) ||
+ (bbox1.x < bbox2.x2 && bbox1.x > bbox2.x ||
+ bbox2.x < bbox1.x2 && bbox2.x > bbox1.x) &&
+ (bbox1.y < bbox2.y2 && bbox1.y > bbox2.y || bbox2.y < bbox1.y2 && bbox2.y > bbox1.y);
+ };
+
+ function base3(t, p1, p2, p3, p4) {
+ var t1 = -3 * p1 + 9 * p2 - 9 * p3 + 3 * p4,
+ t2 = t * t1 + 6 * p1 - 12 * p2 + 6 * p3;
+ return t * t2 - 3 * p1 + 3 * p2;
+ }
+
+ function bezlen(x1, y1, x2, y2, x3, y3, x4, y4, z) {
+ if (z == null) {
+ z = 1;
+ }
+ z = z > 1 ? 1 : z < 0 ? 0 : z;
+ var z2 = z / 2,
+ n = 12,
+ Tvalues = [-0.1252, 0.1252, -0.3678, 0.3678, -0.5873, 0.5873, -0.7699, 0.7699, -0.9041, 0.9041, -0.9816, 0.9816],
+ Cvalues = [0.2491, 0.2491, 0.2335, 0.2335, 0.2032, 0.2032, 0.1601, 0.1601, 0.1069, 0.1069, 0.0472, 0.0472],
+ sum = 0;
+ for (var i = 0; i < n; i++) {
+ var ct = z2 * Tvalues[i] + z2,
+ xbase = base3(ct, x1, x2, x3, x4),
+ ybase = base3(ct, y1, y2, y3, y4),
+ comb = xbase * xbase + ybase * ybase;
+ sum += Cvalues[i] * mathSqrt(comb);
+ }
+ return z2 * sum;
+ }
+
+ function getTatLen(x1, y1, x2, y2, x3, y3, x4, y4, ll) {
+ if (ll < 0 || bezlen(x1, y1, x2, y2, x3, y3, x4, y4) < ll) {
+ return;
+ }
+ var t = 1,
+ step = t / 2,
+ t2 = t - step,
+ l,
+ e = .01;
+ l = bezlen(x1, y1, x2, y2, x3, y3, x4, y4, t2);
+ while (abs(l - ll) > e) {
+ step /= 2;
+ t2 += (l < ll ? 1 : -1) * step;
+ l = bezlen(x1, y1, x2, y2, x3, y3, x4, y4, t2);
+ }
+ return t2;
+ }
+
+ function intersect(x1, y1, x2, y2, x3, y3, x4, y4) {
+ if (
+ mmax(x1, x2) < mmin(x3, x4) ||
+ mmin(x1, x2) > mmax(x3, x4) ||
+ mmax(y1, y2) < mmin(y3, y4) ||
+ mmin(y1, y2) > mmax(y3, y4)
+ ) {
+ return;
+ }
+ var nx = (x1 * y2 - y1 * x2) * (x3 - x4) - (x1 - x2) * (x3 * y4 - y3 * x4),
+ ny = (x1 * y2 - y1 * x2) * (y3 - y4) - (y1 - y2) * (x3 * y4 - y3 * x4),
+ denominator = (x1 - x2) * (y3 - y4) - (y1 - y2) * (x3 - x4);
+
+ if (!denominator) {
+ return;
+ }
+ var px = nx / denominator,
+ py = ny / denominator,
+ px2 = +px.toFixed(2),
+ py2 = + py.toFixed(2);
+ if (
+ px2 < +mmin(x1, x2).toFixed(2) ||
+ px2 > +mmax(x1, x2).toFixed(2) ||
+ px2 < +mmin(x3, x4).toFixed(2) ||
+ px2 > +mmax(x3, x4).toFixed(2) ||
+ py2 < +mmin(y1, y2).toFixed(2) ||
+ py2 > +mmax(y1, y2).toFixed(2) ||
+ py2 < +mmin(y3, y4).toFixed(2) ||
+ py2 > +mmax(y3, y4).toFixed(2)
+ ) {
+ return;
+ }
+ return {
+ x: px,
+ y: py
+ };
+ }
+
+ function inter(bez1, bez2) {
+ return interHelper(bez1, bez2);
+ }
+
+ function interCount(bez1, bez2) {
+ return interHelper(bez1, bez2, 1);
+ }
+
+ function interHelper(bez1, bez2, justCount) {
+ var bbox1 = R.bezierBBox(bez1),
+ bbox2 = R.bezierBBox(bez2);
+
+ if (!R.isBBoxIntersect(bbox1, bbox2)) {
+ return justCount ? 0 : [];
+ }
+ var l1 = bezlen.apply(0, bez1),
+ l2 = bezlen.apply(0, bez2),
+ n1 = mmax(~~(l1 / 5), 1),
+ n2 = mmax(~~(l2 / 5), 1),
+ dots1 = [],
+ dots2 = [],
+ xy = {},
+ res = justCount ? 0 : [];
+
+ for (var i = 0; i < n1 + 1; i++) {
+ var p = R.findDotsAtSegment.apply(R, bez1.concat(i / n1));
+ dots1.push({
+ x: p.x,
+ y: p.y,
+ t: i / n1
+ });
+ }
+ for (i = 0; i < n2 + 1; i++) {
+ p = R.findDotsAtSegment.apply(R, bez2.concat(i / n2));
+ dots2.push({
+ x: p.x,
+ y: p.y,
+ t: i / n2
+ });
+ }
+ for (i = 0; i < n1; i++) {
+ for (var j = 0; j < n2; j++) {
+ var di = dots1[i],
+ di1 = dots1[i + 1],
+ dj = dots2[j],
+ dj1 = dots2[j + 1],
+ ci = abs(di1.x - di.x) < .001 ? "y" : "x",
+ cj = abs(dj1.x - dj.x) < .001 ? "y" : "x",
+ is = intersect(di.x, di.y, di1.x, di1.y, dj.x, dj.y, dj1.x, dj1.y);
+ if (is) {
+ if (xy[is.x.toFixed(4)] == is.y.toFixed(4)) {
+ continue;
+ }
+ xy[is.x.toFixed(4)] = is.y.toFixed(4);
+ var t1 = di.t + abs((is[ci] - di[ci]) / (di1[ci] - di[ci])) * (di1.t - di.t),
+ t2 = dj.t + abs((is[cj] - dj[cj]) / (dj1[cj] - dj[cj])) * (dj1.t - dj.t);
+ if (t1 >= 0 && t1 <= 1.001 && t2 >= 0 && t2 <= 1.001) {
+ if (justCount) {
+ res++;
+ } else {
+ res.push({
+ x: is.x,
+ y: is.y,
+ t1: mmin(t1, 1),
+ t2: mmin(t2, 1)
+ });
+ }
+ }
+ }
+ }
+ }
+ return res;
+ }
+
+ /*\
+ * Raphael.pathIntersection
+ [ method ]
+ **
+ * Utility method
+ **
+ * Finds intersections of two paths
+ > Parameters
+ - path1 (string) path string
+ - path2 (string) path string
+ = (array) dots of intersection
+ o [
+ o {
+ o x: (number) x coordinate of the point
+ o y: (number) y coordinate of the point
+ o t1: (number) t value for segment of path1
+ o t2: (number) t value for segment of path2
+ o segment1: (number) order number for segment of path1
+ o segment2: (number) order number for segment of path2
+ o bez1: (array) eight coordinates representing beziér curve for the segment of path1
+ o bez2: (array) eight coordinates representing beziér curve for the segment of path2
+ o }
+ o ]
+ \*/
+ R.pathIntersection = function(path1, path2) {
+ return interPathHelper(path1, path2);
+ };
+ R.pathIntersectionNumber = function(path1, path2) {
+ return interPathHelper(path1, path2, 1);
+ };
+ function interPathHelper(path1, path2, justCount) {
+ path1 = R._path2curve(path1);
+ path2 = R._path2curve(path2);
+ var x1, y1, x2, y2, x1m, y1m, x2m, y2m, bez1, bez2,
+ res = justCount ? 0 : [];
+ for (var i = 0, ii = path1.length; i < ii; i++) {
+ var pi = path1[i];
+ if (pi[0] == "M") {
+ x1 = x1m = pi[1];
+ y1 = y1m = pi[2];
+ } else {
+ if (pi[0] == "C") {
+ bez1 = [x1, y1].concat(pi.slice(1));
+ x1 = bez1[6];
+ y1 = bez1[7];
+ } else {
+ bez1 = [x1, y1, x1, y1, x1m, y1m, x1m, y1m];
+ x1 = x1m;
+ y1 = y1m;
+ }
+ for (var j = 0, jj = path2.length; j < jj; j++) {
+ var pj = path2[j];
+ if (pj[0] == "M") {
+ x2 = x2m = pj[1];
+ y2 = y2m = pj[2];
+ } else {
+ if (pj[0] == "C") {
+ bez2 = [x2, y2].concat(pj.slice(1));
+ x2 = bez2[6];
+ y2 = bez2[7];
+ } else {
+ bez2 = [x2, y2, x2, y2, x2m, y2m, x2m, y2m];
+ x2 = x2m;
+ y2 = y2m;
+ }
+ var intr = interHelper(bez1, bez2, justCount);
+ if (justCount) {
+ res += intr;
+ } else {
+ for (var k = 0, kk = intr.length; k < kk; k++) {
+ intr[k].segment1 = i;
+ intr[k].segment2 = j;
+ intr[k].bez1 = bez1;
+ intr[k].bez2 = bez2;
+ }
+ res = res.concat(intr);
+ }
+ }
+ }
+ }
+ }
+ return res;
+ }
+
+ /*\
+ * Raphael.isPointInsidePath
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns `true` if given point is inside a given closed path.
+ > Parameters
+ - path (string) path string
+ - x (number) x of the point
+ - y (number) y of the point
+ = (boolean) true, if point is inside the path
+ \*/
+ R.isPointInsidePath = function(path, x, y) {
+ var bbox = R.pathBBox(path);
+ return R.isPointInsideBBox(bbox, x, y) &&
+ ((interPathHelper(path, [["M", x, y], ["H", bbox.x2 + 10]], 1) % 2 == 1) ||
+ (interPathHelper(path, [["M", x, y], ["V", bbox.y2 + 10]], 1) % 2 == 1))
+ };
+ R._removedFactory = function(methodname) {
+ return function() {
+ eve("raphael.log", null, "Rapha\xebl: you are calling to method \u201c" + methodname + "\u201d of removed object", methodname);
+ };
+ };
+
+ /*\
+ * Raphael.pathBBox
+ [ method ]
+ **
+ * Utility method
+ **
+ * Return bounding box of a given path
+ > Parameters
+ - path (string) path string
+ = (object) bounding box
+ o {
+ o x: (number) x coordinate of the left top point of the box
+ o y: (number) y coordinate of the left top point of the box
+ o x2: (number) x coordinate of the right bottom point of the box
+ o y2: (number) y coordinate of the right bottom point of the box
+ o width: (number) width of the box
+ o height: (number) height of the box
+ o cx: (number) x coordinate of the center of the box
+ o cy: (number) y coordinate of the center of the box
+ o }
+ \*/
+ var pathDimensions = R.pathBBox = function(path) {
+ var pth = paths(path);
+ if (pth.bbox) {
+ return pth.bbox;
+ }
+ if (!path) {
+ return {
+ x: 0,
+ y: 0,
+ width: 0,
+ height: 0,
+ x2: 0,
+ y2: 0
+ };
+ }
+ path = path2curve(path);
+ var x = 0,
+ y = 0,
+ X = [],
+ Y = [],
+ p;
+ for (var i = 0, ii = path.length; i < ii; i++) {
+ p = path[i];
+ if (p[0] == "M") {
+ x = p[1];
+ y = p[2];
+ X.push(x);
+ Y.push(y);
+ } else {
+ var dim = curveDim(x, y, p[1], p[2], p[3], p[4], p[5], p[6]);
+ X = X[concat](dim.min.x, dim.max.x);
+ Y = Y[concat](dim.min.y, dim.max.y);
+ x = p[5];
+ y = p[6];
+ }
+ }
+ var xmin = mmin[apply](0, X),
+ ymin = mmin[apply](0, Y),
+ xmax = mmax[apply](0, X),
+ ymax = mmax[apply](0, Y),
+ bb = {
+ x: xmin,
+ y: ymin,
+ x2: xmax,
+ y2: ymax,
+ width: xmax - xmin,
+ height: ymax - ymin
+ };
+ pth.bbox = clone(bb);
+ return bb;
+ },
+ pathClone = function(pathArray) {
+ var res = clone(pathArray);
+ res.toString = R._path2string;
+ return res;
+ },
+ pathToRelative = R._pathToRelative = function(pathArray) {
+ var pth = paths(pathArray);
+ if (pth.rel) {
+ return pathClone(pth.rel);
+ }
+ if (!R.is(pathArray, array) || !R.is(pathArray && pathArray[0], array)) { // rough assumption
+ pathArray = R.parsePathString(pathArray);
+ }
+ var res = [],
+ x = 0,
+ y = 0,
+ mx = 0,
+ my = 0,
+ start = 0;
+ if (pathArray[0][0] == "M") {
+ x = pathArray[0][1];
+ y = pathArray[0][2];
+ mx = x;
+ my = y;
+ start++;
+ res.push(["M", x, y]);
+ }
+ for (var i = start, ii = pathArray.length; i < ii; i++) {
+ var r = res[i] = [],
+ pa = pathArray[i];
+ if (pa[0] != lowerCase.call(pa[0])) {
+ r[0] = lowerCase.call(pa[0]);
+ switch (r[0]) {
+ case "a":
+ r[1] = pa[1];
+ r[2] = pa[2];
+ r[3] = pa[3];
+ r[4] = pa[4];
+ r[5] = pa[5];
+ r[6] = +(pa[6] - x).toFixed(3);
+ r[7] = +(pa[7] - y).toFixed(3);
+ break;
+ case "v":
+ r[1] = +(pa[1] - y).toFixed(3);
+ break;
+ case "m":
+ mx = pa[1];
+ my = pa[2];
+ default:
+ for (var j = 1, jj = pa.length; j < jj; j++) {
+ r[j] = +(pa[j] - ((j % 2) ? x : y)).toFixed(3);
+ }
+ }
+ } else {
+ r = res[i] = [];
+ if (pa[0] == "m") {
+ mx = pa[1] + x;
+ my = pa[2] + y;
+ }
+ for (var k = 0, kk = pa.length; k < kk; k++) {
+ res[i][k] = pa[k];
+ }
+ }
+ var len = res[i].length;
+ switch (res[i][0]) {
+ case "z":
+ x = mx;
+ y = my;
+ break;
+ case "h":
+ x += +res[i][len - 1];
+ break;
+ case "v":
+ y += +res[i][len - 1];
+ break;
+ default:
+ x += +res[i][len - 2];
+ y += +res[i][len - 1];
+ }
+ }
+ res.toString = R._path2string;
+ pth.rel = pathClone(res);
+ return res;
+ },
+ pathToAbsolute = R._pathToAbsolute = function(pathArray) {
+ var pth = paths(pathArray), res;
+ if (pth.abs) {
+ return pathClone(pth.abs);
+ }
+ if (!R.is(pathArray, array) || !R.is(pathArray && pathArray[0], array)) { // rough assumption
+ pathArray = R.parsePathString(pathArray);
+ }
+ if (!pathArray || !pathArray.length) {
+ res = ["M", 0, 0];
+ res.toString = R._path2string;
+ return res;
+ }
+ var x = 0,
+ y = 0,
+ mx = 0,
+ my = 0,
+ start = 0;
+ res = [];
+ if (pathArray[0][0] == "M") {
+ x = +pathArray[0][1];
+ y = +pathArray[0][2];
+ mx = x;
+ my = y;
+ start++;
+ res[0] = ["M", x, y];
+ }
+ var crz = pathArray.length == 3 && pathArray[0][0] == "M" && pathArray[1][0].toUpperCase() == "R" && pathArray[2][0].toUpperCase() == "Z";
+ for (var r, pa, i = start, ii = pathArray.length; i < ii; i++) {
+ res.push(r = []);
+ pa = pathArray[i];
+ if (pa[0] != upperCase.call(pa[0])) {
+ r[0] = upperCase.call(pa[0]);
+ switch (r[0]) {
+ case "A":
+ r[1] = pa[1];
+ r[2] = pa[2];
+ r[3] = pa[3];
+ r[4] = pa[4];
+ r[5] = pa[5];
+ r[6] = +(pa[6] + x);
+ r[7] = +(pa[7] + y);
+ break;
+ case "V":
+ r[1] = +pa[1] + y;
+ break;
+ case "H":
+ r[1] = +pa[1] + x;
+ break;
+ case "R":
+ var dots = [x, y][concat](pa.slice(1));
+ for (var j = 2, jj = dots.length; j < jj; j++) {
+ dots[j] = +dots[j] + x;
+ dots[++j] = +dots[j] + y;
+ }
+ res.pop();
+ res = res[concat](catmullRom2bezier(dots, crz));
+ break;
+ case "M":
+ mx = +pa[1] + x;
+ my = +pa[2] + y;
+ default:
+ for (j = 1, jj = pa.length; j < jj; j++) {
+ r[j] = +pa[j] + ((j % 2) ? x : y);
+ }
+ }
+ } else if (pa[0] == "R") {
+ dots = [x, y][concat](pa.slice(1));
+ res.pop();
+ res = res[concat](catmullRom2bezier(dots, crz));
+ r = ["R"][concat](pa.slice(-2));
+ } else {
+ for (var k = 0, kk = pa.length; k < kk; k++) {
+ r[k] = pa[k];
+ }
+ }
+ switch (r[0]) {
+ case "Z":
+ x = mx;
+ y = my;
+ break;
+ case "H":
+ x = r[1];
+ break;
+ case "V":
+ y = r[1];
+ break;
+ case "M":
+ mx = r[r.length - 2];
+ my = r[r.length - 1];
+ default:
+ x = r[r.length - 2];
+ y = r[r.length - 1];
+ }
+ }
+ res.toString = R._path2string;
+ pth.abs = pathClone(res);
+ return res;
+ },
+ l2c = function(x1, y1, x2, y2) {
+ return [x1, y1, x2, y2, x2, y2];
+ },
+ q2c = function(x1, y1, ax, ay, x2, y2) {
+ var _13 = 1 / 3,
+ _23 = 2 / 3;
+ return [
+ _13 * x1 + _23 * ax,
+ _13 * y1 + _23 * ay,
+ _13 * x2 + _23 * ax,
+ _13 * y2 + _23 * ay,
+ x2,
+ y2
+ ];
+ },
+ a2c = function(x1, y1, rx, ry, angle, large_arc_flag, sweep_flag, x2, y2, recursive) {
+ // for more information of where this math came from visit:
+ // http://www.w3.org/TR/SVG11/implnote.html#ArcImplementationNotes
+ var _120 = PI * 120 / 180,
+ rad = deg2rad * (+angle || 0),
+ res = [],
+ xy,
+ rotate = cacher(function(x, y, rad) {
+ var X = x * mathCos(rad) - y * mathSin(rad),
+ Y = x * mathSin(rad) + y * mathCos(rad);
+ return {
+ x: X,
+ y: Y
+ };
+ });
+ if (!recursive) {
+ xy = rotate(x1, y1, -rad);
+ x1 = xy.x;
+ y1 = xy.y;
+ xy = rotate(x2, y2, -rad);
+ x2 = xy.x;
+ y2 = xy.y;
+ var cos = mathCos(deg2rad * angle),
+ sin = mathSin(deg2rad * angle),
+ x = (x1 - x2) / 2,
+ y = (y1 - y2) / 2;
+ var h = (x * x) / (rx * rx) + (y * y) / (ry * ry);
+ if (h > 1) {
+ h = mathSqrt(h);
+ rx = h * rx;
+ ry = h * ry;
+ }
+ var rx2 = rx * rx,
+ ry2 = ry * ry,
+ k = (large_arc_flag == sweep_flag ? -1 : 1) *
+ mathSqrt(abs((rx2 * ry2 - rx2 * y * y - ry2 * x * x) / (rx2 * y * y + ry2 * x * x))),
+ cx = k * rx * y / ry + (x1 + x2) / 2,
+ cy = k * -ry * x / rx + (y1 + y2) / 2,
+ f1 = math.asin(((y1 - cy) / ry).toFixed(9)),
+ f2 = math.asin(((y2 - cy) / ry).toFixed(9));
+
+ f1 = x1 < cx ? PI - f1 : f1;
+ f2 = x2 < cx ? PI - f2 : f2;
+ f1 < 0 && (f1 = PI * 2 + f1);
+ f2 < 0 && (f2 = PI * 2 + f2);
+ if (sweep_flag && f1 > f2) {
+ f1 = f1 - PI * 2;
+ }
+ if (!sweep_flag && f2 > f1) {
+ f2 = f2 - PI * 2;
+ }
+ } else {
+ f1 = recursive[0];
+ f2 = recursive[1];
+ cx = recursive[2];
+ cy = recursive[3];
+ }
+ var df = f2 - f1;
+ if (abs(df) > _120) {
+ var f2old = f2,
+ x2old = x2,
+ y2old = y2;
+ f2 = f1 + _120 * (sweep_flag && f2 > f1 ? 1 : -1);
+ x2 = cx + rx * mathCos(f2);
+ y2 = cy + ry * mathSin(f2);
+ res = a2c(x2, y2, rx, ry, angle, 0, sweep_flag, x2old, y2old, [f2, f2old, cx, cy]);
+ }
+ df = f2 - f1;
+ var c1 = mathCos(f1),
+ s1 = mathSin(f1),
+ c2 = mathCos(f2),
+ s2 = mathSin(f2),
+ t = math.tan(df / 4),
+ hx = 4 / 3 * rx * t,
+ hy = 4 / 3 * ry * t,
+ m1 = [x1, y1],
+ m2 = [x1 + hx * s1, y1 - hy * c1],
+ m3 = [x2 + hx * s2, y2 - hy * c2],
+ m4 = [x2, y2];
+ m2[0] = 2 * m1[0] - m2[0];
+ m2[1] = 2 * m1[1] - m2[1];
+ if (recursive) {
+ return [m2, m3, m4][concat](res);
+ } else {
+ res = [m2, m3, m4][concat](res).join()[split](",");
+ var newres = [];
+ for (var i = 0, ii = res.length; i < ii; i++) {
+ newres[i] = i % 2 ? rotate(res[i - 1], res[i], rad).y : rotate(res[i], res[i + 1], rad).x;
+ }
+ return newres;
+ }
+ },
+ findDotAtSegment = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t) {
+ var t1 = 1 - t;
+ return {
+ x: pow(t1, 3) * p1x + pow(t1, 2) * 3 * t * c1x + t1 * 3 * t * t * c2x + pow(t, 3) * p2x,
+ y: pow(t1, 3) * p1y + pow(t1, 2) * 3 * t * c1y + t1 * 3 * t * t * c2y + pow(t, 3) * p2y
+ };
+ },
+ curveDim = cacher(function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y) {
+ var a = (c2x - 2 * c1x + p1x) - (p2x - 2 * c2x + c1x),
+ b = 2 * (c1x - p1x) - 2 * (c2x - c1x),
+ c = p1x - c1x,
+ t1 = (-b + mathSqrt(b * b - 4 * a * c)) / 2 / a,
+ t2 = (-b - mathSqrt(b * b - 4 * a * c)) / 2 / a,
+ y = [p1y, p2y],
+ x = [p1x, p2x],
+ dot;
+ abs(t1) > "1e12" && (t1 = .5);
+ abs(t2) > "1e12" && (t2 = .5);
+ if (t1 > 0 && t1 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t1);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ if (t2 > 0 && t2 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t2);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ a = (c2y - 2 * c1y + p1y) - (p2y - 2 * c2y + c1y);
+ b = 2 * (c1y - p1y) - 2 * (c2y - c1y);
+ c = p1y - c1y;
+ t1 = (-b + mathSqrt(b * b - 4 * a * c)) / 2 / a;
+ t2 = (-b - mathSqrt(b * b - 4 * a * c)) / 2 / a;
+ abs(t1) > "1e12" && (t1 = .5);
+ abs(t2) > "1e12" && (t2 = .5);
+ if (t1 > 0 && t1 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t1);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ if (t2 > 0 && t2 < 1) {
+ dot = findDotAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, t2);
+ x.push(dot.x);
+ y.push(dot.y);
+ }
+ return {
+ min: {
+ x: mmin[apply](0, x),
+ y: mmin[apply](0, y)
+ },
+ max: {
+ x: mmax[apply](0, x),
+ y: mmax[apply](0, y)
+ }
+ };
+ }),
+ path2curve = R._path2curve = cacher(function(path, path2) {
+ var pth = !path2 && paths(path);
+ if (!path2 && pth.curve) {
+ return pathClone(pth.curve);
+ }
+ var p = pathToAbsolute(path),
+ p2 = path2 && pathToAbsolute(path2),
+ attrs = {
+ x: 0,
+ y: 0,
+ bx: 0,
+ by: 0,
+ X: 0,
+ Y: 0,
+ qx: null,
+ qy: null
+ },
+ attrs2 = {
+ x: 0,
+ y: 0,
+ bx: 0,
+ by: 0,
+ X: 0,
+ Y: 0,
+ qx: null,
+ qy: null
+ },
+ processPath = function(path, d) {
+ var nx, ny;
+ if (!path) {
+ return ["C", d.x, d.y, d.x, d.y, d.x, d.y];
+ }
+ !(path[0] in {
+ T: 1,
+ Q: 1
+ }) && (d.qx = d.qy = null);
+ switch (path[0]) {
+ case "M":
+ d.X = path[1];
+ d.Y = path[2];
+ break;
+ case "A":
+ path = ["C"][concat](a2c[apply](0, [d.x, d.y][concat](path.slice(1))));
+ break;
+ case "S":
+ nx = d.x + (d.x - (d.bx || d.x));
+ ny = d.y + (d.y - (d.by || d.y));
+ path = ["C", nx, ny][concat](path.slice(1));
+ break;
+ case "T":
+ d.qx = d.x + (d.x - (d.qx || d.x));
+ d.qy = d.y + (d.y - (d.qy || d.y));
+ path = ["C"][concat](q2c(d.x, d.y, d.qx, d.qy, path[1], path[2]));
+ break;
+ case "Q":
+ d.qx = path[1];
+ d.qy = path[2];
+ path = ["C"][concat](q2c(d.x, d.y, path[1], path[2], path[3], path[4]));
+ break;
+ case "L":
+ path = ["C"][concat](l2c(d.x, d.y, path[1], path[2]));
+ break;
+ case "H":
+ path = ["C"][concat](l2c(d.x, d.y, path[1], d.y));
+ break;
+ case "V":
+ path = ["C"][concat](l2c(d.x, d.y, d.x, path[1]));
+ break;
+ case "Z":
+ path = ["C"][concat](l2c(d.x, d.y, d.X, d.Y));
+ break;
+ }
+ return path;
+ },
+ fixArc = function(pp, i) {
+ if (pp[i].length > 7) {
+ pp[i].shift();
+ var pi = pp[i];
+ while (pi.length) {
+ pp.splice(i++, 0, ["C"][concat](pi.splice(0, 6)));
+ }
+ pp.splice(i, 1);
+ ii = mmax(p.length, p2 && p2.length || 0);
+ }
+ },
+ fixM = function(path1, path2, a1, a2, i) {
+ if (path1 && path2 && path1[i][0] == "M" && path2[i][0] != "M") {
+ path2.splice(i, 0, ["M", a2.x, a2.y]);
+ a1.bx = 0;
+ a1.by = 0;
+ a1.x = path1[i][1];
+ a1.y = path1[i][2];
+ ii = mmax(p.length, p2 && p2.length || 0);
+ }
+ };
+ for (var i = 0, ii = mmax(p.length, p2 && p2.length || 0); i < ii; i++) {
+ p[i] = processPath(p[i], attrs);
+ fixArc(p, i);
+ p2 && (p2[i] = processPath(p2[i], attrs2));
+ p2 && fixArc(p2, i);
+ fixM(p, p2, attrs, attrs2, i);
+ fixM(p2, p, attrs2, attrs, i);
+ var seg = p[i],
+ seg2 = p2 && p2[i],
+ seglen = seg.length,
+ seg2len = p2 && seg2.length;
+ attrs.x = seg[seglen - 2];
+ attrs.y = seg[seglen - 1];
+ attrs.bx = toFloat(seg[seglen - 4]) || attrs.x;
+ attrs.by = toFloat(seg[seglen - 3]) || attrs.y;
+ attrs2.bx = p2 && (toFloat(seg2[seg2len - 4]) || attrs2.x);
+ attrs2.by = p2 && (toFloat(seg2[seg2len - 3]) || attrs2.y);
+ attrs2.x = p2 && seg2[seg2len - 2];
+ attrs2.y = p2 && seg2[seg2len - 1];
+ }
+ if (!p2) {
+ pth.curve = pathClone(p);
+ }
+ return p2 ? [p, p2] : p;
+ }, null, pathClone),
+ parseDots = R._parseDots = cacher(function(gradient) {
+ var dots = [];
+ for (var i = 0, ii = gradient.length; i < ii; i++) {
+ var dot = {},
+ par = gradient[i].match(/^([^:]*):?([\d\.]*)/);
+ dot.color = R.getRGB(par[1]);
+ if (dot.color.error) {
+ return null;
+ }
+ //store opacity information
+ dot.opacity = dot.color.opacity;
+ dot.color = dot.color.hex;
+ par[2] && (dot.offset = par[2] + "%");
+ dots.push(dot);
+ }
+ for (i = 1, ii = dots.length - 1; i < ii; i++) {
+ if (!dots[i].offset) {
+ var start = toFloat(dots[i - 1].offset || 0),
+ end = 0;
+ for (var j = i + 1; j < ii; j++) {
+ if (dots[j].offset) {
+ end = dots[j].offset;
+ break;
+ }
+ }
+ if (!end) {
+ end = 100;
+ j = ii;
+ }
+ end = toFloat(end);
+ var d = (end - start) / (j - i + 1);
+ for (; i < j; i++) {
+ start += d;
+ dots[i].offset = start + "%";
+ }
+ }
+ }
+ return dots;
+ }),
+ tear = R._tear = function(el, paper) {
+ el == paper.top && (paper.top = el.prev);
+ el == paper.bottom && (paper.bottom = el.next);
+ el.next && (el.next.prev = el.prev);
+ el.prev && (el.prev.next = el.next);
+ },
+ tofront = R._tofront = function(el, paper) {
+ if (paper.top === el) {
+ return false;
+ }
+ tear(el, paper);
+ el.next = null;
+ el.prev = paper.top;
+ paper.top.next = el;
+ paper.top = el;
+ return true;
+ },
+ toback = R._toback = function(el, paper) {
+ if (paper.bottom === el) {
+ return false;
+ }
+ tear(el, paper);
+ el.next = paper.bottom;
+ el.prev = null;
+ paper.bottom.prev = el;
+ paper.bottom = el;
+ return true;
+ },
+ insertafter = R._insertafter = function(el, el2, paper, paper2) {
+ tear(el, paper);
+ el.parent = paper2;
+ el2 === paper2.top && (paper2.top = el);
+ el2.next && (el2.next.prev = el);
+ el.next = el2.next;
+ el.prev = el2;
+ el2.next = el;
+ },
+ insertbefore = R._insertbefore = function(el, el2, paper, paper2) {
+ tear(el, paper);
+ el.parent = paper2;
+ el2 === paper2.bottom && (paper2.bottom = el);
+ el2.prev && (el2.prev.next = el);
+ el.prev = el2.prev;
+ el2.prev = el;
+ el.next = el2;
+ },
+
+ /*\
+ * Raphael.toMatrix
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns matrix of transformations applied to a given path
+ > Parameters
+ - path (string) path string
+ - transform (string|array) transformation string
+ = (object) @Matrix
+ \*/
+ toMatrix = R.toMatrix = function(path, transform) {
+ var bb = pathDimensions(path),
+ el = {
+ _: {
+ transform: E
+ },
+ getBBox: function() {
+ return bb;
+ }
+ };
+ extractTransform(el, transform);
+ return el.matrix;
+ },
+
+ /*\
+ * Raphael.transformPath
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns path transformed by a given transformation
+ > Parameters
+ - path (string) path string
+ - transform (string|array) transformation string
+ = (string) path
+ \*/
+ transformPath = R.transformPath = function(path, transform) {
+ return mapPath(path, toMatrix(path, transform));
+ },
+ extractTransform = R._extractTransform = function(el, tstr) {
+ if (tstr == null) {
+ return el._.transform;
+ }
+ tstr = Str(tstr).replace(/\.{3}|\u2026/g, el._.transform || E);
+ var tdata = R.parseTransformString(tstr),
+ deg = 0,
+ dx = 0,
+ dy = 0,
+ sx = 1,
+ sy = 1,
+ _ = el._,
+ m = new Matrix;
+ _.transform = tdata || [];
+ if (tdata) {
+ for (var i = 0, ii = tdata.length; i < ii; i++) {
+ var t = tdata[i],
+ tlen = t.length,
+ command = Str(t[0]).toLowerCase(),
+ absolute = t[0] != command,
+ inver = absolute ? m.invert() : 0,
+ x1,
+ y1,
+ x2,
+ y2,
+ bb;
+ if (command == "t" && tlen == 3) {
+ if (absolute) {
+ x1 = inver.x(0, 0);
+ y1 = inver.y(0, 0);
+ x2 = inver.x(t[1], t[2]);
+ y2 = inver.y(t[1], t[2]);
+ m.translate(x2 - x1, y2 - y1);
+ } else {
+ m.translate(t[1], t[2]);
+ }
+ } else if (command == "r") {
+ if (tlen == 2) {
+ bb = bb || el.getBBox(1);
+ m.rotate(t[1], bb.x + bb.width / 2, bb.y + bb.height / 2);
+ deg += t[1];
+ } else if (tlen == 4) {
+ if (absolute) {
+ x2 = inver.x(t[2], t[3]);
+ y2 = inver.y(t[2], t[3]);
+ m.rotate(t[1], x2, y2);
+ } else {
+ m.rotate(t[1], t[2], t[3]);
+ }
+ deg += t[1];
+ }
+ } else if (command == "s") {
+ if (tlen == 2 || tlen == 3) {
+ bb = bb || el.getBBox(1);
+ m.scale(t[1], t[tlen - 1], bb.x + bb.width / 2, bb.y + bb.height / 2);
+ sx *= t[1];
+ sy *= t[tlen - 1];
+ } else if (tlen == 5) {
+ if (absolute) {
+ x2 = inver.x(t[3], t[4]);
+ y2 = inver.y(t[3], t[4]);
+ m.scale(t[1], t[2], x2, y2);
+ } else {
+ m.scale(t[1], t[2], t[3], t[4]);
+ }
+ sx *= t[1];
+ sy *= t[2];
+ }
+ } else if (command == "m" && tlen == 7) {
+ m.add(t[1], t[2], t[3], t[4], t[5], t[6]);
+ }
+ _.dirtyT = 1;
+ el.matrix = m;
+ }
+ }
+
+ /*\
+ * Element.matrix
+ [ property (object) ]
+ **
+ * Keeps @Matrix object, which represents element transformation
+ \*/
+ el.matrix = m;
+
+ _.sx = sx;
+ _.sy = sy;
+ _.deg = deg;
+ _.dx = dx = m.e;
+ _.dy = dy = m.f;
+
+ if (sx == 1 && sy == 1 && !deg && _.bbox) {
+ _.bbox.x += +dx;
+ _.bbox.y += +dy;
+ } else {
+ _.dirtyT = 1;
+ }
+ },
+ getEmpty = function(item) {
+ var l = item[0];
+ switch (l.toLowerCase()) {
+ case "t":
+ return [l, 0, 0];
+ case "m":
+ return [l, 1, 0, 0, 1, 0, 0];
+ case "r":
+ if (item.length == 4) {
+ return [l, 0, item[2], item[3]];
+ } else {
+ return [l, 0];
+ }
+ case "s":
+ if (item.length == 5) {
+ return [l, 1, 1, item[3], item[4]];
+ } else if (item.length == 3) {
+ return [l, 1, 1];
+ } else {
+ return [l, 1];
+ }
+ }
+ },
+ equaliseTransform = R._equaliseTransform = function(t1, t2) {
+ t2 = Str(t2).replace(/\.{3}|\u2026/g, t1);
+ t1 = R.parseTransformString(t1) || [];
+ t2 = R.parseTransformString(t2) || [];
+ var maxlength = mmax(t1.length, t2.length),
+ from = [],
+ to = [],
+ i = 0, j, jj,
+ tt1, tt2;
+ for (; i < maxlength; i++) {
+ tt1 = t1[i] || getEmpty(t2[i]);
+ tt2 = t2[i] || getEmpty(tt1);
+ if ((tt1[0] != tt2[0]) ||
+ (tt1[0].toLowerCase() == "r" && (tt1[2] != tt2[2] || tt1[3] != tt2[3])) ||
+ (tt1[0].toLowerCase() == "s" && (tt1[3] != tt2[3] || tt1[4] != tt2[4]))
+ ) {
+ return;
+ }
+ from[i] = [];
+ to[i] = [];
+ for (j = 0, jj = mmax(tt1.length, tt2.length); j < jj; j++) {
+ j in tt1 && (from[i][j] = tt1[j]);
+ j in tt2 && (to[i][j] = tt2[j]);
+ }
+ }
+ return {
+ from: from,
+ to: to
+ };
+ };
+ R._getContainer = function(x, y, w, h) {
+ var container;
+ container = h == null && !R.is(x, object) ? g.doc.getElementById(x) : x;
+ if (container == null) {
+ return;
+ }
+ if (container.tagName) {
+ if (y == null) {
+ return {
+ container: container,
+ width: container.style.pixelWidth || container.offsetWidth,
+ height: container.style.pixelHeight || container.offsetHeight
+ };
+ } else {
+ return {
+ container: container,
+ width: y,
+ height: w
+ };
+ }
+ }
+ return {
+ container: 1,
+ x: x,
+ y: y,
+ width: w,
+ height: h
+ };
+ };
+
+ /*\
+ * Raphael.pathToRelative
+ [ method ]
+ **
+ * Utility method
+ **
+ * Converts path to relative form
+ > Parameters
+ - pathString (string|array) path string or array of segments
+ = (array) array of segments.
+ \*/
+ R.pathToRelative = pathToRelative;
+ R._engine = {};
+
+ /*\
+ * Raphael.path2curve
+ [ method ]
+ **
+ * Utility method
+ **
+ * Converts path to a new path where all segments are cubic bezier curves.
+ > Parameters
+ - pathString (string|array) path string or array of segments
+ = (array) array of segments.
+ \*/
+ R.path2curve = path2curve;
+
+ /*\
+ * Raphael.matrix
+ [ method ]
+ **
+ * Utility method
+ **
+ * Returns matrix based on given parameters.
+ > Parameters
+ - a (number)
+ - b (number)
+ - c (number)
+ - d (number)
+ - e (number)
+ - f (number)
+ = (object) @Matrix
+ \*/
+ R.matrix = function(a, b, c, d, e, f) {
+ return new Matrix(a, b, c, d, e, f);
+ };
+
+ function Matrix(a, b, c, d, e, f) {
+ if (a != null) {
+ this.a = +a;
+ this.b = +b;
+ this.c = +c;
+ this.d = +d;
+ this.e = +e;
+ this.f = +f;
+ } else {
+ this.a = 1;
+ this.b = 0;
+ this.c = 0;
+ this.d = 1;
+ this.e = 0;
+ this.f = 0;
+ }
+ }
+ (function(matrixproto) {
+
+ /*\
+ * Matrix.add
+ [ method ]
+ **
+ * Adds given matrix to existing one.
+ > Parameters
+ - a (number)
+ - b (number)
+ - c (number)
+ - d (number)
+ - e (number)
+ - f (number)
+ or
+ - matrix (object) @Matrix
+ \*/
+ matrixproto.add = function(a, b, c, d, e, f) {
+ var out = [[], [], []],
+ m = [[this.a, this.c, this.e], [this.b, this.d, this.f], [0, 0, 1]],
+ matrix = [[a, c, e], [b, d, f], [0, 0, 1]],
+ x, y, z, res;
+
+ if (a && a instanceof Matrix) {
+ matrix = [[a.a, a.c, a.e], [a.b, a.d, a.f], [0, 0, 1]];
+ }
+
+ for (x = 0; x < 3; x++) {
+ for (y = 0; y < 3; y++) {
+ res = 0;
+ for (z = 0; z < 3; z++) {
+ res += m[x][z] * matrix[z][y];
+ }
+ out[x][y] = res;
+ }
+ }
+ this.a = out[0][0];
+ this.b = out[1][0];
+ this.c = out[0][1];
+ this.d = out[1][1];
+ this.e = out[0][2];
+ this.f = out[1][2];
+ };
+
+ /*\
+ * Matrix.invert
+ [ method ]
+ **
+ * Returns inverted version of the matrix
+ = (object) @Matrix
+ \*/
+ matrixproto.invert = function() {
+ var me = this,
+ x = me.a * me.d - me.b * me.c;
+ return new Matrix(me.d / x, -me.b / x, -me.c / x, me.a / x, (me.c * me.f - me.d * me.e) / x, (me.b * me.e - me.a * me.f) / x);
+ };
+
+ /*\
+ * Matrix.clone
+ [ method ]
+ **
+ * Returns copy of the matrix
+ = (object) @Matrix
+ \*/
+ matrixproto.clone = function() {
+ return new Matrix(this.a, this.b, this.c, this.d, this.e, this.f);
+ };
+
+ /*\
+ * Matrix.translate
+ [ method ]
+ **
+ * Translate the matrix
+ > Parameters
+ - x (number)
+ - y (number)
+ \*/
+ matrixproto.translate = function(x, y) {
+ this.add(1, 0, 0, 1, x, y);
+ };
+
+ /*\
+ * Matrix.scale
+ [ method ]
+ **
+ * Scales the matrix
+ > Parameters
+ - x (number)
+ - y (number) #optional
+ - cx (number) #optional
+ - cy (number) #optional
+ \*/
+ matrixproto.scale = function(x, y, cx, cy) {
+ y == null && (y = x);
+ (cx || cy) && this.add(1, 0, 0, 1, cx, cy);
+ this.add(x, 0, 0, y, 0, 0);
+ (cx || cy) && this.add(1, 0, 0, 1, -cx, -cy);
+ };
+
+ /*\
+ * Matrix.rotate
+ [ method ]
+ **
+ * Rotates the matrix
+ > Parameters
+ - a (number)
+ - x (number)
+ - y (number)
+ \*/
+ matrixproto.rotate = function(a, x, y) {
+ a = R.rad(a);
+ x = x || 0;
+ y = y || 0;
+ var cos = +mathCos(a).toFixed(9),
+ sin = + mathSin(a).toFixed(9);
+ this.add(cos, sin, -sin, cos, x, y);
+ this.add(1, 0, 0, 1, -x, -y);
+ };
+
+ /*\
+ * Matrix.x
+ [ method ]
+ **
+ * Return x coordinate for given point after transformation described by the matrix. See also @Matrix.y
+ > Parameters
+ - x (number)
+ - y (number)
+ = (number) x
+ \*/
+ matrixproto.x = function(x, y) {
+ return x * this.a + y * this.c + this.e;
+ };
+
+ /*\
+ * Matrix.y
+ [ method ]
+ **
+ * Return y coordinate for given point after transformation described by the matrix. See also @Matrix.x
+ > Parameters
+ - x (number)
+ - y (number)
+ = (number) y
+ \*/
+ matrixproto.y = function(x, y) {
+ return x * this.b + y * this.d + this.f;
+ };
+ matrixproto.get = function(i) {
+ return +this[Str.fromCharCode(97 + i)].toFixed(4);
+ };
+ matrixproto.toString = function() {
+ return R.svg ?
+ "matrix(" + [this.get(0), this.get(1), this.get(2), this.get(3), this.get(4), this.get(5)].join() + ")" :
+ [this.get(0), this.get(2), this.get(1), this.get(3), 0, 0].join();
+ };
+ matrixproto.toMatrixString = function() {
+ return "matrix(" + [this.get(0), this.get(1), this.get(2), this.get(3), this.get(4), this.get(5)].join() + ")";
+ };
+ matrixproto.toFilter = function() {
+ return "progid:DXImageTransform.Microsoft.Matrix(M11=" + this.get(0) +
+ ", M12=" + this.get(2) + ", M21=" + this.get(1) + ", M22=" + this.get(3) +
+ ", Dx=" + this.get(4) + ", Dy=" + this.get(5) + ", sizingmethod='auto expand')";
+ };
+ matrixproto.offset = function() {
+ return [this.e.toFixed(4), this.f.toFixed(4)];
+ };
+ function norm(a) {
+ return a[0] * a[0] + a[1] * a[1];
+ }
+ function normalize(a) {
+ var mag = mathSqrt(norm(a));
+ a[0] && (a[0] /= mag);
+ a[1] && (a[1] /= mag);
+ }
+
+ /*\
+ * Matrix.split
+ [ method ]
+ **
+ * Splits matrix into primitive transformations
+ = (object) in format:
+ o dx (number) translation by x
+ o dy (number) translation by y
+ o scalex (number) scale by x
+ o scaley (number) scale by y
+ o shear (number) shear
+ o rotate (number) rotation in deg
+ o isSimple (boolean) could it be represented via simple transformations
+ \*/
+ matrixproto.split = function() {
+ var out = {};
+ // translation
+ out.dx = this.e;
+ out.dy = this.f;
+
+ // scale and shear
+ var row = [[this.a, this.c], [this.b, this.d]];
+ out.scalex = mathSqrt(norm(row[0]));
+ normalize(row[0]);
+
+ out.shear = row[0][0] * row[1][0] + row[0][1] * row[1][1];
+ row[1] = [row[1][0] - row[0][0] * out.shear, row[1][1] - row[0][1] * out.shear];
+
+ out.scaley = mathSqrt(norm(row[1]));
+ normalize(row[1]);
+ out.shear /= out.scaley;
+
+ // rotation
+ var sin = -row[0][1],
+ cos = row[1][1];
+ if (cos < 0) {
+ out.rotate = R.deg(math.acos(cos));
+ if (sin < 0) {
+ out.rotate = 360 - out.rotate;
+ }
+ } else {
+ out.rotate = R.deg(math.asin(sin));
+ }
+
+ out.isSimple = !+out.shear.toFixed(9) && (out.scalex.toFixed(9) == out.scaley.toFixed(9) || !out.rotate);
+ out.isSuperSimple = !+out.shear.toFixed(9) && out.scalex.toFixed(9) == out.scaley.toFixed(9) && !out.rotate;
+ out.noRotation = !+out.shear.toFixed(9) && !out.rotate;
+ return out;
+ };
+
+ /*\
+ * Matrix.toTransformString
+ [ method ]
+ **
+ * Return transform string that represents given matrix
+ = (string) transform string
+ \*/
+ matrixproto.toTransformString = function(shorter) {
+ var s = shorter || this[split]();
+ if (s.isSimple) {
+ s.scalex = +s.scalex.toFixed(4);
+ s.scaley = +s.scaley.toFixed(4);
+ s.rotate = +s.rotate.toFixed(4);
+ return (s.dx || s.dy ? "t" + [s.dx, s.dy] : E) +
+ (s.scalex != 1 || s.scaley != 1 ? "s" + [s.scalex, s.scaley, 0, 0] : E) +
+ (s.rotate ? "r" + [s.rotate, 0, 0] : E);
+ } else {
+ return "m" + [this.get(0), this.get(1), this.get(2), this.get(3), this.get(4), this.get(5)];
+ }
+ };
+ })(Matrix.prototype);
+
+ // WebKit rendering bug workaround method
+ var version = navigator.userAgent.match(/Version\/(.*?)\s/) || navigator.userAgent.match(/Chrome\/(\d+)/);
+ if ((navigator.vendor == "Apple Computer, Inc.") && (version && version[1] < 4 || navigator.platform.slice(0, 2) == "iP") ||
+ (navigator.vendor == "Google Inc." && version && version[1] < 8)) {
+
+ /*\
+ * Paper.safari
+ [ method ]
+ **
+ * There is an inconvenient rendering bug in Safari (WebKit):
+ * sometimes the rendering should be forced.
+ * This method should help with dealing with this bug.
+ \*/
+ paperproto.safari = function() {
+ var rect = this.rect(-99, -99, this.width + 99, this.height + 99).attr({
+ stroke: "none"
+ });
+ setTimeout(function() {
+ rect.remove();
+ });
+ return true;
+ };
+ } else {
+ paperproto.safari = fun;
+ }
+
+ var preventDefault = function() {
+ this.returnValue = false;
+ },
+ preventTouch = function() {
+ return this.originalEvent.preventDefault();
+ },
+ stopPropagation = function() {
+ this.cancelBubble = true;
+ },
+ stopTouch = function() {
+ return this.originalEvent.stopPropagation();
+ },
+ addEvent = R.addEvent = (function() {
+ if (g.doc.addEventListener) {
+ return function(obj, type, fn, element) {
+ var realName = supportsTouch && touchMap[type] ? touchMap[type] : type,
+ f = function(e) {
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft;
+ if (supportsTouch && touchMap[has](type)) {
+ for (var i = 0, ii = e.targetTouches && e.targetTouches.length; i < ii; i++) {
+ if (e.targetTouches[i].target == obj) {
+ var olde = e;
+ e = e.targetTouches[i];
+ e.originalEvent = olde;
+ e.preventDefault = preventTouch;
+ e.stopPropagation = stopTouch;
+ break;
+ }
+ }
+ }
+ return fn.call(element, e, e.clientX + scrollX, e.clientY + scrollY);
+ };
+ obj.addEventListener(realName, f, false);
+ return function() {
+ obj.removeEventListener(realName, f, false);
+ return true;
+ };
+ };
+ } else if (g.doc.attachEvent) {
+ return function(obj, type, fn, element) {
+ var f = function(e) {
+ e = e || g.win.event;
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft,
+ x = e.clientX + scrollX,
+ y = e.clientY + scrollY;
+ e.preventDefault = e.preventDefault || preventDefault;
+ e.stopPropagation = e.stopPropagation || stopPropagation;
+ return fn.call(element, e, x, y);
+ };
+ obj.attachEvent("on" + type, f);
+ var detacher = function() {
+ obj.detachEvent("on" + type, f);
+ return true;
+ };
+ return detacher;
+ };
+ }
+ })(),
+ drag = [],
+ dragMove = function(e) {
+ var x = e.clientX,
+ y = e.clientY,
+ scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft,
+ dragi,
+ j = drag.length;
+ while (j--) {
+ dragi = drag[j];
+ if (supportsTouch) {
+ var i = e.touches.length,
+ touch;
+ while (i--) {
+ touch = e.touches[i];
+ if (touch.identifier == dragi.el._drag.id) {
+ x = touch.clientX;
+ y = touch.clientY;
+ (e.originalEvent ? e.originalEvent : e).preventDefault();
+ break;
+ }
+ }
+ } else {
+ e.preventDefault();
+ }
+ var node = dragi.el.node,
+ o,
+ next = node.nextSibling,
+ parent = node.parentNode,
+ display = node.style.display;
+ g.win.opera && parent.removeChild(node);
+ node.style.display = "none";
+ o = dragi.el.paper.getElementByPoint(x, y);
+ node.style.display = display;
+ g.win.opera && (next ? parent.insertBefore(node, next) : parent.appendChild(node));
+ o && eve("raphael.drag.over." + dragi.el.id, dragi.el, o);
+ x += scrollX;
+ y += scrollY;
+ eve("raphael.drag.move." + dragi.el.id, dragi.move_scope || dragi.el, x - dragi.el._drag.x, y - dragi.el._drag.y, x, y, e);
+ }
+ },
+ dragUp = function(e) {
+ R.unmousemove(dragMove).unmouseup(dragUp);
+ var i = drag.length,
+ dragi;
+ while (i--) {
+ dragi = drag[i];
+ dragi.el._drag = {};
+ eve("raphael.drag.end." + dragi.el.id, dragi.end_scope || dragi.start_scope || dragi.move_scope || dragi.el, e);
+ }
+ drag = [];
+ },
+
+ /*\
+ * Raphael.el
+ [ property (object) ]
+ **
+ * You can add your own method to elements. This is usefull when you want to hack default functionality or
+ * want to wrap some common transformation or attributes in one method. In difference to canvas methods,
+ * you can redefine element method at any time. Expending element methods wouldn’t affect set.
+ > Usage
+ | Raphael.el.red = function () {
+ | this.attr({fill: "#f00"});
+ | };
+ | // then use it
+ | paper.circle(100, 100, 20).red();
+ \*/
+ elproto = R.el = {};
+
+ /*\
+ * Element.click
+ [ method ]
+ **
+ * Adds event handler for click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unclick
+ [ method ]
+ **
+ * Removes event handler for click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.dblclick
+ [ method ]
+ **
+ * Adds event handler for double click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.undblclick
+ [ method ]
+ **
+ * Removes event handler for double click for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mousedown
+ [ method ]
+ **
+ * Adds event handler for mousedown for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmousedown
+ [ method ]
+ **
+ * Removes event handler for mousedown for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mousemove
+ [ method ]
+ **
+ * Adds event handler for mousemove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmousemove
+ [ method ]
+ **
+ * Removes event handler for mousemove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mouseout
+ [ method ]
+ **
+ * Adds event handler for mouseout for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmouseout
+ [ method ]
+ **
+ * Removes event handler for mouseout for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mouseover
+ [ method ]
+ **
+ * Adds event handler for mouseover for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmouseover
+ [ method ]
+ **
+ * Removes event handler for mouseover for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.mouseup
+ [ method ]
+ **
+ * Adds event handler for mouseup for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.unmouseup
+ [ method ]
+ **
+ * Removes event handler for mouseup for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchstart
+ [ method ]
+ **
+ * Adds event handler for touchstart for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchstart
+ [ method ]
+ **
+ * Removes event handler for touchstart for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchmove
+ [ method ]
+ **
+ * Adds event handler for touchmove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchmove
+ [ method ]
+ **
+ * Removes event handler for touchmove for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchend
+ [ method ]
+ **
+ * Adds event handler for touchend for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchend
+ [ method ]
+ **
+ * Removes event handler for touchend for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+
+ /*\
+ * Element.touchcancel
+ [ method ]
+ **
+ * Adds event handler for touchcancel for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ /*\
+ * Element.untouchcancel
+ [ method ]
+ **
+ * Removes event handler for touchcancel for the element.
+ > Parameters
+ - handler (function) handler for the event
+ = (object) @Element
+ \*/
+ for (var i = events.length; i--; ) {
+ (function(eventName) {
+ R[eventName] = elproto[eventName] = function(fn, scope) {
+ if (R.is(fn, "function")) {
+ this.events = this.events || [];
+ this.events.push({
+ name: eventName,
+ f: fn,
+ unbind: addEvent(this.shape || this.node || g.doc, eventName, fn, scope || this)
+ });
+ }
+ return this;
+ };
+ R["un" + eventName] = elproto["un" + eventName] = function(fn) {
+ var events = this.events || [],
+ l = events.length;
+ while (l--)
+ if (events[l].name == eventName && events[l].f == fn) {
+ events[l].unbind();
+ events.splice(l, 1);
+ !events.length && delete this.events;
+ return this;
+ }
+ return this;
+ };
+ })(events[i]);
+ }
+
+ /*\
+ * Element.data
+ [ method ]
+ **
+ * Adds or retrieves given value asociated with given key.
+ **
+ * See also @Element.removeData
+ > Parameters
+ - key (string) key to store data
+ - value (any) #optional value to store
+ = (object) @Element
+ * or, if value is not specified:
+ = (any) value
+ > Usage
+ | for (var i = 0, i < 5, i++) {
+ | paper.circle(10 + 15 * i, 10, 10)
+ | .attr({fill: "#000"})
+ | .data("i", i)
+ | .click(function () {
+ | alert(this.data("i"));
+ | });
+ | }
+ \*/
+ elproto.data = function(key, value) {
+ var data = eldata[this.id] = eldata[this.id] || {};
+ if (arguments.length == 1) {
+ if (R.is(key, object)) {
+ for (var i in key)
+ if (key[has](i)) {
+ this.data(i, key[i]);
+ }
+ return this;
+ }
+ eve("raphael.data.get." + this.id, this, data[key], key);
+ return data[key];
+ }
+ data[key] = value;
+ eve("raphael.data.set." + this.id, this, value, key);
+ return this;
+ };
+
+ /*\
+ * Element.removeData
+ [ method ]
+ **
+ * Removes value associated with an element by given key.
+ * If key is not provided, removes all the data of the element.
+ > Parameters
+ - key (string) #optional key
+ = (object) @Element
+ \*/
+ elproto.removeData = function(key) {
+ if (key == null) {
+ eldata[this.id] = {};
+ } else {
+ eldata[this.id] && delete eldata[this.id][key];
+ }
+ return this;
+ };
+
+ /*\
+ * Element.getData
+ [ method ]
+ **
+ * Retrieves the element data
+ = (object) data
+ \*/
+ elproto.getData = function () {
+ return clone(eldata[this.id] || {});
+ };
+
+ var downables = [],
+ mouseDown = function () {
+ this.untrack = addEvent(g.doc, 'mouseup', mouseUp, this);
+ },
+ mouseUp = function () {
+ this.untrack();
+ this.untrack = null;
+ return this.fn && this.fn.apply(this.scope || this.el, arguments);
+
+ };
+ elproto.mouseup = function (fn, scope, track) {
+ if (!track) {
+ return R.mouseup.apply(this, arguments);
+ }
+ downables.push(track = {
+ el: this,
+ fn: fn,
+ scope: scope
+ });
+ track.unbind = addEvent(this.shape || this.node || g.doc,
+ 'mousedown', mouseDown, track);
+
+ return this;
+ };
+
+ elproto.unmouseup = function (fn) {
+ var i = downables.length,
+ undowned;
+ while (i--) {
+ if (downables[i].el === this && downables[i].fn === fn) {
+ undowned = downables[i];
+ undowned.unbind();
+ undowned.untrack && undowned.untrack();
+ downables.splice(i, 1);
+ }
+ }
+ return undowned ? this : R.unmouseup.apply(this, arguments);
+ };
+
+ /*\
+ * Element.hover
+ [ method ]
+ **
+ * Adds event handlers for hover for the element.
+ > Parameters
+ - f_in (function) handler for hover in
+ - f_out (function) handler for hover out
+ - icontext (object) #optional context for hover in handler
+ - ocontext (object) #optional context for hover out handler
+ = (object) @Element
+ \*/
+ elproto.hover = function(f_in, f_out, scope_in, scope_out) {
+ return this.mouseover(f_in, scope_in).mouseout(f_out, scope_out || scope_in);
+ };
+
+ /*\
+ * Element.unhover
+ [ method ]
+ **
+ * Removes event handlers for hover for the element.
+ > Parameters
+ - f_in (function) handler for hover in
+ - f_out (function) handler for hover out
+ = (object) @Element
+ \*/
+ elproto.unhover = function(f_in, f_out) {
+ return this.unmouseover(f_in).unmouseout(f_out);
+ };
+ var draggable = [];
+
+ /*\
+ * Element.drag
+ [ method ]
+ **
+ * Adds event handlers for drag of the element.
+ > Parameters
+ - onmove (function) handler for moving
+ - onstart (function) handler for drag start
+ - onend (function) handler for drag end
+ - mcontext (object) #optional context for moving handler
+ - scontext (object) #optional context for drag start handler
+ - econtext (object) #optional context for drag end handler
+ * Additionaly following `drag` events will be triggered: `drag.start.` on start,
+ * `drag.end.` on end and `drag.move.` on every move. When element will be dragged over another element
+ * `drag.over.` will be fired as well.
+ *
+ * Start event and start handler will be called in specified context or in context of the element with following parameters:
+ o x (number) x position of the mouse
+ o y (number) y position of the mouse
+ o event (object) DOM event object
+ * Move event and move handler will be called in specified context or in context of the element with following parameters:
+ o dx (number) shift by x from the start point
+ o dy (number) shift by y from the start point
+ o x (number) x position of the mouse
+ o y (number) y position of the mouse
+ o event (object) DOM event object
+ * End event and end handler will be called in specified context or in context of the element with following parameters:
+ o event (object) DOM event object
+ = (object) @Element
+ \*/
+ elproto.drag = function(onmove, onstart, onend, move_scope, start_scope, end_scope) {
+ function start(e) {
+ (e.originalEvent || e).preventDefault();
+ var scrollY = g.doc.documentElement.scrollTop || g.doc.body.scrollTop,
+ scrollX = g.doc.documentElement.scrollLeft || g.doc.body.scrollLeft;
+ this._drag.x = e.clientX + scrollX;
+ this._drag.y = e.clientY + scrollY;
+ this._drag.id = e.identifier;
+ !drag.length && R.mousemove(dragMove).mouseup(dragUp);
+ drag.push({
+ el: this,
+ move_scope: move_scope,
+ start_scope: start_scope,
+ end_scope: end_scope
+ });
+ onstart && eve.on("raphael.drag.start." + this.id, onstart);
+ onmove && eve.on("raphael.drag.move." + this.id, onmove);
+ onend && eve.on("raphael.drag.end." + this.id, onend);
+ eve("raphael.drag.start." + this.id, start_scope || move_scope || this, e.clientX + scrollX, e.clientY + scrollY, e);
+ }
+ this._drag = {};
+ draggable.push({
+ el: this,
+ start: start
+ });
+ this.mousedown(start);
+ return this;
+ };
+
+ /*\
+ * Element.onDragOver
+ [ method ]
+ **
+ * Shortcut for assigning event handler for `drag.over.` event, where id is id of the element (see @Element.id).
+ > Parameters
+ - f (function) handler for event, first argument would be the element you are dragging over
+ \*/
+ elproto.onDragOver = function(f) {
+ f ? eve.on("raphael.drag.over." + this.id, f) : eve.unbind("raphael.drag.over." + this.id);
+ };
+
+ /*\
+ * Element.undrag
+ [ method ]
+ **
+ * Removes all drag event handlers from given element.
+ \*/
+ elproto.undrag = function() {
+ var i = draggable.length;
+ while (i--)
+ if (draggable[i].el == this) {
+ this.unmousedown(draggable[i].start);
+ draggable.splice(i, 1);
+ eve.unbind("raphael.drag.*." + this.id);
+ }
+ !draggable.length && R.unmousemove(dragMove).unmouseup(dragUp);
+ };
+
+ elproto.follow = function(el, callback, stalk) {
+ if (el.removed || el.constructor !== R.el.constructor) {
+ return this;
+ }
+ el.followers.push({
+ el: this,
+ stalk: (stalk = {before: 'insertBefore', after: 'insertAfter'}[stalk]),
+ cb: callback
+ });
+
+ stalk && this[stalk](el);
+ return this;
+ };
+
+ elproto.unfollow = function(el) {
+ if (el.removed || el.constructor !== R.el.constructor) {
+ return this;
+ }
+ for (var i = 0, ii = el.followers.length; i < ii; i++) {
+ if (el.followers[i].el === this) {
+ el.followers.splice(i, 1);
+ break;
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Paper.hide
+ [ method ]
+ **
+ * Hides a paper
+ **
+ > Usage
+ | paper.hide();
+ \*/
+ paperproto.hide = function () {
+ var paper = this;
+ paper.canvas.style.visibility = "hidden";
+ return paper;
+ };
+
+ /*\
+ * Paper.show
+ [ method ]
+ **
+ * Shows a hidden paper
+ **
+ > Usage
+ | paper.show();
+ \*/
+ paperproto.show = function () {
+ var paper = this;
+ paper.canvas.style.visibility = E;
+ return paper;
+ };
+
+ /*\
+ * Paper.group
+ [ method ]
+ **
+ * Creates a group
+ **
+ > Parameters
+ **
+ - id (number) id of the group
+ = (object) Raphaël element object with type “group”
+ **
+ > Usage
+ | var g = paper.group();
+ \*/
+ paperproto.group = function () { // id
+ var paper = this,
+ out,
+ args = arguments,
+ group = lastArgIfGroup(args, true);
+
+ out = R._engine.group(paper, args[0], group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.circle
+ [ method ]
+ **
+ * Draws a circle.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the centre
+ - y (number) y coordinate of the centre
+ - r (number) radius
+ = (object) Raphaël element object with type “circle”
+ **
+ > Usage
+ | var c = paper.circle(50, 50, 40);
+ \*/
+ paperproto.circle = function () { // x, y, r
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.circle(paper, args[0] || 0, args[1] || 0,
+ args[2] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+
+ /*\
+ * Paper.rect
+ [ method ]
+ *
+ * Draws a rectangle.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the top left corner
+ - y (number) y coordinate of the top left corner
+ - width (number) width
+ - height (number) height
+ - r (number) #optional radius for rounded corners, default is 0
+ = (object) Raphaël element object with type “rect”
+ **
+ > Usage
+ | // regular rectangle
+ | var c = paper.rect(10, 10, 50, 50);
+ | // rectangle with rounded corners
+ | var c = paper.rect(40, 40, 50, 50, 10);
+ \*/
+ paperproto.rect = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.rect(paper, args[0] || 0, args[1] || 0, args[2] || 0,
+ args[3] || 0, args[4] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.ellipse
+ [ method ]
+ **
+ * Draws an ellipse.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the centre
+ - y (number) y coordinate of the centre
+ - rx (number) horizontal radius
+ - ry (number) vertical radius
+ = (object) Raphaël element object with type “ellipse”
+ **
+ > Usage
+ | var c = paper.ellipse(50, 50, 40, 20);
+ \*/
+ paperproto.ellipse = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.ellipse(this, args[0] || 0, args[1] || 0,
+ args[2] || 0, args[3] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.path
+ [ method ]
+ **
+ * Creates a path element by given path data string.
+ > Parameters
+ - pathString (string) #optional path string in SVG format.
+ * Path string consists of one-letter commands, followed by comma seprarated arguments in numercal form. Example:
+ | "M10,20L30,40"
+ * Here we can see two commands: “M”, with arguments `(10, 20)` and “L” with arguments `(30, 40)`. Upper case letter mean command is absolute, lower case—relative.
+ *
+ # Here is short list of commands available, for more details see SVG path string format .
+ # Command Name Parameters
+ # M moveto (x y)+
+ # Z closepath (none)
+ # L lineto (x y)+
+ # H horizontal lineto x+
+ # V vertical lineto y+
+ # C curveto (x1 y1 x2 y2 x y)+
+ # S smooth curveto (x2 y2 x y)+
+ # Q quadratic Bézier curveto (x1 y1 x y)+
+ # T smooth quadratic Bézier curveto (x y)+
+ # A elliptical arc (rx ry x-axis-rotation large-arc-flag sweep-flag x y)+
+ # R Catmull-Rom curveto *x1 y1 (x y)+
+ * * “Catmull-Rom curveto” is a not standard SVG command and added in 2.0 to make life easier.
+ * Note: there is a special case when path consist of just three commands: “M10,10R…z”. In this case path will smoothly connects to its beginning.
+ > Usage
+ | var c = paper.path("M10 10L90 90");
+ | // draw a diagonal line:
+ | // move to 10,10, line to 90,90
+ * For example of path strings, check out these icons: http://raphaeljs.com/icons/
+ \*/
+ paperproto.path = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ pathString,
+ out;
+
+ pathString = args[0];
+ pathString && !R.is(pathString, string) &&
+ !R.is(pathString[0], array) && (pathString += E);
+
+ out = R._engine.path(R.format[apply](R, arguments), paper, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.image
+ [ method ]
+ **
+ * Embeds an image into the surface.
+ **
+ > Parameters
+ **
+ - src (string) URI of the source image
+ - x (number) x coordinate position
+ - y (number) y coordinate position
+ - width (number) width of the image
+ - height (number) height of the image
+ = (object) Raphaël element object with type “image”
+ **
+ > Usage
+ | var c = paper.image("apple.png", 10, 10, 80, 80);
+ \*/
+ paperproto.image = function () {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.image(paper, args[0] || "about:blank", args[1] || 0,
+ args[2] || 0, args[3] || 0, args[4] || 0, group);
+
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.text
+ [ method ]
+ **
+ * Draws a text string. If you need line breaks, put “\n” in the string.
+ **
+ > Parameters
+ **
+ - x (number) x coordinate position
+ - y (number) y coordinate position
+ - text (string) The text string to draw
+ = (object) Raphaël element object with type “text”
+ **
+ > Usage
+ | var t = paper.text(50, 50, "Raphaël\nkicks\nbutt!");
+ \*/
+ paperproto.text = function() {
+ var paper = this,
+ args = arguments,
+ group = lastArgIfGroup(args, true),
+ out;
+
+ out = R._engine.text(paper, args[0] || 0, args[1] || 0, Str(args[2] || E),
+ group);
+ paper.__set__ && paper.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.set
+ [ method ]
+ **
+ * Creates array-like object to keep and operate several elements at once.
+ * Warning: it doesn’t create any elements for itself in the page, it just groups existing elements.
+ * Sets act as pseudo elements — all methods available to an element can be used on a set.
+ = (object) array-like object that represents set of elements
+ **
+ > Usage
+ | var st = paper.set();
+ | st.push(
+ | paper.circle(10, 10, 5),
+ | paper.circle(30, 10, 5)
+ | );
+ | st.attr({fill: "red"}); // changes the fill of both circles
+ \*/
+ paperproto.set = function(itemsArray) {
+ !R.is(itemsArray, "array") && (itemsArray = arraySplice.call(arguments, 0, arguments.length));
+ var out = new Set(itemsArray);
+ this.__set__ && this.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Paper.setStart
+ [ method ]
+ **
+ * Creates @Paper.set. All elements that will be created after calling this method and before calling
+ * @Paper.setFinish will be added to the set.
+ **
+ > Usage
+ | paper.setStart();
+ | paper.circle(10, 10, 5),
+ | paper.circle(30, 10, 5)
+ | var st = paper.setFinish();
+ | st.attr({fill: "red"}); // changes the fill of both circles
+ \*/
+ paperproto.setStart = function(set) {
+ this.__set__ = set || this.set();
+ };
+
+ /*\
+ * Paper.setFinish
+ [ method ]
+ **
+ * See @Paper.setStart. This method finishes catching and returns resulting set.
+ **
+ = (object) set
+ \*/
+ paperproto.setFinish = function(set) {
+ var out = this.__set__;
+ delete this.__set__;
+ return out;
+ };
+
+ /*\
+ * Paper.setSize
+ [ method ]
+ **
+ * If you need to change dimensions of the canvas call this method
+ **
+ > Parameters
+ **
+ - width (number) new width of the canvas
+ - height (number) new height of the canvas
+ \*/
+ paperproto.setSize = function(width, height) {
+ return R._engine.setSize.call(this, width, height);
+ };
+
+ /*\
+ * Paper.setViewBox
+ [ method ]
+ **
+ * Sets the view box of the paper. Practically it gives you ability to zoom and pan whole paper surface by
+ * specifying new boundaries.
+ **
+ > Parameters
+ **
+ - x (number) new x position, default is `0`
+ - y (number) new y position, default is `0`
+ - w (number) new width of the canvas
+ - h (number) new height of the canvas
+ - fit (boolean) `true` if you want graphics to fit into new boundary box
+ \*/
+ paperproto.setViewBox = function(x, y, w, h, fit) {
+ return R._engine.setViewBox.call(this, x, y, w, h, fit);
+ };
+
+ /*\
+ * Paper.top
+ [ property ]
+ **
+ * Points to the topmost element on the paper
+ \*/
+ /*\
+ * Paper.bottom
+ [ property ]
+ **
+ * Points to the bottom element on the paper
+ \*/
+ paperproto.top = paperproto.bottom = null;
+
+ /*\
+ * Paper.raphael
+ [ property ]
+ **
+ * Points to the @Raphael object/function
+ \*/
+ paperproto.raphael = R;
+
+ var getOffset = function(elem) {
+ var box = elem.getBoundingClientRect(),
+ doc = elem.ownerDocument,
+ body = doc.body,
+ docElem = doc.documentElement,
+ clientTop = docElem.clientTop || body.clientTop || 0, clientLeft = docElem.clientLeft || body.clientLeft || 0,
+ top = box.top + (g.win.pageYOffset || docElem.scrollTop || body.scrollTop) - clientTop,
+ left = box.left + (g.win.pageXOffset || docElem.scrollLeft || body.scrollLeft) - clientLeft;
+ return {
+ y: top,
+ x: left
+ };
+ };
+
+ /*\
+ * Paper.getElementByPoint
+ [ method ]
+ **
+ * Returns you topmost element under given point.
+ **
+ = (object) Raphaël element object
+ > Parameters
+ **
+ - x (number) x coordinate from the top left corner of the window
+ - y (number) y coordinate from the top left corner of the window
+ > Usage
+ | paper.getElementByPoint(mouseX, mouseY).attr({stroke: "#f00"});
+ \*/
+ paperproto.getElementByPoint = function(x, y) {
+ var paper = this,
+ svg = paper.canvas,
+ target = g.doc.elementFromPoint(x, y);
+ if (g.win.opera && target.tagName == "svg") {
+ var so = getOffset(svg),
+ sr = svg.createSVGRect();
+ sr.x = x - so.x;
+ sr.y = y - so.y;
+ sr.width = sr.height = 1;
+ var hits = svg.getIntersectionList(sr, null);
+ if (hits.length) {
+ target = hits[hits.length - 1];
+ }
+ }
+ if (!target) {
+ return null;
+ }
+ while (target.parentNode && target != svg.parentNode && !target.raphael) {
+ target = target.parentNode;
+ }
+ target == paper.canvas.parentNode && (target = svg);
+ target = target && target.raphael ? paper.getById(target.raphaelid) : null;
+ return target;
+ };
+
+ /*\
+ * Paper.getElementsByBBox
+ [ method ]
+ **
+ * Returns set of elements that have an intersecting bounding box
+ **
+ > Parameters
+ **
+ - bbox (object) bbox to check with
+ = (object) @Set
+ \*/
+ paperproto.getElementsByBBox = function (bbox) {
+ var set = this.set();
+ this.forEach(function (el) {
+ if (R.isBBoxIntersect(el.getBBox(), bbox)) {
+ set.push(el);
+ }
+ });
+ return set;
+ };
+
+ paperproto.getById = function(id) {
+ var bot = this.bottom;
+ while (bot) {
+ if (bot.id == id) {
+ return bot;
+ }
+ bot = bot.next;
+ }
+ return null;
+ };
+
+ /*\
+ * Paper.forEach
+ [ method ]
+ **
+ * Executes given function for each element on the paper
+ *
+ * If callback function returns `false` it will stop loop running.
+ **
+ > Parameters
+ **
+ - callback (function) function to run
+ - thisArg (object) context object for the callback
+ = (object) Paper object
+ > Usage
+ | paper.forEach(function (el) {
+ | el.attr({ stroke: "blue" });
+ | });
+ \*/
+ paperproto.forEach = function(callback, thisArg) {
+ var bot = this.bottom;
+ while (bot) {
+ if (callback.call(thisArg, bot) === false) {
+ return this;
+ }
+ bot = bot.next;
+ }
+ return this;
+ };
+
+ /*\
+ * Paper.getElementsByPoint
+ [ method ]
+ **
+ * Returns set of elements that have common point inside
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the point
+ - y (number) y coordinate of the point
+ = (object) @Set
+ \*/
+ paperproto.getElementsByPoint = function(x, y) {
+ var set = this.set();
+ this.forEach(function(el) {
+ if (el.isPointInside(x, y)) {
+ set.push(el);
+ }
+ });
+ return set;
+ };
+ function x_y() {
+ return this.x + S + this.y;
+ }
+ function x_y_w_h() {
+ return this.x + S + this.y + S + this.width + " \xd7 " + this.height;
+ }
+
+ /*\
+ * Element.isPointInside
+ [ method ]
+ **
+ * Determine if given point is inside this element’s shape
+ **
+ > Parameters
+ **
+ - x (number) x coordinate of the point
+ - y (number) y coordinate of the point
+ = (boolean) `true` if point inside the shape
+ \*/
+ elproto.isPointInside = function(x, y) {
+ var rp = this.realPath = this.realPath || getPath[this.type](this),
+ tr;
+ return R.isPointInsidePath(((tr = this.attr('transform')) &&
+ tr.length && R.transformPath(rp, tr)) || rp, x, y);
+ };
+
+ /*\
+ * Element.getBBox
+ [ method ]
+ **
+ * Return bounding box for a given element
+ **
+ > Parameters
+ **
+ - isWithoutTransform (boolean) flag, `true` if you want to have bounding box before transformations. Default is `false`.
+ = (object) Bounding box object:
+ o {
+ o x: (number) top left corner x
+ o y: (number) top left corner y
+ o x2: (number) bottom right corner x
+ o y2: (number) bottom right corner y
+ o width: (number) width
+ o height: (number) height
+ o }
+ \*/
+ elproto.getBBox = function(isWithoutTransform) {
+ if (this.removed) {
+ return {};
+ }
+ var _ = this._;
+ if (isWithoutTransform) {
+ if (_.dirty || !_.bboxwt) {
+ this.realPath = getPath[this.type](this);
+ _.bboxwt = pathDimensions(this.realPath);
+ _.bboxwt.toString = x_y_w_h;
+ _.dirty = 0;
+ }
+ return _.bboxwt;
+ }
+ if (_.dirty || _.dirtyT || !_.bbox) {
+ if (_.dirty || !this.realPath) {
+ _.bboxwt = 0;
+ this.realPath = getPath[this.type](this);
+ }
+ _.bbox = pathDimensions(mapPath(this.realPath, this.matrix));
+ _.bbox.toString = x_y_w_h;
+ _.dirty = _.dirtyT = 0;
+ }
+ return _.bbox;
+ };
+
+ /*\
+ * Element.clone
+ [ method ]
+ **
+ = (object) clone of a given element
+ **
+ \*/
+ elproto.clone = function() {
+ if (this.removed) {
+ return null;
+ }
+ var o = this,
+ out = o.paper[o.type]().attr(o.attr());
+ o.__set__ && o.__set__.push(out);
+ return out;
+ };
+
+ /*\
+ * Element.glow
+ [ method ]
+ **
+ * Return set of elements that create glow-like effect around given element. See @Paper.set.
+ *
+ * Note: Glow is not connected to the element. If you change element attributes it won’t adjust itself.
+ **
+ > Parameters
+ **
+ - glow (object) #optional parameters object with all properties optional:
+ o {
+ o width (number) size of the glow, default is `10`
+ o fill (boolean) will it be filled, default is `false`
+ o opacity (number) opacity, default is `0.5`
+ o offsetx (number) horizontal offset, default is `0`
+ o offsety (number) vertical offset, default is `0`
+ o color (string) glow colour, default is `black`
+ o }
+ = (object) @Paper.set of elements that represents glow
+ \*/
+ elproto.glow = function(glow) {
+ if (this.type == "text") {
+ return null;
+ }
+ glow = glow || {};
+ var s = {
+ width: (glow.width || 10) + (+this.attr("stroke-width") || 1),
+ fill: glow.fill || false,
+ opacity: glow.opacity || .5,
+ offsetx: glow.offsetx || 0,
+ offsety: glow.offsety || 0,
+ color: glow.color || "#000"
+ },
+ c = s.width / 2,
+ r = this.paper,
+ out = r.set(),
+ path = this.realPath || getPath[this.type](this);
+ path = this.matrix ? mapPath(path, this.matrix) : path;
+ for (var i = 1; i < c + 1; i++) {
+ out.push(r.path(path).attr({
+ stroke: s.color,
+ fill: s.fill ? s.color : "none",
+ "stroke-linejoin": "round",
+ "stroke-linecap": "round",
+ "stroke-width": +(s.width / c * i).toFixed(3),
+ opacity: +(s.opacity / c).toFixed(3)
+ }));
+ }
+ return out.insertBefore(this).translate(s.offsetx, s.offsety);
+ };
+ var curveslengths = {},
+ getPointAtSegmentLength = function(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, length) {
+ if (length == null) {
+ return bezlen(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y);
+ } else {
+ return R.findDotsAtSegment(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, getTatLen(p1x, p1y, c1x, c1y, c2x, c2y, p2x, p2y, length));
+ }
+ },
+ getLengthFactory = function(istotal, subpath) {
+ return function(path, length, onlystart) {
+ path = path2curve(path);
+ var x, y, p, l, sp = "", subpaths = {}, point,
+ len = 0;
+ for (var i = 0, ii = path.length; i < ii; i++) {
+ p = path[i];
+ if (p[0] == "M") {
+ x = +p[1];
+ y = +p[2];
+ } else {
+ l = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6]);
+ if (len + l > length) {
+ if (subpath && !subpaths.start) {
+ point = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6], length - len);
+ sp += ["C" + point.start.x, point.start.y, point.m.x, point.m.y, point.x, point.y];
+ if (onlystart) {
+ return sp;
+ }
+ subpaths.start = sp;
+ sp = ["M" + point.x, point.y + "C" + point.n.x, point.n.y, point.end.x, point.end.y, p[5], p[6]].join();
+ len += l;
+ x = +p[5];
+ y = +p[6];
+ continue;
+ }
+ if (!istotal && !subpath) {
+ point = getPointAtSegmentLength(x, y, p[1], p[2], p[3], p[4], p[5], p[6], length - len);
+ return {
+ x: point.x,
+ y: point.y,
+ alpha: point.alpha
+ };
+ }
+ }
+ len += l;
+ x = +p[5];
+ y = +p[6];
+ }
+ sp += p.shift() + p;
+ }
+ subpaths.end = sp;
+ point = istotal ? len : subpath ? subpaths : R.findDotsAtSegment(x, y, p[0], p[1], p[2], p[3], p[4], p[5], 1);
+ point.alpha && (point = {
+ x: point.x,
+ y: point.y,
+ alpha: point.alpha
+ });
+ return point;
+ };
+ };
+ var getTotalLength = getLengthFactory(1),
+ getPointAtLength = getLengthFactory(),
+ getSubpathsAtLength = getLengthFactory(0, 1);
+
+ R.getTotalLength = getTotalLength;
+
+ R.getPointAtLength = getPointAtLength;
+
+ R.getSubpath = function(path, from, to) {
+ if (this.getTotalLength(path) - to < 1e-6) {
+ return getSubpathsAtLength(path, from).end;
+ }
+ var a = getSubpathsAtLength(path, to, 1);
+ return from ? getSubpathsAtLength(a, from).end : a;
+ };
+
+ /*\
+ * Raphael.getTotalLength
+ [ method ]
+ **
+ * Returns length of the given path in pixels.
+ **
+ > Parameters
+ **
+ - path (string) SVG path string.
+ **
+ = (number) length.
+ \*/
+ elproto.getTotalLength = function() {
+ if (this.type != "path") {
+ return;
+ }
+ if (this.node.getTotalLength) {
+ return this.node.getTotalLength();
+ }
+ return getTotalLength(this.attrs.path);
+ };
+
+ /*\
+ * Raphael.getPointAtLength
+ [ method ]
+ **
+ * Return coordinates of the point located at the given length on the given path.
+ **
+ > Parameters
+ **
+ - path (string) SVG path string
+ - length (number)
+ **
+ = (object) representation of the point:
+ o {
+ o x: (number) x coordinate
+ o y: (number) y coordinate
+ o alpha: (number) angle of derivative
+ o }
+ \*/
+ elproto.getPointAtLength = function(length) {
+ if (this.type != "path") {
+ return;
+ }
+ return getPointAtLength(this.attrs.path, length);
+ };
+
+ /*\
+ * Raphael.getSubpath
+ [ method ]
+ **
+ * Return subpath of a given path from given length to given length.
+ **
+ > Parameters
+ **
+ - path (string) SVG path string
+ - from (number) position of the start of the segment
+ - to (number) position of the end of the segment
+ **
+ = (string) pathstring for the segment
+ \*/
+ elproto.getSubpath = function(from, to) {
+ if (this.type != "path") {
+ return;
+ }
+ return R.getSubpath(this.attrs.path, from, to);
+ };
+
+ /*\
+ * Raphael.easing_formulas
+ [ property ]
+ **
+ * Object that contains easing formulas for animation. You could extend it with your own. By default it has following list of easing:
+ #
+ # “linear”
+ # “<” or “easeIn” or “ease-in”
+ # “>” or “easeOut” or “ease-out”
+ # “<>” or “easeInOut” or “ease-in-out”
+ # “backIn” or “back-in”
+ # “backOut” or “back-out”
+ # “elastic”
+ # “bounce”
+ #
+ # See also Easing demo .
+ \*/
+ var ef = R.easing_formulas = {
+ linear: function(n) {
+ return n;
+ },
+ "<": function(n) {
+ return pow(n, 1.7);
+ },
+ ">": function(n) {
+ return pow(n, .48);
+ },
+ "<>": function(n) {
+ var q = .48 - n / 1.04,
+ Q = mathSqrt(.1734 + q * q),
+ x = Q - q,
+ X = pow(abs(x), 1 / 3) * (x < 0 ? -1 : 1),
+ y = -Q - q,
+ Y = pow(abs(y), 1 / 3) * (y < 0 ? -1 : 1),
+ t = X + Y + .5;
+ return (1 - t) * 3 * t * t + t * t * t;
+ },
+ backIn: function(n) {
+ var s = 1.70158;
+ return n * n * ((s + 1) * n - s);
+ },
+ backOut: function(n) {
+ n = n - 1;
+ var s = 1.70158;
+ return n * n * ((s + 1) * n + s) + 1;
+ },
+ elastic: function(n) {
+ if (n == !!n) {
+ return n;
+ }
+ return pow(2, -10 * n) * mathSin((n - .075) * (2 * PI) / .3) + 1;
+ },
+ bounce: function(n) {
+ var s = 7.5625,
+ p = 2.75,
+ l;
+ if (n < (1 / p)) {
+ l = s * n * n;
+ } else {
+ if (n < (2 / p)) {
+ n -= (1.5 / p);
+ l = s * n * n + .75;
+ } else {
+ if (n < (2.5 / p)) {
+ n -= (2.25 / p);
+ l = s * n * n + .9375;
+ } else {
+ n -= (2.625 / p);
+ l = s * n * n + .984375;
+ }
+ }
+ }
+ return l;
+ }
+ };
+ ef.easeIn = ef["ease-in"] = ef["<"];
+ ef.easeOut = ef["ease-out"] = ef[">"];
+ ef.easeInOut = ef["ease-in-out"] = ef["<>"];
+ ef["back-in"] = ef.backIn;
+ ef["back-out"] = ef.backOut;
+
+ var animationElements = [],
+ requestAnimFrame = window.requestAnimationFrame ||
+ window.webkitRequestAnimationFrame ||
+ window.mozRequestAnimationFrame ||
+ window.oRequestAnimationFrame ||
+ window.msRequestAnimationFrame ||
+ function(callback) {
+ setTimeout(callback, 16);
+ },
+ animation = function() {
+ var Now = +new Date,
+ l = 0;
+ for (; l < animationElements.length; l++) {
+ var e = animationElements[l];
+ if (e.el.removed || e.paused) {
+ continue;
+ }
+ var time = Now - e.start,
+ ms = e.ms,
+ easing = e.easing,
+ from = e.from,
+ diff = e.diff,
+ to = e.to,
+ t = e.t,
+ that = e.el,
+ set = {},
+ now,
+ init = {},
+ key;
+ if (e.initstatus) {
+ time = (e.initstatus * e.anim.top - e.prev) / (e.percent - e.prev) * ms;
+ e.status = e.initstatus;
+ delete e.initstatus;
+ e.stop && animationElements.splice(l--, 1);
+ } else {
+ e.status = (e.prev + (e.percent - e.prev) * (time / ms)) / e.anim.top;
+ }
+ if (time < 0) {
+ continue;
+ }
+ if (time < ms) {
+ var pos = easing(time / ms);
+ for (var attr in from)
+ if (from[has](attr)) {
+ switch (availableAnimAttrs[attr]) {
+ case nu:
+ now = +from[attr] + pos * ms * diff[attr];
+ break;
+ case "colour":
+ now = "rgb(" + [
+ upto255(round(from[attr].r + pos * ms * diff[attr].r)),
+ upto255(round(from[attr].g + pos * ms * diff[attr].g)),
+ upto255(round(from[attr].b + pos * ms * diff[attr].b))
+ ].join(",") + ")";
+ break;
+ case "path":
+ now = [];
+ for (var i = 0, ii = from[attr].length; i < ii; i++) {
+ now[i] = [from[attr][i][0]];
+ for (var j = 1, jj = from[attr][i].length; j < jj; j++) {
+ now[i][j] = (+from[attr][i][j] + pos * ms * diff[attr][i][j]).toFixed(4);
+ }
+ now[i] = now[i].join(S);
+ }
+ now = now.join(S);
+ break;
+ case "transform":
+ if (diff[attr].real) {
+ now = [];
+ for (i = 0, ii = from[attr].length; i < ii; i++) {
+ now[i] = [from[attr][i][0]];
+ for (j = 1, jj = from[attr][i].length; j < jj; j++) {
+ now[i][j] = from[attr][i][j] + pos * ms * diff[attr][i][j];
+ }
+ }
+ } else {
+ var get = function(i) {
+ return +from[attr][i] + pos * ms * diff[attr][i];
+ };
+ // now = [["r", get(2), 0, 0], ["t", get(3), get(4)], ["s", get(0), get(1), 0, 0]];
+ now = [["m", get(0), get(1), get(2), get(3), get(4), get(5)]];
+ }
+ break;
+ case "csv":
+ if (attr == "clip-rect") {
+ now = [];
+ i = 4;
+ while (i--) {
+ now[i] = +from[attr][i] + pos * ms * diff[attr][i];
+ }
+ }
+ break;
+ default:
+ var from2 = [][concat](from[attr]);
+ now = [];
+ i = that.ca[attr].length;
+ while (i--) {
+ now[i] = +from2[i] + pos * ms * diff[attr][i];
+ }
+ break;
+ }
+ set[attr] = now;
+ }
+ that.attr(set);
+ (function(id, that, anim) {
+ setTimeout(function() {
+ eve("raphael.anim.frame." + id, that, anim);
+ });
+ })(that.id, that, e.anim);
+ } else {
+ (function(f, el, a) {
+ setTimeout(function() {
+ eve("raphael.anim.frame." + el.id, el, a);
+ eve("raphael.anim.finish." + el.id, el, a);
+ R.is(f, "function") && f.call(el);
+ });
+ })(e.callback, that, e.anim);
+ that.attr(to);
+ animationElements.splice(l--, 1);
+ if (e.repeat > 1 && !e.next) {
+ for (key in to)
+ if (to[has](key)) {
+ init[key] = e.totalOrigin[key];
+ }
+ e.el.attr(init);
+ runAnimation(e.anim, e.el, e.anim.percents[0], null, e.totalOrigin, e.repeat - 1);
+ }
+ if (e.next && !e.stop) {
+ runAnimation(e.anim, e.el, e.next, null, e.totalOrigin, e.repeat);
+ }
+ }
+ }
+ R.svg && that && that.paper && that.paper.safari();
+ animationElements.length && requestAnimFrame(animation);
+ },
+ upto255 = function(color) {
+ return color > 255 ? 255 : color < 0 ? 0 : color;
+ };
+
+ /*\
+ * Element.animateWith
+ [ method ]
+ **
+ * Acts similar to @Element.animate, but ensure that given animation runs in sync with another given element.
+ **
+ > Parameters
+ **
+ - el (object) element to sync with
+ - anim (object) animation to sync with
+ - params (object) #optional final attributes for the element, see also @Element.attr
+ - ms (number) #optional number of milliseconds for animation to run
+ - easing (string) #optional easing type. Accept on of @Raphael.easing_formulas or CSS format: `cubic‐bezier(XX, XX, XX, XX)`
+ - callback (function) #optional callback function. Will be called at the end of animation.
+ * or
+ - element (object) element to sync with
+ - anim (object) animation to sync with
+ - animation (object) #optional animation object, see @Raphael.animation
+ **
+ = (object) original element
+ \*/
+ elproto.animateWith = function(el, anim, params, ms, easing, callback) {
+ var element = this;
+ if (element.removed) {
+ callback && callback.call(element);
+ return element;
+ }
+ var a = params instanceof Animation ? params : R.animation(params, ms, easing, callback),
+ x, y;
+ runAnimation(a, element, a.percents[0], null, element.attr());
+ for (var i = 0, ii = animationElements.length; i < ii; i++) {
+ if (animationElements[i].anim == anim && animationElements[i].el == el) {
+ animationElements[ii - 1].start = animationElements[i].start;
+ break;
+ }
+ }
+ return element;
+ //
+ //
+ // var a = params ? R.animation(params, ms, easing, callback) : anim,
+ // status = element.status(anim);
+ // return this.animate(a).status(a, status * anim.ms / a.ms);
+ };
+ function CubicBezierAtTime(t, p1x, p1y, p2x, p2y, duration) {
+ var cx = 3 * p1x,
+ bx = 3 * (p2x - p1x) - cx,
+ ax = 1 - cx - bx,
+ cy = 3 * p1y,
+ by = 3 * (p2y - p1y) - cy,
+ ay = 1 - cy - by;
+ function sampleCurveX(t) {
+ return ((ax * t + bx) * t + cx) * t;
+ }
+ function solve(x, epsilon) {
+ var t = solveCurveX(x, epsilon);
+ return ((ay * t + by) * t + cy) * t;
+ }
+ function solveCurveX(x, epsilon) {
+ var t0, t1, t2, x2, d2, i;
+ for (t2 = x, i = 0; i < 8; i++) {
+ x2 = sampleCurveX(t2) - x;
+ if (abs(x2) < epsilon) {
+ return t2;
+ }
+ d2 = (3 * ax * t2 + 2 * bx) * t2 + cx;
+ if (abs(d2) < 1e-6) {
+ break;
+ }
+ t2 = t2 - x2 / d2;
+ }
+ t0 = 0;
+ t1 = 1;
+ t2 = x;
+ if (t2 < t0) {
+ return t0;
+ }
+ if (t2 > t1) {
+ return t1;
+ }
+ while (t0 < t1) {
+ x2 = sampleCurveX(t2);
+ if (abs(x2 - x) < epsilon) {
+ return t2;
+ }
+ if (x > x2) {
+ t0 = t2;
+ } else {
+ t1 = t2;
+ }
+ t2 = (t1 - t0) / 2 + t0;
+ }
+ return t2;
+ }
+ return solve(t, 1 / (200 * duration));
+ }
+ elproto.onAnimation = function(f) {
+ f ? eve.on("raphael.anim.frame." + this.id, f) : eve.unbind("raphael.anim.frame." + this.id);
+ return this;
+ };
+ function Animation(anim, ms) {
+ var percents = [],
+ newAnim = {};
+ this.ms = ms;
+ this.times = 1;
+ if (anim) {
+ for (var attr in anim)
+ if (anim[has](attr)) {
+ newAnim[toFloat(attr)] = anim[attr];
+ percents.push(toFloat(attr));
+ }
+ percents.sort(sortByNumber);
+ }
+ this.anim = newAnim;
+ this.top = percents[percents.length - 1];
+ this.percents = percents;
+ }
+
+ /*\
+ * Animation.delay
+ [ method ]
+ **
+ * Creates a copy of existing animation object with given delay.
+ **
+ > Parameters
+ **
+ - delay (number) number of ms to pass between animation start and actual animation
+ **
+ = (object) new altered Animation object
+ | var anim = Raphael.animation({cx: 10, cy: 20}, 2e3);
+ | circle1.animate(anim); // run the given animation immediately
+ | circle2.animate(anim.delay(500)); // run the given animation after 500 ms
+ \*/
+ Animation.prototype.delay = function(delay) {
+ var a = new Animation(this.anim, this.ms);
+ a.times = this.times;
+ a.del = +delay || 0;
+ return a;
+ };
+
+ /*\
+ * Animation.repeat
+ [ method ]
+ **
+ * Creates a copy of existing animation object with given repetition.
+ **
+ > Parameters
+ **
+ - repeat (number) number iterations of animation. For infinite animation pass `Infinity`
+ **
+ = (object) new altered Animation object
+ \*/
+ Animation.prototype.repeat = function(times) {
+ var a = new Animation(this.anim, this.ms);
+ a.del = this.del;
+ a.times = math.floor(mmax(times, 0)) || 1;
+ return a;
+ };
+ function runAnimation(anim, element, percent, status, totalOrigin, times) {
+ percent = toFloat(percent);
+ var params,
+ isInAnim,
+ isInAnimSet,
+ percents = [],
+ next,
+ prev,
+ timestamp,
+ ms = anim.ms,
+ from = {},
+ to = {},
+ diff = {};
+ if (status) {
+ for (i = 0, ii = animationElements.length; i < ii; i++) {
+ var e = animationElements[i];
+ if (e.el.id == element.id && e.anim == anim) {
+ if (e.percent != percent) {
+ animationElements.splice(i, 1);
+ isInAnimSet = 1;
+ } else {
+ isInAnim = e;
+ }
+ element.attr(e.totalOrigin);
+ break;
+ }
+ }
+ } else {
+ status = +to; // NaN
+ }
+ for (var i = 0, ii = anim.percents.length; i < ii; i++) {
+ if (anim.percents[i] == percent || anim.percents[i] > status * anim.top) {
+ percent = anim.percents[i];
+ prev = anim.percents[i - 1] || 0;
+ ms = ms / anim.top * (percent - prev);
+ next = anim.percents[i + 1];
+ params = anim.anim[percent];
+ break;
+ } else if (status) {
+ element.attr(anim.anim[anim.percents[i]]);
+ }
+ }
+ if (!params) {
+ return;
+ }
+ if (!isInAnim) {
+ for (var attr in params)
+ if (params[has](attr)) {
+ if (availableAnimAttrs[has](attr) || element.ca[attr]) {
+ from[attr] = element.attr(attr);
+ (from[attr] == null) && (from[attr] = availableAttrs[attr]);
+ to[attr] = params[attr];
+ switch (availableAnimAttrs[attr]) {
+ case nu:
+ diff[attr] = (to[attr] - from[attr]) / ms;
+ break;
+ case "colour":
+ from[attr] = R.getRGB(from[attr]);
+ var toColour = R.getRGB(to[attr]);
+ diff[attr] = {
+ r: (toColour.r - from[attr].r) / ms,
+ g: (toColour.g - from[attr].g) / ms,
+ b: (toColour.b - from[attr].b) / ms
+ };
+ break;
+ case "path":
+ var pathes = path2curve(from[attr], to[attr]),
+ toPath = pathes[1];
+ from[attr] = pathes[0];
+ diff[attr] = [];
+ for (i = 0, ii = from[attr].length; i < ii; i++) {
+ diff[attr][i] = [0];
+ for (var j = 1, jj = from[attr][i].length; j < jj; j++) {
+ diff[attr][i][j] = (toPath[i][j] - from[attr][i][j]) / ms;
+ }
+ }
+ break;
+ case "transform":
+ var _ = element._,
+ eq = equaliseTransform(_[attr], to[attr]);
+ if (eq) {
+ from[attr] = eq.from;
+ to[attr] = eq.to;
+ diff[attr] = [];
+ diff[attr].real = true;
+ for (i = 0, ii = from[attr].length; i < ii; i++) {
+ diff[attr][i] = [from[attr][i][0]];
+ for (j = 1, jj = from[attr][i].length; j < jj; j++) {
+ diff[attr][i][j] = (to[attr][i][j] - from[attr][i][j]) / ms;
+ }
+ }
+ } else {
+ var m = (element.matrix || new Matrix),
+ to2 = {
+ _: {
+ transform: _.transform
+ },
+ getBBox: function() {
+ return element.getBBox(1);
+ }
+ };
+ from[attr] = [
+ m.a,
+ m.b,
+ m.c,
+ m.d,
+ m.e,
+ m.f
+ ];
+ extractTransform(to2, to[attr]);
+ to[attr] = to2._.transform;
+ diff[attr] = [
+ (to2.matrix.a - m.a) / ms,
+ (to2.matrix.b - m.b) / ms,
+ (to2.matrix.c - m.c) / ms,
+ (to2.matrix.d - m.d) / ms,
+ (to2.matrix.e - m.e) / ms,
+ (to2.matrix.f - m.f) / ms
+ ];
+ // from[attr] = [_.sx, _.sy, _.deg, _.dx, _.dy];
+ // var to2 = {_:{}, getBBox: function () { return element.getBBox(); }};
+ // extractTransform(to2, to[attr]);
+ // diff[attr] = [
+ // (to2._.sx - _.sx) / ms,
+ // (to2._.sy - _.sy) / ms,
+ // (to2._.deg - _.deg) / ms,
+ // (to2._.dx - _.dx) / ms,
+ // (to2._.dy - _.dy) / ms
+ // ];
+ }
+ break;
+ case "csv":
+ var values = Str(params[attr])[split](separator),
+ from2 = Str(from[attr])[split](separator);
+ if (attr == "clip-rect") {
+ from[attr] = from2;
+ diff[attr] = [];
+ i = from2.length;
+ while (i--) {
+ diff[attr][i] = (values[i] - from[attr][i]) / ms;
+ }
+ }
+ to[attr] = values;
+ break;
+ default:
+ values = [][concat](params[attr]);
+ from2 = [][concat](from[attr]);
+ diff[attr] = [];
+ i = element.ca[attr].length;
+ while (i--) {
+ diff[attr][i] = ((values[i] || 0) - (from2[i] || 0)) / ms;
+ }
+ break;
+ }
+ }
+ }
+ var easing = params.easing,
+ easyeasy = R.easing_formulas[easing];
+ if (!easyeasy) {
+ easyeasy = Str(easing).match(bezierrg);
+ if (easyeasy && easyeasy.length == 5) {
+ var curve = easyeasy;
+ easyeasy = function(t) {
+ return CubicBezierAtTime(t, +curve[1], +curve[2], +curve[3], +curve[4], ms);
+ };
+ } else {
+ easyeasy = pipe;
+ }
+ }
+ timestamp = params.start || anim.start || +new Date;
+ e = {
+ anim: anim,
+ percent: percent,
+ timestamp: timestamp,
+ start: timestamp + (anim.del || 0),
+ status: 0,
+ initstatus: status || 0,
+ stop: false,
+ ms: ms,
+ easing: easyeasy,
+ from: from,
+ diff: diff,
+ to: to,
+ el: element,
+ callback: params.callback,
+ prev: prev,
+ next: next,
+ repeat: times || anim.times,
+ origin: element.attr(),
+ totalOrigin: totalOrigin
+ };
+ animationElements.push(e);
+ if (status && !isInAnim && !isInAnimSet) {
+ e.stop = true;
+ e.start = new Date - ms * status;
+ if (animationElements.length == 1) {
+ return animation();
+ }
+ }
+ if (isInAnimSet) {
+ e.start = new Date - e.ms * status;
+ }
+ animationElements.length == 1 && requestAnimFrame(animation);
+ } else {
+ isInAnim.initstatus = status;
+ isInAnim.start = new Date - isInAnim.ms * status;
+ }
+ eve("raphael.anim.start." + element.id, element, anim);
+ }
+
+ /*\
+ * Raphael.animation
+ [ method ]
+ **
+ * Creates an animation object that can be passed to the @Element.animate or @Element.animateWith methods.
+ * See also @Animation.delay and @Animation.repeat methods.
+ **
+ > Parameters
+ **
+ - params (object) final attributes for the element, see also @Element.attr
+ - ms (number) number of milliseconds for animation to run
+ - easing (string) #optional easing type. Accept one of @Raphael.easing_formulas or CSS format: `cubic‐bezier(XX, XX, XX, XX)`
+ - callback (function) #optional callback function. Will be called at the end of animation.
+ **
+ = (object) @Animation
+ \*/
+ R.animation = function(params, ms, easing, callback) {
+ if (params instanceof Animation) {
+ return params;
+ }
+ if (R.is(easing, "function") || !easing) {
+ callback = callback || easing || null;
+ easing = null;
+ }
+ params = Object(params);
+ ms = +ms || 0;
+ var p = {},
+ json,
+ attr;
+ for (attr in params)
+ if (params[has](attr) && toFloat(attr) != attr && toFloat(attr) + "%" != attr) {
+ json = true;
+ p[attr] = params[attr];
+ }
+ if (!json) {
+ return new Animation(params, ms);
+ } else {
+ easing && (p.easing = easing);
+ callback && (p.callback = callback);
+ return new Animation({
+ 100: p
+ }, ms);
+ }
+ };
+
+ /*\
+ * Element.animate
+ [ method ]
+ **
+ * Creates and starts animation for given element.
+ **
+ > Parameters
+ **
+ - params (object) final attributes for the element, see also @Element.attr
+ - ms (number) number of milliseconds for animation to run
+ - easing (string) #optional easing type. Accept one of @Raphael.easing_formulas or CSS format: `cubic‐bezier(XX, XX, XX, XX)`
+ - callback (function) #optional callback function. Will be called at the end of animation.
+ * or
+ - animation (object) animation object, see @Raphael.animation
+ **
+ = (object) original element
+ \*/
+ elproto.animate = function(params, ms, easing, callback) {
+ var element = this;
+ if (element.removed) {
+ callback && callback.call(element);
+ return element;
+ }
+ var anim = params instanceof Animation ? params : R.animation(params, ms, easing, callback);
+ runAnimation(anim, element, anim.percents[0], null, element.attr());
+ return element;
+ };
+
+ /*\
+ * Element.setTime
+ [ method ]
+ **
+ * Sets the status of animation of the element in milliseconds. Similar to @Element.status method.
+ **
+ > Parameters
+ **
+ - anim (object) animation object
+ - value (number) number of milliseconds from the beginning of the animation
+ **
+ = (object) original element if `value` is specified
+ * Note, that during animation following events are triggered:
+ *
+ * On each animation frame event `anim.frame.`, on start `anim.start.` and on end `anim.finish.`.
+ \*/
+ elproto.setTime = function(anim, value) {
+ if (anim && value != null) {
+ this.status(anim, mmin(value, anim.ms) / anim.ms);
+ }
+ return this;
+ };
+
+ /*\
+ * Element.status
+ [ method ]
+ **
+ * Gets or sets the status of animation of the element.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ - value (number) #optional 0 – 1. If specified, method works like a setter and sets the status of a given animation to the value. This will cause animation to jump to the given position.
+ **
+ = (number) status
+ * or
+ = (array) status if `anim` is not specified. Array of objects in format:
+ o {
+ o anim: (object) animation object
+ o status: (number) status
+ o }
+ * or
+ = (object) original element if `value` is specified
+ \*/
+ elproto.status = function(anim, value) {
+ var out = [],
+ i = 0,
+ len,
+ e;
+ if (value != null) {
+ runAnimation(anim, this, -1, mmin(value, 1));
+ return this;
+ } else {
+ len = animationElements.length;
+ for (; i < len; i++) {
+ e = animationElements[i];
+ if (e.el.id == this.id && (!anim || e.anim == anim)) {
+ if (anim) {
+ return e.status;
+ }
+ out.push({
+ anim: e.anim,
+ status: e.status
+ });
+ }
+ }
+ if (anim) {
+ return 0;
+ }
+ return out;
+ }
+ };
+
+ /*\
+ * Element.pause
+ [ method ]
+ **
+ * Stops animation of the element with ability to resume it later on.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ **
+ = (object) original element
+ \*/
+ elproto.pause = function(anim) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.id == this.id && (!anim || animationElements[i].anim == anim)) {
+ if (eve("raphael.anim.pause." + this.id, this, animationElements[i].anim) !== false) {
+ animationElements[i].paused = true;
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Element.resume
+ [ method ]
+ **
+ * Resumes animation if it was paused with @Element.pause method.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ **
+ = (object) original element
+ \*/
+ elproto.resume = function(anim) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.id == this.id && (!anim || animationElements[i].anim == anim)) {
+ var e = animationElements[i];
+ if (eve("raphael.anim.resume." + this.id, this, e.anim) !== false) {
+ delete e.paused;
+ this.status(e.anim, e.status);
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Element.stop
+ [ method ]
+ **
+ * Stops animation of the element.
+ **
+ > Parameters
+ **
+ - anim (object) #optional animation object
+ **
+ = (object) original element
+ \*/
+ elproto.stop = function(anim) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.id == this.id && (!anim || animationElements[i].anim == anim)) {
+ if (eve("raphael.anim.stop." + this.id, this, animationElements[i].anim) !== false) {
+ animationElements.splice(i--, 1);
+ }
+ }
+ return this;
+ };
+ function stopAnimation(paper) {
+ for (var i = 0; i < animationElements.length; i++)
+ if (animationElements[i].el.paper == paper) {
+ animationElements.splice(i--, 1);
+ }
+ }
+ eve.on("raphael.remove", stopAnimation);
+ eve.on("raphael.clear", stopAnimation);
+ elproto.toString = function() {
+ return "Rapha\xebl\u2019s object";
+ };
+
+ elproto.toFront = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ parent = o.parent,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._tofront(o, parent)) {
+ parent.canvas.appendChild(thisNode);
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+ return o;
+ };
+
+ elproto.toBack = function() {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ parent = o.parent,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (R._toback(o, parent)) {
+ parent.canvas.insertBefore(thisNode, parent.canvas.firstChild);
+ }
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk && follower.el[follower.stalk](o);
+ }
+ return o;
+ };
+
+ elproto.insertAfter = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ thatNode = R._engine.getLastNode(element),
+ parentNode = element.parent.canvas,
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ if (thatNode.nextSibling) {
+ parentNode.insertBefore(thisNode, thatNode.nextSibling);
+ }
+ else {
+ parentNode.appendChild(thisNode);
+ }
+ R._insertafter(o, element, o.parent, element.parent);
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+ return o;
+ };
+
+ elproto.insertBefore = function(element) {
+ if (this.removed) {
+ return this;
+ }
+
+ var o = this,
+ thisNode = R._engine.getNode(o),
+ thatNode = R._engine.getNode(element),
+ followers = o.followers,
+ follower,
+ i,
+ ii;
+
+ element.parent.canvas.insertBefore(thisNode, thatNode);
+ R._insertbefore(o, element, o.parent, element.parent);
+ o.parent = element.parent;
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+ return this;
+ };
+
+ elproto.appendChild = function (element) {
+ if (this.removed || this.type !== 'group') {
+ return this;
+ }
+
+ var group = this,
+ followers = group.followers,
+ follower,
+ thatNode,
+ i,
+ ii;
+
+ // If appending in same group, simply perform toFront().
+ if (element.parent === group) {
+ element.toFront();
+ return group;
+ }
+
+ thatNode = R._engine.getNode(element);
+
+ // first remove from own group
+ R._tear(element, element.parent);
+
+ group.canvas.appendChild(thatNode);
+ element.parent = group;
+
+ !group.bottom && (group.bottom = element);
+ element.prev = group.top;
+ element.next = null;
+ group.top && (group.top.next = element);
+ group.top = element;
+
+ for (i = 0, ii = followers.length; i < ii; i++) {
+ (follower = followers[i]).stalk &&
+ follower.el[follower.stalk](element);
+ }
+
+ return group;
+ };
+
+ elproto.removeChild = function (element) {
+ if (this.removed || this.type !== 'group' || element.parent !== this) {
+ return this;
+ }
+
+ var o = this,
+ thatNode = R._engine.getNode(element),
+ paper = o.paper;
+
+ R._tear(element, o);
+ paper.canvas.appendChild(thatNode);
+
+ o.parent = paper;
+ !paper.bottom && (paper.bottom = o);
+
+ o.prev = paper.top;
+ paper.top && (paper.top.next = o);
+ paper.top = o;
+ o.next = null;
+
+ return o;
+ };
+
+ // Set
+ var Set = function(items) {
+ this.items = [];
+ this.length = 0;
+ this.type = "set";
+ if (items) {
+ for (var i = 0, ii = items.length; i < ii; i++) {
+ if (items[i] && (items[i].constructor == elproto.constructor || items[i].constructor == Set)) {
+ this[this.items.length] = this.items[this.items.length] = items[i];
+ this.length++;
+ }
+ }
+ }
+ },
+ setproto = Set.prototype;
+
+ /*\
+ * Set.push
+ [ method ]
+ **
+ * Adds each argument to the current set.
+ = (object) original element
+ \*/
+ setproto.push = function() {
+ var item,
+ len;
+ for (var i = 0, ii = arguments.length; i < ii; i++) {
+ item = arguments[i];
+ if (item && (item.constructor == elproto.constructor || item.constructor == Set)) {
+ len = this.items.length;
+ this[len] = this.items[len] = item;
+ this.length++;
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Set.pop
+ [ method ]
+ **
+ * Removes last element and returns it.
+ = (object) element
+ \*/
+ setproto.pop = function() {
+ this.length && delete this[this.length--];
+ return this.items.pop();
+ };
+
+ /*\
+ * Set.forEach
+ [ method ]
+ **
+ * Executes given function for each element in the set.
+ *
+ * If function returns `false` it will stop loop running.
+ **
+ > Parameters
+ **
+ - callback (function) function to run
+ - thisArg (object) context object for the callback
+ = (object) Set object
+ \*/
+ setproto.forEach = function(callback, thisArg) {
+ for (var i = 0, ii = this.items.length; i < ii; i++) {
+ if (callback.call(thisArg, this.items[i], i) === false) {
+ return this;
+ }
+ }
+ return this;
+ };
+ for (var method in elproto)
+ if (elproto[has](method)) {
+ setproto[method] = (function(methodname) {
+ return function() {
+ var arg = arguments;
+ return this.forEach(function(el) {
+ el[methodname][apply](el, arg);
+ });
+ };
+ })(method);
+ }
+ setproto.attr = function(name, value) {
+ if (name && R.is(name, array) && R.is(name[0], object)) {
+ for (var j = 0, jj = name.length; j < jj; j++) {
+ this.items[j].attr(name[j]);
+ }
+ } else {
+ for (var i = 0, ii = this.items.length; i < ii; i++) {
+ this.items[i].attr(name, value);
+ }
+ }
+ return this;
+ };
+
+ /*\
+ * Set.clear
+ [ method ]
+ **
+ * Removeds all elements from the set
+ \*/
+ setproto.clear = function() {
+ while (this.length) {
+ this.pop();
+ }
+ };
+
+ /*\
+ * Set.splice
+ [ method ]
+ **
+ * Removes given element from the set
+ **
+ > Parameters
+ **
+ - index (number) position of the deletion
+ - count (number) number of element to remove
+ - insertion… (object) #optional elements to insert
+ = (object) set elements that were deleted
+ \*/
+ setproto.splice = function(index, count, insertion) {
+ index = index < 0 ? mmax(this.length + index, 0) : index;
+ count = mmax(0, mmin(this.length - index, isNaN(count) && this.length || count));
+ var tail = [],
+ todel = [],
+ args = [],
+ i;
+ for (i = 2; i < arguments.length; i++) {
+ args.push(arguments[i]);
+ }
+ for (i = 0; i < count; i++) {
+ todel.push(this[index + i]);
+ }
+ for (; i < this.length - index; i++) {
+ tail.push(this[index + i]);
+ }
+ var arglen = args.length;
+ for (i = 0; i < arglen + tail.length; i++) {
+ this.items[index + i] = this[index + i] = i < arglen ? args[i] : tail[i - arglen];
+ }
+ i = this.items.length = this.length -= count - arglen;
+ while (this[i]) {
+ delete this[i++];
+ }
+ return new Set(todel);
+ };
+
+ /*\
+ * Set.exclude
+ [ method ]
+ **
+ * Removes given element from the set
+ **
+ > Parameters
+ **
+ - element (object) element to remove
+ = (boolean) `true` if object was found & removed from the set
+ \*/
+ setproto.exclude = function(el) {
+ for (var i = 0, ii = this.length; i < ii; i++)
+ if (this[i] == el) {
+ this.splice(i, 1);
+ return true;
+ }
+ };
+ setproto.animate = function(params, ms, easing, callback) {
+ (R.is(easing, "function") || !easing) && (callback = easing || null);
+ var len = this.items.length,
+ i = len,
+ item,
+ set = this,
+ collector;
+ if (!len) {
+ return this;
+ }
+ callback && (collector = function() {
+ !--len && callback.call(set);
+ });
+ easing = R.is(easing, string) ? easing : collector;
+ var anim = R.animation(params, ms, easing, collector);
+ item = this.items[--i].animate(anim);
+ while (i--) {
+ this.items[i] && !this.items[i].removed && this.items[i].animateWith(item, anim, anim);
+ }
+ return this;
+ };
+ setproto.insertAfter = function(el) {
+ var i = this.items.length;
+ while (i--) {
+ this.items[i].insertAfter(el);
+ }
+ return this;
+ };
+ setproto.getBBox = function() {
+ var x = [],
+ y = [],
+ x2 = [],
+ y2 = [];
+ for (var i = this.items.length; i--; )
+ if (!this.items[i].removed) {
+ var box = this.items[i].getBBox();
+ x.push(box.x);
+ y.push(box.y);
+ x2.push(box.x + box.width);
+ y2.push(box.y + box.height);
+ }
+ x = mmin[apply](0, x);
+ y = mmin[apply](0, y);
+ x2 = mmax[apply](0, x2);
+ y2 = mmax[apply](0, y2);
+ return {
+ x: x,
+ y: y,
+ x2: x2,
+ y2: y2,
+ width: x2 - x,
+ height: y2 - y
+ };
+ };
+ setproto.clone = function(s) {
+ s = new Set;
+ for (var i = 0, ii = this.items.length; i < ii; i++) {
+ s.push(this.items[i].clone());
+ }
+ return s;
+ };
+ setproto.toString = function() {
+ return "Rapha\xebl\u2018s set";
+ };
+
+ setproto.glow = function(glowConfig) {
+ var ret = this.paper.set();
+ this.forEach(function(shape, index){
+ var g = shape.glow(glowConfig);
+ if(g != null){
+ g.forEach(function(shape2, index2){
+ ret.push(shape2);
+ });
+ }
+ });
+ return ret;
+ };
+
+ /*\
+ * Raphael.registerFont
+ [ method ]
+ **
+ * Adds given font to the registered set of fonts for Raphaël. Should be used as an internal call from within Cufón’s font file.
+ * Returns original parameter, so it could be used with chaining.
+ # More about Cufón and how to convert your font form TTF, OTF, etc to JavaScript file.
+ **
+ > Parameters
+ **
+ - font (object) the font to register
+ = (object) the font you passed in
+ > Usage
+ | Cufon.registerFont(Raphael.registerFont({…}));
+ \*/
+ R.registerFont = function(font) {
+ if (!font.face) {
+ return font;
+ }
+ this.fonts = this.fonts || {};
+ var fontcopy = {
+ w: font.w,
+ face: {},
+ glyphs: {}
+ },
+ family = font.face["font-family"];
+ for (var prop in font.face)
+ if (font.face[has](prop)) {
+ fontcopy.face[prop] = font.face[prop];
+ }
+ if (this.fonts[family]) {
+ this.fonts[family].push(fontcopy);
+ } else {
+ this.fonts[family] = [fontcopy];
+ }
+ if (!font.svg) {
+ fontcopy.face["units-per-em"] = toInt(font.face["units-per-em"], 10);
+ for (var glyph in font.glyphs)
+ if (font.glyphs[has](glyph)) {
+ var path = font.glyphs[glyph];
+ fontcopy.glyphs[glyph] = {
+ w: path.w,
+ k: {},
+ d: path.d && "M" + path.d.replace(/[mlcxtrv]/g, function(command) {
+ return {
+ l: "L",
+ c: "C",
+ x: "z",
+ t: "m",
+ r: "l",
+ v: "c"
+ }
+ [command] || "M";
+ }) + "z"
+ };
+ if (path.k) {
+ for (var k in path.k)
+ if (path[has](k)) {
+ fontcopy.glyphs[glyph].k[k] = path.k[k];
+ }
+ }
+ }
+ }
+ return font;
+ };
+
+ /*\
+ * Paper.getFont
+ [ method ]
+ **
+ * Finds font object in the registered fonts by given parameters. You could specify only one word from the font name, like “Myriad” for “Myriad Pro”.
+ **
+ > Parameters
+ **
+ - family (string) font family name or any word from it
+ - weight (string) #optional font weight
+ - style (string) #optional font style
+ - stretch (string) #optional font stretch
+ = (object) the font object
+ > Usage
+ | paper.print(100, 100, "Test string", paper.getFont("Times", 800), 30);
+ \*/
+ paperproto.getFont = function(family, weight, style, stretch) {
+ stretch = stretch || "normal";
+ style = style || "normal";
+ weight = +weight || {
+ normal: 400,
+ bold: 700,
+ lighter: 300,
+ bolder: 800
+ }
+ [weight] || 400;
+ if (!R.fonts) {
+ return;
+ }
+ var font = R.fonts[family];
+ if (!font) {
+ var name = new RegExp("(^|\\s)" + family.replace(/[^\w\d\s+!~.:_-]/g, E) + "(\\s|$)", "i");
+ for (var fontName in R.fonts)
+ if (R.fonts[has](fontName)) {
+ if (name.test(fontName)) {
+ font = R.fonts[fontName];
+ break;
+ }
+ }
+ }
+ var thefont;
+ if (font) {
+ for (var i = 0, ii = font.length; i < ii; i++) {
+ thefont = font[i];
+ if (thefont.face["font-weight"] == weight && (thefont.face["font-style"] == style || !thefont.face["font-style"]) && thefont.face["font-stretch"] == stretch) {
+ break;
+ }
+ }
+ }
+ return thefont;
+ };
+
+ /*\
+ * Paper.print
+ [ method ]
+ **
+ * Creates path that represent given text written using given font at given position with given size.
+ * Result of the method is path element that contains whole text as a separate path.
+ **
+ > Parameters
+ **
+ - x (number) x position of the text
+ - y (number) y position of the text
+ - string (string) text to print
+ - font (object) font object, see @Paper.getFont
+ - size (number) #optional size of the font, default is `16`
+ - origin (string) #optional could be `"baseline"` or `"middle"`, default is `"middle"`
+ - letter_spacing (number) #optional number in range `-1..1`, default is `0`
+ = (object) resulting path element, which consist of all letters
+ > Usage
+ | var txt = r.print(10, 50, "print", r.getFont("Museo"), 30).attr({fill: "#fff"});
+ \*/
+ paperproto.print = function(x, y, string, font, size, origin, letter_spacing) {
+ origin = origin || "middle"; // baseline|middle
+ letter_spacing = mmax(mmin(letter_spacing || 0, 1), -1);
+ var letters = Str(string)[split](E),
+ shift = 0,
+ notfirst = 0,
+ path = E,
+ scale;
+ R.is(font, string) && (font = this.getFont(font));
+ if (font) {
+ scale = (size || 16) / font.face["units-per-em"];
+ var bb = font.face.bbox[split](separator),
+ top = +bb[0],
+ lineHeight = bb[3] - bb[1],
+ shifty = 0,
+ height = + bb[1] + (origin == "baseline" ? lineHeight + ( + font.face.descent) : lineHeight / 2);
+ for (var i = 0, ii = letters.length; i < ii; i++) {
+ if (letters[i] == "\n") {
+ shift = 0;
+ curr = 0;
+ notfirst = 0;
+ shifty += lineHeight;
+ } else {
+ var prev = notfirst && font.glyphs[letters[i - 1]] || {},
+ curr = font.glyphs[letters[i]];
+ shift += notfirst ? (prev.w || font.w) + (prev.k && prev.k[letters[i]] || 0) + (font.w * letter_spacing) : 0;
+ notfirst = 1;
+ }
+ if (curr && curr.d) {
+ path += R.transformPath(curr.d, ["t", shift * scale, shifty * scale, "s", scale, scale, top, height, "t", (x - top) / scale, (y - height) / scale]);
+ }
+ }
+ }
+ return this.path(path).attr({
+ fill: "#000",
+ stroke: "none"
+ });
+ };
+
+ /*\
+ * Paper.add
+ [ method ]
+ **
+ * Imports elements in JSON array in format `{type: type, }`
+ **
+ > Parameters
+ **
+ - json (array)
+ = (object) resulting set of imported elements
+ > Usage
+ | paper.add([
+ | {
+ | type: "circle",
+ | cx: 10,
+ | cy: 10,
+ | r: 5
+ | },
+ | {
+ | type: "rect",
+ | x: 10,
+ | y: 10,
+ | width: 10,
+ | height: 10,
+ | fill: "#fc0"
+ | }
+ | ]);
+ \*/
+ paperproto.add = function(json) {
+ if (R.is(json, "array")) {
+ var res = this.set(),
+ i = 0,
+ ii = json.length,
+ j;
+ for (; i < ii; i++) {
+ j = json[i] || {};
+ elements[has](j.type) && res.push(this[j.type]().attr(j));
+ }
+ }
+ return res;
+ };
+
+ /*\
+ * Raphael.format
+ [ method ]
+ **
+ * Simple format function. Replaces construction of type “`{}`” to the corresponding argument.
+ **
+ > Parameters
+ **
+ - token (string) string to format
+ - … (string) rest of arguments will be treated as parameters for replacement
+ = (string) formated string
+ > Usage
+ | var x = 10,
+ | y = 20,
+ | width = 40,
+ | height = 50;
+ | // this will draw a rectangular shape equivalent to "M10,20h40v50h-40z"
+ | paper.path(Raphael.format("M{0},{1}h{2}v{3}h{4}z", x, y, width, height, -width));
+ \*/
+ R.format = function(token, params) {
+ var args = R.is(params, array) ? [0][concat](params) : arguments;
+ token && R.is(token, string) && args.length - 1 && (token = token.replace(formatrg, function(str, i) {
+ return args[++i] == null ? E : args[i];
+ }));
+ return token || E;
+ };
+
+ /*\
+ * Raphael.fullfill
+ [ method ]
+ **
+ * A little bit more advanced format function than @Raphael.format. Replaces construction of type “`{}`” to the corresponding argument.
+ **
+ > Parameters
+ **
+ - token (string) string to format
+ - json (object) object which properties will be used as a replacement
+ = (string) formated string
+ > Usage
+ | // this will draw a rectangular shape equivalent to "M10,20h40v50h-40z"
+ | paper.path(Raphael.fullfill("M{x},{y}h{dim.width}v{dim.height}h{dim['negative width']}z", {
+ | x: 10,
+ | y: 20,
+ | dim: {
+ | width: 40,
+ | height: 50,
+ | "negative width": -40
+ | }
+ | }));
+ \*/
+ R.fullfill = (function() {
+ var tokenRegex = /\{([^\}]+)\}/g,
+ objNotationRegex = /(?:(?:^|\.)(.+?)(?=\[|\.|$|\()|\[('|")(.+?)\2\])(\(\))?/g, // matches .xxxxx or ["xxxxx"] to run over object properties
+ replacer = function(all, key, obj) {
+ var res = obj;
+ key.replace(objNotationRegex, function(all, name, quote, quotedName, isFunc) {
+ name = name || quotedName;
+ if (res) {
+ if (name in res) {
+ res = res[name];
+ }
+ typeof res == "function" && isFunc && (res = res());
+ }
+ });
+ res = (res == null || res == obj ? all : res) + "";
+ return res;
+ };
+ return function(str, obj) {
+ return String(str).replace(tokenRegex, function(all, key) {
+ return replacer(all, key, obj);
+ });
+ };
+ })();
+
+ /*\
+ * Raphael.ninja
+ [ method ]
+ **
+ * If you want to leave no trace of Raphaël (Well, Raphaël creates only one global variable `Raphael`, but anyway.) You can use `ninja` method.
+ * Beware, that in this case plugins could stop working, because they are depending on global variable existance.
+ **
+ = (object) Raphael object
+ > Usage
+ | (function (local_raphael) {
+ | var paper = local_raphael(10, 10, 320, 200);
+ | …
+ | })(Raphael.ninja());
+ \*/
+ R.ninja = function() {
+ oldRaphael.was ? (g.win.Raphael = oldRaphael.is) : delete Raphael;
+ return R;
+ };
+
+ var crispFixer = (R.vml && 0.5 || 0);
+
+ R.crispBound = cacher(function (x, y, w, h, s) {
+ var at = {},
+ normalizer;
+
+ x = x || 0;
+ y = y || 0;
+ w = w || 0;
+ h = h || 0;
+ s = s || 0;
+ normalizer = s % 2 / 2 + crispFixer;
+
+ // normalize for crisp edges
+ at.x = round(x + normalizer) - normalizer;
+ at.y = round(y + normalizer) - normalizer;
+ at.width = round(x + w + normalizer) - normalizer - at.x;
+ at.height = round(y + h + normalizer) - normalizer - at.y;
+ at['stroke-width'] = s;
+
+ // adjust to single pixel if resultant dimension is zero.
+ (at.width === 0 && w !== 0) && (at.width = 1);
+ (at.height === 0 && h !== 0) && (at.height = 1);
+
+ return at;
+ }, R);
+
+ elproto.crisp = function () {
+ var o = this,
+ attrs = o.attrs,
+ key,
+ attr = {},
+ values = o.attr(['x', 'y', 'width', 'height', 'stroke-width']);
+
+ values = R.crispBound(values.x, values.y, values.width, values.height,
+ values['stroke-width']);
+
+ for (key in values) {
+ if (attrs[key] === values[key]) { // only set attribute if changed
+ delete values[key];
+ }
+ }
+
+ return o.attr(values);
+ };
+
+ /*\
+ * Raphael.st
+ [ property (object) ]
+ **
+ * You can add your own method to elements and sets. It is wise to add a set method for each element method
+ * you added, so you will be able to call the same method on sets too.
+ **
+ * See also @Raphael.el.
+ > Usage
+ | Raphael.el.red = function () {
+ | this.attr({fill: "#f00"});
+ | };
+ | Raphael.st.red = function () {
+ | this.forEach(function (el) {
+ | el.red();
+ | });
+ | };
+ | // then use it
+ | paper.set(paper.circle(100, 100, 20), paper.circle(110, 100, 20)).red();
+ \*/
+ R.st = setproto;
+
+ /*\
+ * Raphael.define
+ [ method ]
+ **
+ * Allows a unified definition of composite shapes and other behaviours using
+ * simple directives.
+ **
+ > Parameters
+ **
+ - definition (object) the shape definition
+ \*/
+ R.define = function (name, init, ca, fn, e) {
+ var i,
+ ii;
+
+ // multi definition
+ if (R.is(name, array)) {
+ for (i = 0, ii = name.length; i < ii; i++) {
+ R.define(name[i]);
+ }
+ return;
+ }
+ // object definition
+ else if (R.is(name, object)) {
+ R.define(name.name, name[name.name], name.ca, name.fn, name.e);
+ return;
+ }
+ // invalid or duplicate definition
+ else if (!name || R.fn[name]) {
+ return;
+ }
+
+ R.fn[name] = function () {
+ var args = arguments,
+ element = init.apply(this, args),
+ key;
+
+ if (fn && R.is(fn, object)) {
+ for (key in fn) {
+ element[key] = fn[key];
+ }
+ }
+
+ if (e && R.is(e, object)) {
+ for (key in e) {
+ element[key] && element[key](e[key]);
+ }
+ }
+
+ if (ca) {
+ if (R.is(ca, 'function')) {
+ element.ca[name] = ca;
+ }
+ else {
+ for (key in ca) {
+ element.ca[key] = ca[key];
+ }
+ }
+
+ // Check if namesake ca exists and apply it
+ if (element.ca[name]) {
+ R._lastArgIfGroup(args, true); // purge group
+ element.attr(name, arraySlice.call(args))
+ }
+ }
+
+ return element;
+ };
+
+ if (ca) { R.fn[name].ca = ca; }
+ if (fn) { R.fn[name].fn = fn; }
+ if (e) { R.fn[name].e = e; }
+
+ return R.fn[name];
+ };
+ // Firefox <3.6 fix: http://webreflection.blogspot.com/2009/11/195-chars-to-help-lazy-loading.html
+ (function(doc, loaded, f) {
+ if (doc.readyState == null && doc.addEventListener) {
+ doc.addEventListener(loaded, f = function() {
+ doc.removeEventListener(loaded, f, false);
+ doc.readyState = "complete";
+ }, false);
+ doc.readyState = "loading";
+ }
+ function isLoaded() {
+ (/in/).test(doc.readyState) ? setTimeout(isLoaded, 9) : R.eve("raphael.DOMload");
+ }
+ isLoaded();
+ })(document, "DOMContentLoaded");
+
+ eve.on("raphael.DOMload", function() {
+ loaded = true;
+ });
+
+
+
+ // EXPOSE
+ // SVG and VML are appended just before the EXPOSE line
+ // Even with AMD, Raphael should be defined globally
+ oldRaphael.was ? (g.win.Raphael = R) : (Raphael = R);
+
+ return R;
+}));
\ No newline at end of file
diff --git a/source/raphael.svg.js b/source/raphael.svg.js
new file mode 100644
index 0000000..3797b27
--- /dev/null
+++ b/source/raphael.svg.js
@@ -0,0 +1,1598 @@
+/**!
+* RedRaphael 1.0.0 - JavaScript Vector Library SVG Module
+* Copyright (c) 2012-2013 FusionCharts Technologies
+*
+* Raphael 2.1.0 - JavaScript Vector Library SVG Module
+* Copyright (c) 2008-2012 Dmitry Baranovskiy
+* Copyright © 2008-2012 Sencha Labs
+*
+* Licensed under the MIT license.
+*/
+window.Raphael && window.Raphael.svg && function(R) {
+ var has = "hasOwnProperty",
+ Str = String,
+ toFloat = parseFloat,
+ toInt = parseInt,
+ math = Math,
+ mmax = math.max,
+ abs = math.abs,
+ pow = math.pow,
+ sqrt = math.sqrt,
+ separator = /[, ]+/,
+ zeroStrokeFix = !!(/AppleWebKit/.test(R._g.win.navigator.userAgent) &&
+ (!/Chrome/.test(R._g.win.navigator.userAgent) ||
+ R._g.win.navigator.appVersion.match(/Chrome\/(\d+)\./)[1] < 29)),
+ eve = R.eve,
+ E = "",
+ S = " ",
+ xlink = "http://www.w3.org/1999/xlink",
+ markers = {
+ block: "M5,0 0,2.5 5,5z",
+ classic: "M5,0 0,2.5 5,5 3.5,3 3.5,2z",
+ diamond: "M2.5,0 5,2.5 2.5,5 0,2.5z",
+ open: "M6,1 1,3.5 6,6",
+ oval: "M2.5,0A2.5,2.5,0,0,1,2.5,5 2.5,2.5,0,0,1,2.5,0z"
+ },
+ markerCounter = {},
+ updateReferenceUrl = function () {
+ return R._url = R._g.win.location.href.replace(/#.*?$/, E);
+ };
+
+ R.toString = function() {
+ return "Your browser supports SVG.\nYou are running Rapha\xebl " + this.version;
+ };
+
+ // Automatic gradient and other reference update on state change
+ R._url = (/msie/i.test(navigator.userAgent) && !window.opera) ?
+ E : updateReferenceUrl();
+ if (R._url && R._g.win.history.pushState) {
+ R._g.win.history.pushState = (function () {
+ var fn = R._g.win.history.pushState;
+ return function () {
+ var ret = fn.apply(R._g.win.history, arguments);
+ return updateReferenceUrl(), ret;
+ };
+ }());
+ R._g.win.addEventListener("popstate", updateReferenceUrl, false);
+ }
+
+ var $ = R._createNode = function(el, attr) {
+ if (attr) {
+ if (typeof el == "string") {
+ el = $(el);
+ }
+ for (var key in attr)
+ if (attr[has](key)) {
+ if (key.substring(0, 6) == "xlink:") {
+ el.setAttributeNS(xlink, key.substring(6), Str(attr[key]));
+ } else {
+ el.setAttribute(key, Str(attr[key]));
+ }
+ }
+ } else {
+ el = R._g.doc.createElementNS("http://www.w3.org/2000/svg", el);
+ }
+ return el;
+ },
+ gradientUnitNames = {
+ userSpaceOnUse: 'userSpaceOnUse',
+ objectBoundingBox: 'objectBoundingBox'
+ },
+ gradientSpreadNames = {
+ pad: 'pad',
+ redlect: 'reflect',
+ repeat: 'repeat'
+ },
+ addGradientFill = function(element, gradient) {
+ var type = "linear",
+ id = element.id + gradient,
+ fx = .5, fy = .5, r, cx, cy, units, spread,
+ o = element.node,
+ SVG = element.paper,
+ s = o.style,
+ el = R._g.doc.getElementById(id);
+ if (!el && SVG.defs) {
+ gradient = Str(gradient).replace(R._radial_gradient, function(all, opts) {
+ type = "radial";
+ opts = opts && opts.split(',') || [];
+ units = opts[5];
+ spread = opts[6];
+
+ var _fx = opts[0],
+ _fy = opts[1],
+ _r = opts[2],
+ _cx = opts[3],
+ _cy = opts[4],
+ shifted = (_fx && _fy),
+ dir,
+ sqx;
+
+ if (_r) {
+ r = /\%/.test(_r) ? _r : toFloat(_r);
+ }
+
+ if (units === gradientUnitNames.userSpaceOnUse) {
+ if (shifted) {
+ fx = _fx;
+ fy = _fy;
+ }
+ if (_cx && _cy) {
+ cx = _cx;
+ cy = _cy;
+ if (!shifted) {
+ fx = cx;
+ fy = cy;
+ }
+ }
+ return E;
+ }
+
+ if (shifted) {
+ fx = toFloat(_fx);
+ fy = toFloat(_fy);
+ dir = ((fy > .5) * 2 - 1);
+ (sqx = pow(fx - .5, 2)) + pow(fy - .5, 2) > .25 &&
+ (sqx < .25) && (fy = sqrt(.25 - sqx) * dir + .5) &&
+ fy !== .5 &&
+ (fy = fy.toFixed(5) - 1e-5 * dir);
+ }
+ if (_cx && _cy) {
+ cx = toFloat(_cx);
+ cy = toFloat(_cy);
+ dir = ((cy > .5) * 2 - 1);
+
+ (sqx = pow(cx - .5, 2)) + pow(cy - .5, 2) > .25 &&
+ (sqx < .25) && (cy = sqrt(.25 - sqx) * dir + .5) &&
+ cy !== .5 &&
+ (cy = cy.toFixed(5) - 1e-5 * dir);
+
+ if (!shifted) {
+ fx = cx;
+ fy = cy;
+ }
+ }
+
+ return E;
+ });
+ gradient = gradient.split(/\s*\-\s*/);
+ if (type == "linear") {
+ var angle = gradient.shift(),
+ specs = angle.match(/\((.*)\)/),
+ vector,
+ max;
+
+ specs = specs && specs[1] && specs[1].split(/\s*\,\s*/);
+ angle = -toFloat(angle);
+ if (isNaN(angle)) {
+ return null;
+ }
+ if (specs && specs.length) {
+ if (specs[0] in gradientUnitNames) {
+ units = specs.shift();
+ (specs[0] in gradientSpreadNames) &&
+ (spread = specs.shift());
+ }
+ else {
+ specs[4] && (units = specs[4]);
+ specs[5] && (spread = specs[5]);
+ }
+
+ /* @todo apply angle rotation and validation */
+ vector = [
+ specs[0] || "0%", specs[1] || "0%",
+ specs[2] || "100%", specs[3] || "0%"
+ ];
+ }
+ else {
+ vector = [0, 0, math.cos(R.rad(angle)), math.sin(R.rad(angle))];
+ max = 1 / (mmax(abs(vector[2]), abs(vector[3])) || 1);
+ vector[2] *= max;
+ vector[3] *= max;
+ if (vector[2] < 0) {
+ vector[0] = -vector[2];
+ vector[2] = 0;
+ }
+ if (vector[3] < 0) {
+ vector[1] = -vector[3];
+ vector[3] = 0;
+ }
+ }
+ }
+ var dots = R._parseDots(gradient);
+ if (!dots) {
+ return null;
+ }
+ id = id.replace(/[\(\)\s,\xb0#]/g, "_");
+
+ if (element.gradient && id !== element.gradient.id) {
+ SVG.defs.removeChild(element.gradient);
+ delete element.gradient;
+ }
+
+ if (!element.gradient) {
+ el = $(type + "Gradient", {
+ id: id
+ });
+ element.gradient = el;
+ (units in gradientUnitNames) &&
+ el.setAttribute('gradientUnits', Str(units));
+ (spread in gradientSpreadNames) &&
+ el.setAttribute('spreadMethod', Str(spread));
+ if (type === "radial") {
+ (r !== undefined) && el.setAttribute('r', Str(r));
+
+ if (cx !== undefined && cy !== undefined) {
+ el.setAttribute('cx', Str(cx));
+ el.setAttribute('cy', Str(cy));
+ }
+ el.setAttribute('fx', Str(fx));
+ el.setAttribute('fy', Str(fy));
+ }
+ else {
+ $(el, {
+ x1: vector[0],
+ y1: vector[1],
+ x2: vector[2],
+ y2: vector[3],
+ gradientTransform: element.matrix.invert()
+ });
+ }
+ SVG.defs.appendChild(el);
+ for (var i = 0, ii = dots.length; i < ii; i++) {
+ el.appendChild($("stop", {
+ offset: dots[i].offset ? dots[i].offset : i ? "100%" : "0%",
+ "stop-color": dots[i].color || "#fff",
+ //add stop opacity information
+ "stop-opacity": dots[i].opacity === undefined ? 1 : dots[i].opacity
+ }));
+ }
+ }
+ }
+ $(o, {
+ fill: "url('" + R._url + "#" + id + "')",
+ opacity: 1,
+ "fill-opacity": 1
+ });
+ s.fill = E;
+ s.opacity = 1;
+ s.fillOpacity = 1;
+ return 1;
+ },
+ updatePosition = function(o) {
+ var bbox = o.getBBox(1);
+ $(o.pattern, {
+ patternTransform: o.matrix.invert() + " translate(" + bbox.x + "," + bbox.y + ")"
+ });
+ },
+ addArrow = function(o, value, isEnd) {
+ if (o.type == "path") {
+ var values = Str(value).toLowerCase().split("-"),
+ p = o.paper,
+ se = isEnd ? "end" : "start",
+ node = o.node,
+ attrs = o.attrs,
+ stroke = attrs["stroke-width"],
+ i = values.length,
+ type = "classic",
+ from,
+ to,
+ dx,
+ refX,
+ attr,
+ w = 3,
+ h = 3,
+ t = 5;
+ while (i--) {
+ switch (values[i]) {
+ case "block":
+ case "classic":
+ case "oval":
+ case "diamond":
+ case "open":
+ case "none":
+ type = values[i];
+ break;
+ case "wide":
+ h = 5;
+ break;
+ case "narrow":
+ h = 2;
+ break;
+ case "long":
+ w = 5;
+ break;
+ case "short":
+ w = 2;
+ break;
+ }
+ }
+ if (type == "open") {
+ w += 2;
+ h += 2;
+ t += 2;
+ dx = 1;
+ refX = isEnd ? 4 : 1;
+ attr = {
+ fill: "none",
+ stroke: attrs.stroke
+ };
+ } else {
+ refX = dx = w / 2;
+ attr = {
+ fill: attrs.stroke,
+ stroke: "none"
+ };
+ }
+ if (o._.arrows) {
+ if (isEnd) {
+ o._.arrows.endPath && markerCounter[o._.arrows.endPath]--;
+ o._.arrows.endMarker && markerCounter[o._.arrows.endMarker]--;
+ } else {
+ o._.arrows.startPath && markerCounter[o._.arrows.startPath]--;
+ o._.arrows.startMarker && markerCounter[o._.arrows.startMarker]--;
+ }
+ } else {
+ o._.arrows = {};
+ }
+ if (type != "none") {
+ var pathId = "raphael-marker-" + type,
+ markerId = "raphael-marker-" + se + type + w + h + "-obj" + o.id;
+ if (!R._g.doc.getElementById(pathId)) {
+ p.defs.appendChild($($("path"), {
+ "stroke-linecap": "round",
+ d: markers[type],
+ id: pathId
+ }));
+ markerCounter[pathId] = 1;
+ } else {
+ markerCounter[pathId]++;
+ }
+ var marker = R._g.doc.getElementById(markerId),
+ use;
+ if (!marker) {
+ marker = $($("marker"), {
+ id: markerId,
+ markerHeight: h,
+ markerWidth: w,
+ orient: "auto",
+ refX: refX,
+ refY: h / 2
+ });
+ use = $($("use"), {
+ "xlink:href": "#" + pathId,
+ transform: (isEnd ? "rotate(180 " + w / 2 + " " + h / 2 + ") " : E) + "scale(" + w / t + "," + h / t + ")",
+ "stroke-width": (1 / ((w / t + h / t) / 2)).toFixed(4)
+ });
+ marker.appendChild(use);
+ p.defs.appendChild(marker);
+ markerCounter[markerId] = 1;
+ } else {
+ markerCounter[markerId]++;
+ use = marker.getElementsByTagName("use")[0];
+ }
+ $(use, attr);
+ var delta = dx * (type != "diamond" && type != "oval");
+ if (isEnd) {
+ from = o._.arrows.startdx * stroke || 0;
+ to = R.getTotalLength(attrs.path) - delta * stroke;
+ } else {
+ from = delta * stroke;
+ to = R.getTotalLength(attrs.path) - (o._.arrows.enddx * stroke || 0);
+ }
+ attr = {};
+ attr["marker-" + se] = "url('" + R._url + "#" + markerId + "')";
+ if (to || from) {
+ attr.d = Raphael.getSubpath(attrs.path, from, to);
+ }
+ $(node, attr);
+ o._.arrows[se + "Path"] = pathId;
+ o._.arrows[se + "Marker"] = markerId;
+ o._.arrows[se + "dx"] = delta;
+ o._.arrows[se + "Type"] = type;
+ o._.arrows[se + "String"] = value;
+ } else {
+ if (isEnd) {
+ from = o._.arrows.startdx * stroke || 0;
+ to = R.getTotalLength(attrs.path) - from;
+ } else {
+ from = 0;
+ to = R.getTotalLength(attrs.path) - (o._.arrows.enddx * stroke || 0);
+ }
+ o._.arrows[se + "Path"] && $(node, {
+ d: Raphael.getSubpath(attrs.path, from, to)
+ });
+ delete o._.arrows[se + "Path"];
+ delete o._.arrows[se + "Marker"];
+ delete o._.arrows[se + "dx"];
+ delete o._.arrows[se + "Type"];
+ delete o._.arrows[se + "String"];
+ }
+ for (attr in markerCounter)
+ if (markerCounter[has](attr) && !markerCounter[attr]) {
+ var item = R._g.doc.getElementById(attr);
+ item && item.parentNode.removeChild(item);
+ }
+ }
+ },
+ dasharray = {
+ "": [0],
+ "none": [0],
+ "-": [3, 1],
+ ".": [1, 1],
+ "-.": [3, 1, 1, 1],
+ "-..": [3, 1, 1, 1, 1, 1],
+ ". ": [1, 3],
+ "- ": [4, 3],
+ "--": [8, 3],
+ "- .": [4, 3, 1, 3],
+ "--.": [8, 3, 1, 3],
+ "--..": [8, 3, 1, 3, 1, 3]
+ },
+ addDashes = function(o, value, params) {
+ var predefValue = dasharray[Str(value).toLowerCase()];
+ value = predefValue || ((value !== undefined) && [].concat(value));
+ if (value) {
+ var width = o.attrs["stroke-width"] || "1",
+ butt = {
+ round: width,
+ square: width,
+ butt: 0
+ }[o.attrs["stroke-linecap"] || params["stroke-linecap"]] || 0,
+ i,
+ l = i = value.length;
+ if (predefValue) {
+ while (i--) {
+ value[i] = value[i] * width + ((i % 2) ? 1 : -1) * butt;
+ }
+ }
+ else {
+ for (i = 0; i < l; i += 2) {
+ value[i] -= butt;
+ value[i + 1] && (value[i + 1] += butt);
+ if (value[i] <= 0) {
+ value[i] = 0.1;
+ }
+ }
+ }
+ if (R.is(value, 'array')) {
+ $(o.node, {
+ "stroke-dasharray": value.join(",")
+ });
+ }
+ }
+ },
+ setFillAndStroke = R._setFillAndStroke = function(o, params) {
+ if (!o.paper.canvas) {
+ return;
+ }
+ var node = o.node,
+ attrs = o.attrs,
+ paper = o.paper,
+ s = node.style,
+ vis = s.visibility;
+
+ s.visibility = "hidden";
+ for (var att in params) {
+ if (params[has](att)) {
+ if (!R._availableAttrs[has](att)) {
+ continue;
+ }
+ var value = params[att];
+ attrs[att] = value;
+ switch (att) {
+ case "blur":
+ o.blur(value);
+ break;
+ case "href":
+ case "title":
+ case "target":
+ var pn = node.parentNode;
+ if (pn.tagName.toLowerCase() != "a") {
+ if (value == E) { break; }
+ var hl = $("a");
+ pn.insertBefore(hl, node);
+ hl.appendChild(node);
+ pn = hl;
+ }
+ if (att == "target") {
+ pn.setAttributeNS(xlink, "show", value == "blank" ? "new" : value);
+ } else {
+ pn.setAttributeNS(xlink, att, value);
+ }
+ node.titleNode = pn;
+ break;
+ case "cursor":
+ s.cursor = value;
+ break;
+ case "transform":
+ o.transform(value);
+ break;
+ case "rotation":
+ if (R.is(value, "array")) {
+ o.rotate.apply(o, value);
+ }
+ else {
+ o.rotate(value);
+ }
+ break;
+ case "arrow-start":
+ addArrow(o, value);
+ break;
+ case "arrow-end":
+ addArrow(o, value, 1);
+ break;
+ case "clip-path":
+ var pathClip = true;
+ case "clip-rect":
+ var rect = !pathClip && Str(value).split(separator);
+ o._.clipispath = !!pathClip;
+ if (pathClip || rect.length == 4) {
+ o.clip && o.clip.parentNode.parentNode.removeChild(o.clip.parentNode);
+ var el = $("clipPath"),
+ rc = $(pathClip ? "path" : "rect");
+ el.id = R.createUUID();
+ $(rc, pathClip ? {
+ d: value ? attrs['clip-path'] = R._pathToAbsolute(value) : R._availableAttrs.path,
+ fill: 'none'
+ } : {
+ x: rect[0],
+ y: rect[1],
+ width: rect[2],
+ height: rect[3],
+ transform: o.matrix.invert()
+ });
+ el.appendChild(rc);
+ paper.defs.appendChild(el);
+ $(node, {
+ "clip-path": "url('" + R._url +"#" + el.id + "')"
+ });
+ o.clip = rc;
+ }
+ if (!value) {
+ var path = node.getAttribute("clip-path");
+ if (path) {
+ var clip = R._g.doc.getElementById(path.replace(/(^url\(#|\)$)/g, E));
+ clip && clip.parentNode.removeChild(clip);
+ $(node, {
+ "clip-path": E
+ });
+ delete o.clip;
+ }
+ }
+ break;
+ case "path":
+ if (o.type == "path") {
+ $(node, {
+ d: value ? attrs.path = R._pathToAbsolute(value) : R._availableAttrs.path
+ });
+ o._.dirty = 1;
+ if (o._.arrows) {
+ "startString" in o._.arrows && addArrow(o, o._.arrows.startString);
+ "endString" in o._.arrows && addArrow(o, o._.arrows.endString, 1);
+ }
+ }
+ break;
+ case "width":
+ node.setAttribute(att, value);
+ o._.dirty = 1;
+ if (attrs.fx) {
+ att = "x";
+ value = attrs.x;
+ } else {
+ break;
+ }
+ case "x":
+ if (attrs.fx) {
+ value = -attrs.x - (attrs.width || 0);
+ }
+ case "rx":
+ if (att == "rx" && o.type == "rect") {
+ break;
+ }
+ case "cx":
+ node.setAttribute(att, value);
+ o.pattern && updatePosition(o);
+ o._.dirty = 1;
+ break;
+ case "height":
+ node.setAttribute(att, value);
+ o._.dirty = 1;
+ if (attrs.fy) {
+ att = "y";
+ value = attrs.y;
+ } else {
+ break;
+ }
+ case "y":
+ if (attrs.fy) {
+ value = -attrs.y - (attrs.height || 0);
+ }
+ case "ry":
+ if (att == "ry" && o.type == "rect") {
+ break;
+ }
+ case "cy":
+ node.setAttribute(att, value);
+ o.pattern && updatePosition(o);
+ o._.dirty = 1;
+ break;
+ case "r":
+ if (o.type == "rect") {
+ $(node, {
+ rx: value,
+ ry: value
+ });
+ } else {
+ node.setAttribute(att, value);
+ }
+ o._.dirty = 1;
+ break;
+ case "src":
+ if (o.type == "image") {
+ node.setAttributeNS(xlink, "href", value);
+ }
+ break;
+ case "stroke-width":
+ if (o._.sx != 1 || o._.sy != 1) {
+ value /= mmax(abs(o._.sx), abs(o._.sy)) || 1;
+ }
+ if (paper._vbSize) {
+ value *= paper._vbSize;
+ }
+ if (zeroStrokeFix && value === 0) {
+ value = 0.000001;
+ }
+ node.setAttribute(att, value);
+ if (attrs["stroke-dasharray"]) {
+ addDashes(o, attrs["stroke-dasharray"], params);
+ }
+ if (o._.arrows) {
+ "startString" in o._.arrows && addArrow(o, o._.arrows.startString);
+ "endString" in o._.arrows && addArrow(o, o._.arrows.endString, 1);
+ }
+ break;
+ case "stroke-dasharray":
+ addDashes(o, value, params);
+ break;
+ case "fill":
+ var isURL = Str(value).match(R._ISURL);
+ if (isURL) {
+ el = $("pattern");
+ var ig = $("image");
+ el.id = R.createUUID();
+ $(el, {
+ x: 0,
+ y: 0,
+ patternUnits: "userSpaceOnUse",
+ height: 1,
+ width: 1
+ });
+ $(ig, {
+ x: 0,
+ y: 0,
+ "xlink:href": isURL[1]
+ });
+ el.appendChild(ig);
+
+ (function(el) {
+ R._preload(isURL[1], function() {
+ var w = this.offsetWidth,
+ h = this.offsetHeight;
+ $(el, {
+ width: w,
+ height: h
+ });
+ $(ig, {
+ width: w,
+ height: h
+ });
+ paper.safari();
+ });
+ })(el);
+ paper.defs.appendChild(el);
+ $(node, {
+ fill: "url('" + R._url + "#" + el.id + "')"
+ });
+ o.pattern = el;
+ o.pattern && updatePosition(o);
+ break;
+ }
+ var clr = R.getRGB(value);
+ if (!clr.error) {
+ delete params.gradient;
+ delete attrs.gradient;
+ !R.is(attrs.opacity, "undefined") &&
+ R.is(params.opacity, "undefined") &&
+ $(node, {
+ opacity: attrs.opacity
+ });
+ !R.is(attrs["fill-opacity"], "undefined") &&
+ R.is(params["fill-opacity"], "undefined") &&
+ $(node, {
+ "fill-opacity": attrs["fill-opacity"]
+ });
+ } else if ((o.type == "circle" || o.type == "ellipse" || Str(value).charAt() != "r") && addGradientFill(o, value)) {
+ if ("opacity" in attrs || "fill-opacity" in attrs) {
+ var gradient = R._g.doc.getElementById(node.getAttribute("fill").replace(/^url\(#|\)$/g, E));
+ if (gradient) {
+ var stops = gradient.getElementsByTagName("stop");
+ $(stops[stops.length - 1], {
+ "stop-opacity": ("opacity" in attrs ? attrs.opacity : 1) * ("fill-opacity" in attrs ? attrs["fill-opacity"] : 1)
+ });
+ }
+ }
+ attrs.gradient = value;
+ attrs.fill = "none";
+ break;
+ }
+ if (clr[has]("opacity")) {
+ $(node, {
+ "fill-opacity": (s.fillOpacity =
+ (clr.opacity > 1 ? clr.opacity / 100 : clr.opacity))
+ });
+ o._.opacitydirty = true;
+ }
+ else if (o._.opacitydirty && R.is(attrs['fill-opacity'], "undefined") &&
+ R.is(params["fill-opacity"], "undefined")) {
+ node.removeAttribute('fill-opacity');
+ s.fillOpacity = E;
+ delete o._.opacitydirty;
+ }
+ case "stroke":
+ clr = R.getRGB(value);
+ node.setAttribute(att, clr.hex);
+ att == "stroke" && clr[has]("opacity") && $(node, {
+ "stroke-opacity": clr.opacity > 1 ? clr.opacity / 100 : clr.opacity
+ });
+ if (att == "stroke" && o._.arrows) {
+ "startString" in o._.arrows && addArrow(o, o._.arrows.startString);
+ "endString" in o._.arrows && addArrow(o, o._.arrows.endString, 1);
+ }
+ break;
+ case "gradient":
+ (o.type == "circle" || o.type == "ellipse" || Str(value).charAt() != "r") && addGradientFill(o, value);
+ break;
+ case 'line-height': // do not apply
+ case 'vertical-align': // do not apply
+ break;
+ case "visibility":
+ value === 'hidden' ? o.hide() : o.show();
+ break;
+ case "opacity":
+ if (attrs.gradient && !attrs[has]("stroke-opacity")) {
+ $(node, {
+ "stroke-opacity": value > 1 ? value / 100 : value
+ });
+ }
+ // fall
+ case "fill-opacity":
+ if (attrs.gradient) {
+ gradient = R._g.doc.getElementById(node.getAttribute("fill").replace(/^url\(#|\)$/g, E));
+ if (gradient) {
+ stops = gradient.getElementsByTagName("stop");
+ $(stops[stops.length - 1], {
+ "stop-opacity": value
+ });
+ }
+ break;
+ }
+ default:
+ att == "font-size" && (value = toInt(value, 10) + "px");
+ var cssrule = att.replace(/(\-.)/g, function(w) {
+ return w.substring(1).toUpperCase();
+ });
+ s[cssrule] = value;
+ o._.dirty = 1;
+ node.setAttribute(att, value);
+ break;
+ }
+ }
+ }
+
+ tuneText(o, params);
+ s.visibility = vis;
+ },
+ leading = 1.2,
+ tuneText = function(el, params) {
+ if (el.type != "text" || !(params[has]("text") || params[has]("font") ||
+ params[has]("font-size") || params[has]("x") || params[has]("y") ||
+ params[has]("line-height") || params[has]("vertical-align"))) {
+ return;
+ }
+ var a = el.attrs,
+ node = el.node,
+ computedStyle = node.firstChild && R._g.doc.defaultView.getComputedStyle(node.firstChild, E),
+ fontSize = computedStyle ? toFloat(R._g.doc.defaultView.getComputedStyle(node.firstChild, E).getPropertyValue("font-size")) : 10,
+ lineHeight = toFloat(params['line-height'] || a['line-height']) || fontSize * leading,
+ valign = a[has]("vertical-align") ? a["vertical-align"] : "middle";
+
+ if (isNaN(lineHeight)) {
+ lineHeight = fontSize * leading;
+ }
+
+ valign = valign === 'top' ? -0.5 : (valign === 'bottom' ? 0.5 : 0);
+
+ if (params[has]("text") && (params.text !== a.text || el._textdirty)) {
+ a.text = params.text;
+ while (node.firstChild) {
+ node.removeChild(node.firstChild);
+ }
+ var texts = Str(params.text).split(/\n| /ig),
+ tspans = [],
+ tspan;
+ for (var i = 0, ii = texts.length; i < ii; i++) {
+ tspan = $("tspan");
+ if (i) {
+ $(tspan, {
+ dy: lineHeight,
+ x: a.x
+ });
+ } else {
+ $(tspan, {
+ dy: lineHeight * texts.length * valign,
+ x: a.x
+ });
+ }
+ if (!texts[i]) { // preserve blank lines
+ tspan.setAttributeNS("http://www.w3.org/XML/1998/namespace",
+ "xml:space","preserve");
+ texts[i] = " ";
+ }
+ tspan.appendChild(R._g.doc.createTextNode(texts[i]));
+ node.appendChild(tspan);
+ tspans[i] = tspan;
+ }
+ el._textdirty = false;
+ } else {
+ tspans = node.getElementsByTagName("tspan");
+ for (i = 0, ii = tspans.length; i < ii; i++)
+ if (i) {
+ $(tspans[i], {
+ dy: lineHeight,
+ x: a.x
+ });
+ } else {
+ $(tspans[0], {
+ dy: lineHeight * tspans.length * valign,
+ x: a.x
+ });
+ }
+ }
+ $(node, {
+ x: a.x,
+ y: a.y
+ });
+ el._.dirty = 1;
+ var bb = el._getBBox(),
+ dif = a.y - (bb.y + bb.height / 2);
+
+ // If the bbox is calculated then we need to make additional adjustments,
+ // to account for the fact that the calculated bbox already has the
+ // text alignment, both horizontal and vertical, included in the calculation.
+ if (bb.isCalculated) {
+ switch (a['vertical-align']) {
+ case "top":
+ dif = bb.height * .75;
+ break;
+ case "bottom":
+ dif = - (bb.height * .25);
+ break;
+ default:
+ dif = a.y - (bb.y + bb.height * .25);
+ break;
+ };
+ }
+
+ dif && R.is(dif, "finite") && tspans[0] && $(tspans[0], {
+ dy: dif
+ });
+ },
+ Element = function(node, svg, group) {
+ var o = this,
+ parent = group || svg;
+
+ o.node = o[0] = node;
+ node.raphael = true;
+ node.raphaelid = o.id = R._oid++;
+
+ o.matrix = R.matrix();
+ o.realPath = null;
+
+ o.attrs = o.attrs || {};
+ o.styles = o.styles || {};
+ o.followers = o.followers || [];
+
+ o.paper = svg;
+ o.ca = o.customAttributes = o.customAttributes ||
+ new svg._CustomAttributes();
+
+ o._ = {
+ transform: [],
+ sx: 1,
+ sy: 1,
+ deg: 0,
+ dx: 0,
+ dy: 0,
+ dirty: 1
+ };
+
+ o.parent = parent;
+ !parent.bottom && (parent.bottom = o);
+
+ o.prev = parent.top;
+ parent.top && (parent.top.next = o);
+ parent.top = o;
+ o.next = null;
+ },
+ elproto = R.el;
+
+ Element.prototype = elproto;
+ elproto.constructor = Element;
+
+ R._engine.getNode = function (el) {
+ var node = el.node || el[0].node;
+ return node.titleNode || node;
+ };
+ R._engine.getLastNode = function (el) {
+ var node = el.node || el[el.length - 1].node;
+ return node.titleNode || node;
+ };
+
+ R._engine.path = function(pathString, SVG, group) {
+ var el = $("path");
+
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (SVG.canvas && SVG.canvas.appendChild(el));
+
+ var p = new Element(el, SVG, group);
+ p.type = "path";
+ setFillAndStroke(p, {
+ fill: "none",
+ stroke: "#000",
+ path: pathString
+ });
+ return p;
+ };
+
+ elproto.rotate = function(deg, cx, cy) {
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ deg = Str(deg).split(separator);
+ if (deg.length - 1) {
+ cx = toFloat(deg[1]);
+ cy = toFloat(deg[2]);
+ }
+ deg = toFloat(deg[0]);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ cx = bbox.x + bbox.width / 2;
+ cy = bbox.y + bbox.height / 2;
+ }
+ o.transform(o._.transform.concat([["r", deg, cx, cy]]));
+ return o;
+ };
+
+ elproto.scale = function(sx, sy, cx, cy) {
+ var o = this,
+ bbox;
+ if (o.removed) {
+ return o;
+ }
+ sx = Str(sx).split(separator);
+ if (sx.length - 1) {
+ sy = toFloat(sx[1]);
+ cx = toFloat(sx[2]);
+ cy = toFloat(sx[3]);
+ }
+ sx = toFloat(sx[0]);
+ (sy == null) && (sy = sx);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ bbox = o.getBBox(1);
+ }
+ cx = cx == null ? bbox.x + bbox.width / 2 : cx;
+ cy = cy == null ? bbox.y + bbox.height / 2 : cy;
+ o.transform(o._.transform.concat([["s", sx, sy, cx, cy]]));
+ return o;
+ };
+
+ elproto.translate = function(dx, dy) {
+ var o = this;
+ if (o.removed) {
+ return o;
+ }
+ dx = Str(dx).split(separator);
+ if (dx.length - 1) {
+ dy = toFloat(dx[1]);
+ }
+ dx = toFloat(dx[0]) || 0;
+ dy = +dy || 0;
+ o.transform(o._.transform.concat([["t", dx, dy]]));
+ return o;
+ };
+
+ elproto.transform = function(tstr) {
+ var o = this,
+ _ = o._,
+ sw;
+
+ if (tstr == null) {
+ return _.transform;
+ }
+ R._extractTransform(o, tstr);
+
+ o.clip && !_.clipispath && $(o.clip, {
+ transform: o.matrix.invert()
+ });
+ o.pattern && updatePosition(o);
+ o.node && $(o.node, {
+ transform: o.matrix
+ });
+
+ if (_.sx != 1 || _.sy != 1) {
+ sw = o.attrs[has]("stroke-width") ? o.attrs["stroke-width"] : 1;
+ o.attr({
+ "stroke-width": sw
+ });
+ }
+
+ return o;
+ };
+
+ elproto.hide = function() {
+ var o = this;
+ !o.removed && o.paper.safari(o.node.style.display = "none");
+ return o;
+ };
+
+ elproto.show = function() {
+ var o = this;
+ !o.removed && o.paper.safari(o.node.style.display = E);
+ return o;
+ };
+
+ elproto.remove = function() {
+ if (this.removed || !this.parent.canvas) {
+ return;
+ }
+
+ var o = this,
+ node = R._engine.getNode(o),
+ paper = o.paper,
+ defs = paper.defs,
+ i;
+
+
+ paper.__set__ && paper.__set__.exclude(o);
+ eve.unbind("raphael.*.*." + o.id);
+
+ if (o.gradient && defs) {
+ defs.removeChild(o.gradient);
+ }
+ while (i = o.followers.pop()) {
+ i.el.remove();
+ }
+ o.parent.canvas.removeChild(node);
+ R._tear(o, paper);
+ for (i in o) {
+ o[i] = typeof o[i] === "function" ? R._removedFactory(i) : null;
+ }
+ o.removed = true;
+ };
+ elproto._getBBox = function() {
+ var o = this,
+ node = o.node,
+ bbox = {},
+ a = o.attrs,
+ align,
+ hide;
+
+ if (node.style.display === "none") {
+ o.show();
+ hide = true;
+ }
+
+ try {
+ bbox = node.getBBox();
+
+ if (o.type == "text") {
+ // If bbox does not have x / y, which is possible in certain
+ // environments, we mathematically calculate these values by
+ // using x, y (adjusted using the values of text-anchor, and
+ // vertical-align attributes), of the element along with the
+ // width and height provided by the getBBox().
+ if (bbox.x === undefined) {
+ bbox.isCalculated = true;
+ align = a['text-anchor'];
+ bbox.x = (a.x || 0) - (bbox.width * ((align === "start") ?
+ 0 : (align === "middle") ? 0.5 : 1));
+ }
+
+ if (bbox.y === undefined) {
+ bbox.isCalculated = true;
+ align = a['vertical-align'];
+ bbox.y = (a.y || 0) - (bbox.height * ((align === "bottom") ?
+ 1 : (align === "middle") ? 0.5 : 0));
+ }
+ }
+
+ } catch (e) {
+ // Firefox 3.0.x plays badly here
+ } finally {
+ bbox = bbox || {};
+ }
+ hide && o.hide();
+ return bbox;
+ };
+
+ elproto.css = function (name, value) {
+ // do not parse css in case element is removed.
+ if (this.removed) {
+ return this;
+ }
+
+ // process as getter when a single key is sent as parameter.
+ if (value == null && R.is(name, "string")) {
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.styles) {
+ out[name] = this.styles[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ // process as getter when multiple keys are pre-sent as array.
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.styles(name[i]);
+ }
+ return out;
+ }
+ // convert single key-value setter into object style standard.
+ if (value != null) {
+ var params = {};
+ params[name] = value;
+ } else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+ // iterate on keys and set style or raise events.
+ var otherkey, doattrs = {};
+ for (var key in params) {
+ otherkey = key.replace(/\B([A-Z]{1})/g, "-$1").toLowerCase();
+
+ // If keys are supported via attr then use attr instead of css.
+ if (R._availableAttrs[has](otherkey) || otherkey === 'color') {
+ // Replace "color" with fill
+ (otherkey === 'color' && this.type === 'text') && (otherkey = 'fill');
+
+ doattrs[otherkey] = params[key];
+ doattrs.dirty = true;
+ continue;
+ }
+ eve("raphael.css." + otherkey + "." + this.id, this, params[key], otherkey);
+ this.node.style[otherkey] = params[key];
+ this.styles[otherkey] = params[key];
+ }
+ // run on followers
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ this.followers[i].el.css(params);
+ }
+ // apply css via attrs
+ if (doattrs[has]("dirty")) {
+ delete doattrs.dirty;
+ this.attr(doattrs);
+ }
+ return this;
+ };
+
+ elproto.attr = function(name, value) {
+ if (this.removed) {
+ return this;
+ }
+ if (name == null) {
+ var res = {};
+ for (var a in this.attrs)
+ if (this.attrs[has](a)) {
+ res[a] = this.attrs[a];
+ }
+ res.gradient && res.fill == "none" && (res.fill = res.gradient) && delete res.gradient;
+ res.transform = this._.transform;
+ res.visibility = this.node.style.display === "none" ? "hidden" : "visible";
+ return res;
+ }
+ if (value == null && R.is(name, "string")) {
+ if (name == "fill" && this.attrs.fill == "none" && this.attrs.gradient) {
+ return this.attrs.gradient;
+ }
+ if (name == "transform") {
+ return this._.transform;
+ }
+ if (name == "visibility") {
+ return this.node.style.display === "none" ? "hidden" : "visible";
+ }
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.attrs) {
+ out[name] = this.attrs[name];
+ } else if (R.is(this.ca[name], "function")) {
+ out[name] = this.ca[name].def;
+ } else {
+ out[name] = R._availableAttrs[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.attr(name[i]);
+ }
+ return out;
+ }
+ if (value != null) {
+ var params = {};
+ params[name] = value;
+ } else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+ for (var key in params) {
+ eve("raphael.attr." + key + "." + this.id, this, params[key], key);
+ }
+ var todel = {};
+ for (key in this.ca) {
+ if (this.ca[key] && params[has](key) && R.is(this.ca[key], "function") && !this.ca['_invoked' + key]) {
+
+ this.ca['_invoked'+key] = true; // prevent recursion
+ var par = this.ca[key].apply(this, [].concat(params[key]));
+ delete this.ca['_invoked'+key];
+
+ for (var subkey in par) {
+ if (par[has](subkey)) {
+ params[subkey] = par[subkey];
+ }
+ }
+ this.attrs[key] = params[key];
+ if (par === false) {
+ todel[key] = params[key];
+ delete params[key];
+ }
+ }
+ }
+
+ setFillAndStroke(this, params);
+
+ var follower;
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ follower = this.followers[i];
+ (follower.cb && !follower.cb.call(follower.el, params, this)) ||
+ follower.el.attr(params);
+ }
+
+ for (subkey in todel) {
+ params[subkey] = todel[subkey];
+ }
+ return this;
+ };
+
+ elproto.blur = function(size) {
+ // Experimental. No Safari support. Use it on your own risk.
+ var t = this;
+ if (+size !== 0) {
+ var fltr = $("filter"),
+ blur = $("feGaussianBlur");
+ t.attrs.blur = size;
+ fltr.id = R.createUUID();
+ $(blur, {
+ stdDeviation: +size || 1.5
+ });
+ fltr.appendChild(blur);
+ t.paper.defs.appendChild(fltr);
+ t._blur = fltr;
+ $(t.node, {
+ filter: "url('" + R._url + "#" + fltr.id + "')"
+ });
+ } else {
+ if (t._blur) {
+ t._blur.parentNode.removeChild(t._blur);
+ delete t._blur;
+ delete t.attrs.blur;
+ }
+ t.node.removeAttribute("filter");
+ }
+ };
+
+ elproto.on = function(eventType, handler) {
+ if (this.removed) {
+ return this;
+ }
+
+ var fn = handler;
+ if (R.supportsTouch) {
+ eventType = R._touchMap[eventType] ||
+ (eventType === 'click' && 'touchstart') || eventType;
+ fn = function(e) {
+ e.preventDefault();
+ handler();
+ };
+ }
+ this.node['on'+ eventType] = fn;
+ return this;
+ };
+
+
+ R._engine.group = function(svg, id, group) {
+ var el = $("g");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var g = new Element(el, svg, group);
+ g.type = "group";
+ g.canvas = g.node;
+ g.top = null;
+ g.bottom = null;
+ id && el.setAttribute('class', ['red', id, g.id].join('-'));
+
+ return g;
+ };
+
+ R._engine.circle = function(svg, x, y, r, group) {
+ var el = $("circle");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ cx: x,
+ cy: y,
+ r: r,
+ fill: "none",
+ stroke: "#000"
+ };
+ res.type = "circle";
+ $(el, res.attrs);
+ return res;
+ };
+ R._engine.rect = function(svg, x, y, w, h, r, group) {
+ var el = $("rect");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ x: x,
+ y: y,
+ width: w,
+ height: h,
+ r: r || 0,
+ rx: r || 0,
+ ry: r || 0,
+ fill: "none",
+ stroke: "#000"
+ };
+ res.type = "rect";
+ $(el, res.attrs);
+ return res;
+ };
+ R._engine.ellipse = function(svg, x, y, rx, ry, group) {
+ var el = $("ellipse");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ cx: x,
+ cy: y,
+ rx: rx,
+ ry: ry,
+ fill: "none",
+ stroke: "#000"
+ };
+ res.type = "ellipse";
+ $(el, res.attrs);
+ return res;
+ };
+ R._engine.image = function(svg, src, x, y, w, h, group) {
+ var el = $("image");
+ $(el, {
+ x: x,
+ y: y,
+ width: w,
+ height: h,
+ preserveAspectRatio: "none"
+ });
+ el.setAttributeNS(xlink, "href", src);
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ x: x,
+ y: y,
+ width: w,
+ height: h,
+ src: src
+ };
+ res.type = "image";
+ return res;
+ };
+ R._engine.text = function(svg, x, y, text, group) {
+ var el = $("text");
+ (group && group.canvas && group.canvas.appendChild(el)) ||
+ (svg.canvas && svg.canvas.appendChild(el));
+
+ var res = new Element(el, svg, group);
+ res.attrs = {
+ x: x,
+ y: y,
+ "text-anchor": "middle",
+ "vertical-align": "middle",
+ text: text,
+ //font: R._availableAttrs.font,
+ stroke: "none",
+ fill: "#000"
+ };
+ res.type = "text";
+ res._textdirty = true;
+ setFillAndStroke(res, res.attrs);
+ return res;
+ };
+ /* @diffend */
+
+ R._engine.setSize = function(width, height) {
+ this.width = width || this.width;
+ this.height = height || this.height;
+ this.canvas.setAttribute("width", this.width);
+ this.canvas.setAttribute("height", this.height);
+ if (this._viewBox) {
+ this.setViewBox.apply(this, this._viewBox);
+ }
+ return this;
+ };
+ R._engine.create = function() {
+ var con = R._getContainer.apply(0, arguments),
+ container = con && con.container,
+ x = con.x,
+ y = con.y,
+ width = con.width,
+ height = con.height;
+ if (!container) {
+ throw new Error("SVG container not found.");
+ }
+ var cnvs = $("svg"),
+ css = "overflow:hidden;-webkit-tap-highlight-color:rgba(0,0,0,0);"+
+ "-webkit-user-select:none;-moz-user-select:-moz-none;-khtml-user-select:none;"+
+ "-ms-user-select:none;user-select:none;-o-user-select:none;cursor:default;",
+ isFloating;
+ x = x || 0;
+ y = y || 0;
+ width = width || 512;
+ height = height || 342;
+ $(cnvs, {
+ height: height,
+ version: 1.1,
+ width: width,
+ xmlns: "http://www.w3.org/2000/svg"
+ });
+ if (container == 1) {
+ cnvs.style.cssText = css + "position:absolute;left:" + x + "px;top:" + y + "px";
+ R._g.doc.body.appendChild(cnvs);
+ isFloating = 1;
+ } else {
+ cnvs.style.cssText = css + "position:relative";
+ if (container.firstChild) {
+ container.insertBefore(cnvs, container.firstChild);
+ } else {
+ container.appendChild(cnvs);
+ }
+ }
+ container = new R._Paper;
+ container.width = width;
+ container.height = height;
+ container.canvas = cnvs;
+ container.clear();
+ container._left = container._top = 0;
+ isFloating && (container.renderfix = function() {
+ });
+ container.renderfix();
+ return container;
+ };
+ R._engine.setViewBox = function(x, y, w, h, fit) {
+ eve("raphael.setViewBox", this, this._viewBox, [x, y, w, h, fit]);
+ var size = mmax(w / this.width, h / this.height),
+ top = this.top,
+ aspectRatio = fit ? "meet" : "xMinYMin",
+ vb,
+ sw;
+ if (x == null) {
+ if (this._vbSize) {
+ size = 1;
+ }
+ delete this._vbSize;
+ vb = "0 0 " + this.width + S + this.height;
+ } else {
+ this._vbSize = size;
+ vb = x + S + y + S + w + S + h;
+ }
+ $(this.canvas, {
+ viewBox: vb,
+ preserveAspectRatio: aspectRatio
+ });
+ while (size && top) {
+ sw = "stroke-width" in top.attrs ? top.attrs["stroke-width"] : 1;
+ top.attr({
+ "stroke-width": sw
+ });
+ top._.dirty = 1;
+ top._.dirtyT = 1;
+ top = top.prev;
+ }
+ this._viewBox = [x, y, w, h, !!fit];
+ return this;
+ };
+
+ R.prototype.renderfix = function() {
+ var cnvs = this.canvas,
+ s = cnvs.style,
+ pos;
+ try {
+ pos = cnvs.getScreenCTM() || cnvs.createSVGMatrix();
+ } catch (e) {
+ pos = cnvs.createSVGMatrix();
+ }
+ var left = -pos.e % 1,
+ top = - pos.f % 1;
+ if (left || top) {
+ if (left) {
+ this._left = (this._left + left) % 1;
+ s.left = this._left + "px";
+ }
+ if (top) {
+ this._top = (this._top + top) % 1;
+ s.top = this._top + "px";
+ }
+ }
+ };
+
+ R.prototype.clear = function() {
+ eve("raphael.clear", this);
+ var c = this.canvas;
+ while (c.firstChild) {
+ c.removeChild(c.firstChild);
+ }
+ this.bottom = this.top = null;
+ (this.desc = $("desc")).appendChild(R._g.doc.createTextNode(R.is(R.desc, "string") && R.desc ||
+ "Created with Red Rapha\xebl " + R.version));
+ c.appendChild(this.desc);
+ c.appendChild(this.defs = $("defs"));
+ };
+
+ R.prototype.remove = function() {
+ eve("raphael.remove", this);
+ this.canvas.parentNode && this.canvas.parentNode.removeChild(this.canvas);
+ for (var i in this) {
+ this[i] = typeof this[i] == "function" ? R._removedFactory(i) : null;
+ }
+ this.removed = true;
+ };
+ var setproto = R.st;
+ for (var method in elproto)
+ if (elproto[has](method) && !setproto[has](method)) {
+ setproto[method] = (function(methodname) {
+ return function() {
+ var arg = arguments;
+ return this.forEach(function(el) {
+ el[methodname].apply(el, arg);
+ });
+ };
+ })(method);
+ }
+}(window.Raphael);
diff --git a/source/raphael.vml.js b/source/raphael.vml.js
new file mode 100644
index 0000000..ca081fc
--- /dev/null
+++ b/source/raphael.vml.js
@@ -0,0 +1,1257 @@
+/**!
+* RedRaphael 1.0.0 - JavaScript Vector Library VML Module
+* Copyright (c) 2012-2013 FusionCharts Technologies
+*
+* Raphael 2.1.0 - JavaScript Vector Library VML Module
+* Copyright (c) 2008-2012 Dmitry Baranovskiy
+* Copyright © 2008-2012 Sencha Labs
+*
+* Licensed under the MIT license.
+*/
+
+window.Raphael && window.Raphael.vml && function(R) {
+ var has = "hasOwnProperty",
+ Str = String,
+ toFloat = parseFloat,
+ math = Math,
+ round = math.round,
+ mmax = math.max,
+ mmin = math.min,
+ sqrt = math.sqrt,
+ abs = math.abs,
+ fillString = "fill",
+ separator = /[, ]+/,
+ eve = R.eve,
+ ms = " progid:DXImageTransform.Microsoft",
+ S = " ",
+ E = "",
+ map = {
+ M: "m",
+ L: "l",
+ C: "c",
+ Z: "x",
+ m: "t",
+ l: "r",
+ c: "v",
+ z: "x"
+ },
+ bites = /([clmz]),?([^clmz]*)/gi,
+ blurregexp = / progid:\S+Blur\([^\)]+\)/g,
+ val = /-?[^,\s-]+/g,
+ cssDot = "position:absolute;left:0;top:0;width:1px;height:1px",
+ zoom = 21600,
+ pathTypes = {
+ path: 1,
+ rect: 1,
+ image: 1
+ },
+ ovalTypes = {
+ circle: 1,
+ ellipse: 1
+ },
+ path2vml = function(path) {
+ var total = /[ahqstv]/ig,
+ command = R._pathToAbsolute;
+ Str(path).match(total) && (command = R._path2curve);
+ total = /[clmz]/g;
+ if (command == R._pathToAbsolute && !Str(path).match(total)) {
+ var res = Str(path).replace(bites, function(all, command, args) {
+ var vals = [],
+ isMove = command.toLowerCase() == "m",
+ res = map[command];
+ args.replace(val, function(value) {
+ if (isMove && vals.length == 2) {
+ res += vals + map[command == "m" ? "l" : "L"];
+ vals = [];
+ }
+ vals.push(round(value * zoom));
+ });
+ return res + vals;
+ });
+
+ return res || 'm0,0';
+ }
+ var pa = command(path), p, r;
+ res = [];
+ for (var i = 0, ii = pa.length; i < ii; i++) {
+ p = pa[i];
+ r = pa[i][0].toLowerCase();
+ r == "z" && (r = "x");
+ for (var j = 1, jj = p.length; j < jj; j++) {
+ r += round(p[j] * zoom) + (j != jj - 1 ? "," : E);
+ }
+ res.push(r);
+ }
+ return res.length ? res.join(S) : 'm0,0';
+ },
+ compensation = function(deg, dx, dy) {
+ var m = R.matrix();
+ m.rotate(-deg, .5, .5);
+ return {
+ dx: m.x(dx, dy),
+ dy: m.y(dx, dy)
+ };
+ },
+ setCoords = function(p, sx, sy, dx, dy, deg) {
+ var _ = p._,
+ m = p.matrix,
+ fillpos = _.fillpos,
+ o = p.node,
+ s = o.style,
+ y = 1,
+ flip = "",
+ dxdy,
+ kx = zoom / sx,
+ ky = zoom / sy;
+ s.visibility = "hidden";
+ if (!sx || !sy) {
+ return;
+ }
+ o.coordsize = abs(kx) + S + abs(ky);
+ s.rotation = deg * (sx * sy < 0 ? -1 : 1);
+ if (deg) {
+ var c = compensation(deg, dx, dy);
+ dx = c.dx;
+ dy = c.dy;
+ }
+ sx < 0 && (flip += "x");
+ sy < 0 && (flip += " y") && (y = -1);
+ s.flip = flip;
+ o.coordorigin = (dx * -kx) + S + (dy * -ky);
+ if (fillpos || _.fillsize) {
+ var fill = o.getElementsByTagName(fillString);
+ fill = fill && fill[0];
+ o.removeChild(fill);
+ if (fillpos) {
+ c = compensation(deg, m.x(fillpos[0], fillpos[1]), m.y(fillpos[0], fillpos[1]));
+ fill.position = c.dx * y + S + c.dy * y;
+ }
+ if (_.fillsize) {
+ fill.size = _.fillsize[0] * abs(sx) + S + _.fillsize[1] * abs(sy);
+ }
+ o.appendChild(fill);
+ }
+ s.visibility = "visible";
+ };
+ R._url = E;
+ R.toString = function() {
+ return "Your browser doesn\u2019t support SVG. Falling down to VML.\nYou are running Rapha\xebl " + this.version;
+ };
+ var addArrow = function(o, value, isEnd) {
+ var values = Str(value).toLowerCase().split("-"),
+ se = isEnd ? "end" : "start",
+ i = values.length,
+ type = "classic",
+ w = "medium",
+ h = "medium";
+ while (i--) {
+ switch (values[i]) {
+ case "block":
+ case "classic":
+ case "oval":
+ case "diamond":
+ case "open":
+ case "none":
+ type = values[i];
+ break;
+ case "wide":
+ case "narrow":
+ h = values[i];
+ break;
+ case "long":
+ case "short":
+ w = values[i];
+ break;
+ }
+ }
+ var stroke = o.node.getElementsByTagName("stroke")[0];
+ stroke[se + "arrow"] = type;
+ stroke[se + "arrowlength"] = w;
+ stroke[se + "arrowwidth"] = h;
+ },
+ setFillAndStroke = R._setFillAndStroke = function(o, params) {
+ if (!o.paper.canvas) return;
+ // o.paper.canvas.style.display = "none";
+ o.attrs = o.attrs || {};
+ var node = o.node,
+ a = o.attrs,
+ s = node.style,
+ xy,
+ newpath = pathTypes[o.type] && (params.x != a.x || params.y != a.y || params.width != a.width || params.height != a.height || params.cx != a.cx || params.cy != a.cy || params.rx != a.rx || params.ry != a.ry || params.r != a.r),
+ isOval = ovalTypes[o.type] && (a.cx != params.cx || a.cy != params.cy || a.r != params.r || a.rx != params.rx || a.ry != params.ry),
+ isGroup = o.type === 'group',
+ res = o;
+
+
+ for (var par in params)
+ if (params[has](par)) {
+ a[par] = params[par];
+ }
+ if (newpath) {
+ a.path = R._getPath[o.type](o);
+ o._.dirty = 1;
+ }
+ params.href && (node.href = params.href);
+ params.title && (node.title = params.title);
+ params.target && (node.target = params.target);
+ params.cursor && (s.cursor = params.cursor);
+ "blur" in params && o.blur(params.blur);
+ if (params.path && o.type == "path" || newpath) {
+ node.path = path2vml(~Str(a.path).toLowerCase().indexOf("r") ? R._pathToAbsolute(a.path) : a.path);
+ if (o.type == "image") {
+ o._.fillpos = [a.x, a.y];
+ o._.fillsize = [a.width, a.height];
+ setCoords(o, 1, 1, 0, 0, 0);
+ }
+ }
+ "transform" in params && o.transform(params.transform);
+ if ("rotation" in params) {
+ var rotation = params.rotation;
+ if (R.is(rotation, "array")) {
+ o.rotate.apply(o, rotation);
+ }
+ else {
+ o.rotate(rotation);
+ }
+ }
+ if ("visibility" in params) {
+ params.visibility === 'hidden' ? o.hide() : o.show();
+ }
+ if (isOval) {
+ var cx = +a.cx,
+ cy = +a.cy,
+ rx = +a.rx || +a.r || 0,
+ ry = + a.ry || + a.r || 0;
+ node.path = R.format("ar{0},{1},{2},{3},{4},{1},{4},{1}x", round((cx - rx) * zoom), round((cy - ry) * zoom), round((cx + rx) * zoom), round((cy + ry) * zoom), round(cx * zoom));
+ }
+ if ("clip-rect" in params) {
+ var rect = Str(params["clip-rect"]).split(separator);
+
+ if (rect.length == 4) {
+ rect[0] = +rect[0];
+ rect[1] = +rect[1];
+ rect[2] = +rect[2] + rect[0];
+ rect[3] = +rect[3] + rect[1];
+
+ /* @todo create separate element for group clip-rect to
+ * avoid unclipping issue */
+ var div = isGroup ? node : (node.clipRect ||
+ R._g.doc.createElement("div")),
+ offset,
+ dstyle = div.style;
+
+ if (isGroup) {
+ o.clip = rect.slice(); // copy param
+ offset = o.matrix.offset();
+ offset = [toFloat(offset[0]), toFloat(offset[1])];
+ // invert matrix calculation
+ rect[0] -= offset[0];
+ rect[1] -= offset[1];
+ rect[2] -= offset[0];
+ rect[3] -= offset[1];
+ // Fix for bug in ie clip-auto when height/width is not defined
+ /* @todo set dynamic w/h based on clip bounds or find
+ * another workaround fix */
+ dstyle.width = "10800px";
+ dstyle.height = "10800px";
+ }
+ else if (!node.clipRect) {
+ dstyle.top = "0";
+ dstyle.left = "0";
+ dstyle.width = o.paper.width + "px";
+ dstyle.height = o.paper.height + "px";
+ node.parentNode.insertBefore(div, node);
+ div.appendChild(node);
+ node.clipRect = div;
+ }
+ dstyle.position = "absolute";
+ dstyle.clip = R.format("rect({1}px {2}px {3}px {0}px)", rect);
+ }
+ if (!params["clip-rect"]) {
+ if (isGroup && o.clip) {
+ node.style.clip = "rect(auto auto auto auto)";
+ delete o.clip;
+ }
+ else if (node.clipRect) {
+ node.clipRect.style.clip = "rect(auto auto auto auto)";
+ }
+ }
+ }
+ if (o.textpath) {
+ var textpathStyle = o.textpath.style;
+ params.font && (textpathStyle.font = params.font);
+ params["font-family"] && (textpathStyle.fontFamily = '"' + params["font-family"].split(",")[0].replace(/^['"]+|['"]+$/g, E) + '"');
+ params["font-size"] && (textpathStyle.fontSize = params["font-size"]);
+ params["font-weight"] && (textpathStyle.fontWeight = params["font-weight"]);
+ params["font-style"] && (textpathStyle.fontStyle = params["font-style"]);
+ }
+ if ("arrow-start" in params) {
+ addArrow(res, params["arrow-start"]);
+ }
+ if ("arrow-end" in params) {
+ addArrow(res, params["arrow-end"], 1);
+ }
+ if (params.opacity != null ||
+ params["stroke-width"] != null ||
+ params.fill != null ||
+ params.src != null ||
+ params.stroke != null ||
+ params["stroke-width"] != null ||
+ params["stroke-opacity"] != null ||
+ params["fill-opacity"] != null ||
+ params["stroke-dasharray"] != null ||
+ params["stroke-miterlimit"] != null ||
+ params["stroke-linejoin"] != null ||
+ params["stroke-linecap"] != null) {
+ var fill = node.getElementsByTagName(fillString),
+ newfill = false,
+ fillOpacity = -1;
+ fill = fill && fill[0];
+ !fill && (newfill = fill = createNode(fillString));
+ if (o.type == "image" && params.src) {
+ fill.src = params.src;
+ }
+ params.fill && (fill.on = true);
+ if (fill.on == null || params.fill == "none" || params.fill === null) {
+ fill.on = false;
+ }
+ if (fill.on && params.fill) {
+ var isURL = Str(params.fill).match(R._ISURL);
+ if (isURL) {
+ fill.parentNode == node && node.removeChild(fill);
+ fill.rotate = true;
+ fill.src = isURL[1];
+ fill.type = "tile";
+ var bbox = o.getBBox(1);
+ fill.position = bbox.x + S + bbox.y;
+ o._.fillpos = [bbox.x, bbox.y];
+
+ R._preload(isURL[1], function() {
+ o._.fillsize = [this.offsetWidth, this.offsetHeight];
+ });
+ } else {
+ var color = R.getRGB(params.fill);
+ fill.color = color.hex;
+ fill.src = E;
+ fill.type = "solid";
+ if (color.error && (res.type in {
+ circle: 1,
+ ellipse: 1
+ } || Str(params.fill).charAt() != "r") && addGradientFill(res, params.fill, fill)) {
+ a.fill = "none";
+ a.gradient = params.fill;
+ fill.rotate = false;
+ }
+ else if ("opacity" in color && !("fill-opacity" in params)) {
+ fillOpacity = color.opacity;
+ }
+ }
+ }
+ if (fillOpacity !== -1 || "fill-opacity" in params || "opacity" in params) {
+ var opacity = ((+a["fill-opacity"] + 1 || 2) - 1) * ((+a.opacity + 1 || 2) - 1) * ((+fillOpacity + 1 || 2) - 1);
+ opacity = mmin(mmax(opacity, 0), 1);
+ fill.opacity = opacity;
+ if (fill.src) {
+ fill.color = "none";
+ }
+ }
+ node.appendChild(fill);
+ var stroke = (node.getElementsByTagName("stroke") && node.getElementsByTagName("stroke")[0]),
+ newstroke = false;
+ !stroke && (newstroke = stroke = createNode("stroke"));
+ if ((params.stroke && params.stroke != "none") ||
+ params["stroke-width"] ||
+ params["stroke-opacity"] != null ||
+ params["stroke-dasharray"] ||
+ params["stroke-miterlimit"] ||
+ params["stroke-linejoin"] ||
+ params["stroke-linecap"]) {
+ stroke.on = true;
+ }
+ (params.stroke == "none" || params.stroke === null || stroke.on == null || params.stroke == 0 || params["stroke-width"] == 0) && (stroke.on = false);
+ var strokeColor = R.getRGB(('stroke' in params) ? params.stroke : a.stroke);
+ stroke.on && params.stroke && (stroke.color = strokeColor.hex);
+ opacity = ((+a["stroke-opacity"] + 1 || 2) - 1) * ((+a.opacity + 1 || 2) - 1) * ((+strokeColor.opacity + 1 || 2) - 1);
+ var width = (toFloat(params["stroke-width"]) || 1) * .75;
+ opacity = mmin(mmax(opacity, 0), 1);
+ params["stroke-width"] == null && (width = a["stroke-width"]);
+ params["stroke-width"] && (stroke.weight = width);
+ width && width < 1 && (opacity *= width) && (stroke.weight = 1);
+ stroke.opacity = opacity;
+
+ params["stroke-linejoin"] && (stroke.joinstyle = params["stroke-linejoin"]) || newstroke && (newstroke.joinstyle = 'miter');
+ stroke.miterlimit = params["stroke-miterlimit"] || 8;
+ params["stroke-linecap"] && (stroke.endcap = params["stroke-linecap"] == "butt" ? "flat" : params["stroke-linecap"] == "square" ? "square" : "round");
+ if (params["stroke-dasharray"]) {
+ var dasharray = {
+ "-": "shortdash",
+ ".": "shortdot",
+ "-.": "shortdashdot",
+ "-..": "shortdashdotdot",
+ ". ": "dot",
+ "- ": "dash",
+ "--": "longdash",
+ "- .": "dashdot",
+ "--.": "longdashdot",
+ "--..": "longdashdotdot"
+ };
+ stroke.dashstyle = dasharray[has](params["stroke-dasharray"]) ? dasharray[params["stroke-dasharray"]] :
+ ((params["stroke-dasharray"].join && params["stroke-dasharray"].join(' ')) || E);
+ }
+ newstroke && node.appendChild(stroke);
+ }
+ if (res.type == "text") {
+ res.paper.canvas.style.display = E;
+ var span = res.paper.span,
+ m = 100,
+ fontSize = a.font && a.font.match(/\d+(?:\.\d*)?(?=px)/),
+ lineHeight = a['line-height'] && (a['line-height']+E).match(/\d+(?:\.\d*)?(?=px)/);
+ s = span.style;
+ a.font && (s.font = a.font);
+ a["font-family"] && (s.fontFamily = a["font-family"]);
+ a["font-weight"] && (s.fontWeight = a["font-weight"]);
+ a["font-style"] && (s.fontStyle = a["font-style"]);
+ fontSize = toFloat(a["font-size"] || fontSize && fontSize[0]) || 10;
+ s.fontSize = fontSize * m + "px";
+ lineHeight = toFloat(a["line-height"] || lineHeight && lineHeight[0]) || 12;
+ a["line-height"] && (s.lineHeight = lineHeight * m + 'px');
+ res.textpath.string && (span.innerHTML = Str(res.textpath.string).replace(/"));
+ var brect = span.getBoundingClientRect();
+ res.W = a.w = (brect.right - brect.left) / m;
+ res.H = a.h = (brect.bottom - brect.top) / m;
+ // res.paper.canvas.style.display = "none";
+ res.X = a.x;
+ res.Y = a.y;
+ var leading = lineHeight - fontSize;
+ switch(a["vertical-align"]) {
+ case "top":
+ res.bby = res.H / 2; // + leading;
+ break;
+ case "bottom":
+ res.bby = -res.H / 2; // - leading;
+ break;
+ default:
+ res.bby = 0;
+ }
+
+ ("x" in params || "y" in params || res.bby !== undefined) && (res.path.v = R.format("m{0},{1}l{2},{1}", round(a.x * zoom), round((a.y + (res.bby || 0)) * zoom), round(a.x * zoom) + 1));
+ var dirtyattrs = ["x", "y", "text", "font", "font-family", "font-weight", "font-style", "font-size", "line-height"];
+ for (var d = 0, dd = dirtyattrs.length; d < dd; d++)
+ if (dirtyattrs[d] in params) {
+ res._.dirty = 1;
+ break;
+ }
+
+ // text-anchor emulation
+ switch (a["text-anchor"]) {
+ case "start":
+ res.textpath.style["v-text-align"] = "left";
+ res.bbx = res.W / 2;
+ break;
+ case "end":
+ res.textpath.style["v-text-align"] = "right";
+ res.bbx = -res.W / 2;
+ break;
+ default:
+ res.textpath.style["v-text-align"] = "center";
+ res.bbx = 0;
+ break;
+ }
+ res.textpath.style["v-text-kern"] = true;
+ }
+ // res.paper.canvas.style.display = E;
+ },
+ addGradientFill = function(o, gradient, fill) {
+ o.attrs = o.attrs || {};
+ var attrs = o.attrs,
+ pow = Math.pow,
+ opacity,
+ oindex,
+ type = "linear",
+ fxfy = ".5 .5";
+ o.attrs.gradient = gradient;
+ gradient = Str(gradient).replace(R._radial_gradient, function(all, opts) {
+ type = "radial";
+ opts = opts && opts.split(',') || [];
+
+ // fx,fy of vml is cx,cy of svg
+ var cx = opts[0],
+ cy = opts[1],
+ r = opts[2],
+ fx = opts[3],
+ fy = opts[4],
+ units = opts[5];
+ if (fx && fy) {
+ fx = toFloat(fx);
+ fy = toFloat(fy);
+ pow(fx - .5, 2) + pow(fy - .5, 2) > .25 && (fy = sqrt(.25 - pow(fx - .5, 2)) * ((fy > .5) * 2 - 1) + .5);
+ fxfy = fx + S + fy;
+ }
+ return E;
+ });
+ gradient = gradient.split(/\s*\-\s*/);
+ if (type == "linear") {
+ var angle = gradient.shift();
+ angle = -toFloat(angle);
+ if (isNaN(angle)) {
+ return null;
+ }
+ }
+ var dots = R._parseDots(gradient);
+ if (!dots) {
+ return null;
+ }
+ o = o.shape || o.node;
+ if (dots.length) {
+ o.removeChild(fill);
+ fill.on = true;
+ fill.method = "none";
+ fill.color = dots[0].color;
+ fill.color2 = dots[dots.length - 1].color;
+ //For VML use first and last available alpha
+ var clrs = [],
+ opacity1 = 1,
+ opacity2 = dots[0].opacity === undefined ? 1 : dots[0].opacity;
+ for (var i = 0, ii = dots.length; i < ii; i++) {
+ dots[i].offset && clrs.push(dots[i].offset + S + dots[i].color);
+ if (dots[i].opacity !== undefined) {
+ opacity1 = dots[i].opacity;//update with latest avaible opacity
+ }
+ }
+ fill.colors = clrs.length ? clrs.join() : "0% " + fill.color;
+ //set opacity1 & opacity2
+ fill.opacity = opacity1;
+ fill['o:opacity2'] = opacity2;
+ if (type == "radial") {
+ fill.type = "gradientTitle";
+ fill.focus = "100%";
+ fill.focussize = "0 0";
+ fill.focusposition = fxfy;
+ fill.angle = 0;
+ } else {
+ // fill.rotate= true;
+ fill.type = "gradient";
+ fill.angle = (270 - angle) % 360;
+ }
+ o.appendChild(fill);
+ }
+ return 1;
+ },
+ Element = function(node, vml, group) {
+ var o = this,
+ parent = group || vml;
+
+ o.node = o[0] = node;
+ node.raphael = true;
+ node.raphaelid = o.id = R._oid++;
+
+ o.X = 0;
+ o.Y = 0;
+
+ o.attrs = o.attrs || {};
+ o.styles = o.styles || {};
+ o.followers = o.followers || [];
+
+ o.paper = vml;
+ o.ca = o.customAttributes = o.customAttributes ||
+ new vml._CustomAttributes();
+
+ o.matrix = R.matrix();
+ o._ = {
+ transform: [],
+ sx: 1,
+ sy: 1,
+ dx: 0,
+ dy: 0,
+ deg: 0,
+ dirty: 1,
+ dirtyT: 1
+ };
+
+ o.parent = parent;
+ !parent.bottom && (parent.bottom = o);
+
+ o.prev = parent.top;
+ parent.top && (parent.top.next = o);
+ parent.top = o;
+ o.next = null;
+ };
+ var elproto = R.el;
+
+ Element.prototype = elproto;
+ elproto.constructor = Element;
+
+ elproto.transform = function(tstr) {
+ if (tstr == null) {
+ return this._.transform;
+ }
+ var vbs = this.paper._viewBoxShift,
+ vbt = vbs ? "s" + [vbs.scale, vbs.scale] + "-1-1t" + [vbs.dx, vbs.dy] : E,
+ oldt;
+
+ if (vbs) {
+ oldt = tstr = Str(tstr).replace(/\.{3}|\u2026/g, this._.transform || E);
+ }
+
+ R._extractTransform(this, vbt + tstr);
+
+ var matrix = this.matrix.clone(),
+ skew = this.skew,
+ o = this.node,
+ split,
+ isGrad = ~Str(this.attrs.fill).indexOf("-"),
+ isPatt = !Str(this.attrs.fill).indexOf("url(");
+ matrix.translate(-.5, -.5);
+ if (isPatt || isGrad || this.type == "image") {
+ skew.matrix = "1 0 0 1";
+ skew.offset = "0 0";
+ split = matrix.split();
+ if ((isGrad && split.noRotation) || !split.isSimple) {
+ o.style.filter = matrix.toFilter();
+ var bb = this.getBBox(),
+ bbt = this.getBBox(1),
+ xget = bb.x2 && bbt.x2 && 'x2' || 'x',
+ yget = bb.y2 && bbt.y2 && 'y2' || 'y',
+ dx = bb[xget] - bbt[xget],
+ dy = bb[yget] - bbt[yget];
+ o.coordorigin = (dx * -zoom) + S + (dy * -zoom);
+ setCoords(this, 1, 1, dx, dy, 0);
+ } else {
+ o.style.filter = E;
+ setCoords(this, split.scalex, split.scaley, split.dx, split.dy, split.rotate);
+ }
+ } else {
+ o.style.filter = E;
+ skew.matrix = Str(matrix);
+ skew.offset = matrix.offset();
+ }
+ oldt && (this._.transform = oldt);
+
+ return this;
+ };
+ elproto.rotate = function(deg, cx, cy) {
+ if (this.removed) {
+ return this;
+ }
+ if (deg == null) {
+ return;
+ }
+ deg = Str(deg).split(separator);
+ if (deg.length - 1) {
+ cx = toFloat(deg[1]);
+ cy = toFloat(deg[2]);
+ }
+ deg = toFloat(deg[0]);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ var bbox = this.getBBox(1);
+ cx = bbox.x + bbox.width / 2;
+ cy = bbox.y + bbox.height / 2;
+ }
+ this._.dirtyT = 1;
+ this.transform(this._.transform.concat([["r", deg, cx, cy]]));
+ return this;
+ };
+ elproto.translate = function(dx, dy) {
+ if (this.removed) {
+ return this;
+ }
+ dx = Str(dx).split(separator);
+ if (dx.length - 1) {
+ dy = toFloat(dx[1]);
+ }
+ dx = toFloat(dx[0]) || 0;
+ dy = +dy || 0;
+ if (this._.bbox) {
+ this._.bbox.x += dx;
+ this._.bbox.y += dy;
+ }
+ this.transform(this._.transform.concat([["t", dx, dy]]));
+ return this;
+ };
+ elproto.scale = function(sx, sy, cx, cy) {
+ if (this.removed) {
+ return this;
+ }
+ sx = Str(sx).split(separator);
+ if (sx.length - 1) {
+ sy = toFloat(sx[1]);
+ cx = toFloat(sx[2]);
+ cy = toFloat(sx[3]);
+ isNaN(cx) && (cx = null);
+ isNaN(cy) && (cy = null);
+ }
+ sx = toFloat(sx[0]);
+ (sy == null) && (sy = sx);
+ (cy == null) && (cx = cy);
+ if (cx == null || cy == null) {
+ var bbox = this.getBBox(1);
+ }
+ cx = cx == null ? bbox.x + bbox.width / 2 : cx;
+ cy = cy == null ? bbox.y + bbox.height / 2 : cy;
+
+ this.transform(this._.transform.concat([["s", sx, sy, cx, cy]]));
+ this._.dirtyT = 1;
+ return this;
+ };
+ elproto.hide = function(soft) {
+ var o = this;
+ !o.removed && (o.node.style.display = "none");
+ return o;
+ };
+
+ elproto.show = function(soft) {
+ var o = this;
+ !o.removed && (o.node.style.display = E);
+ return o;
+ };
+ elproto._getBBox = function() {
+ if (this.removed) {
+ return {};
+ }
+ return {
+ x: this.X + (this.bbx || 0) - this.W / 2,
+ y: this.Y + (this.bby || 0) - this.H / 2,
+ width: this.W,
+ height: this.H
+ };
+ };
+ elproto.remove = function() {
+ if (this.removed || !this.parent.canvas) {
+ return;
+ }
+ var i,
+ thisNode = R._engine.getNode(this);
+ this.paper.__set__ && this.paper.__set__.exclude(this);
+ eve.unbind("raphael.*.*." + this.id);
+ while (i = this.followers.pop()) {
+ i.el.remove();
+ }
+ this.shape && this.shape.parentNode.removeChild(this.shape);
+ thisNode.parentNode.removeChild(thisNode);
+ R._tear(this, this.paper);
+ for (var i in this) {
+ this[i] = typeof this[i] == "function" ? R._removedFactory(i) : null;
+ }
+ this.removed = true;
+ };
+ elproto.css = function (name, value) {
+ // do not parse css in case element is removed.
+ if (this.removed) {
+ return this;
+ }
+
+ // process as getter when a single key is sent as parameter.
+ if (value == null && R.is(name, "string")) {
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.styles) {
+ out[name] = this.styles[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ // process as getter when multiple keys are pre-sent as array.
+ if (value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.styles(name[i]);
+ }
+ return out;
+ }
+ // convert single key-value setter into object style standard.
+ if (value != null) {
+ var params = {};
+ params[name] = value;
+ } else if (name != null && R.is(name, "object")) {
+ params = name;
+ }
+ // iterate on keys and set style or raise events.
+ var otherkey, doattrs = {};
+ for (var key in params) {
+ otherkey = key.replace(/\B([A-Z]{1})/g, "-$1").toLowerCase();
+ // Replace "color" with fill
+ (otherkey === 'color' && this.type === 'text') && (otherkey = 'fill');
+ // If keys are supported via attr then use attr instead of css.
+ if (R._availableAttrs[has](otherkey)) {
+ doattrs[otherkey] = params[key];
+ doattrs.dirty = true;
+ continue;
+ }
+ eve("raphael.css." + otherkey + "." + this.id, this, params[key], otherkey);
+ (params[key] != undefined) && (this.node.style[otherkey] = params[key]);
+ this.styles[otherkey] = params[key];
+ }
+
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ this.followers[i].el.css(params);
+ }
+
+ // apply css via attrs
+ if (doattrs[has]("dirty")) {
+ delete doattrs.dirty;
+ this.attr(doattrs);
+ }
+
+ return this;
+ };
+ elproto.attr = function(name, value) {
+ if (this.removed) {
+ return this;
+ }
+ if (name == null) {
+ var res = {};
+ for (var a in this.attrs)
+ if (this.attrs[has](a)) {
+ res[a] = this.attrs[a];
+ }
+ res.gradient && res.fill == "none" && (res.fill = res.gradient) && delete res.gradient;
+ res.transform = this._.transform;
+ res.visibility = this.node.style.display === "none" ? "hidden" : "visible";
+ return res;
+ }
+ if (value == null && R.is(name, "string")) {
+ if (name == fillString && this.attrs.fill == "none" && this.attrs.gradient) {
+ return this.attrs.gradient;
+ }
+ if (name == "visibility") {
+ return this.node.style.display === "none" ? "hidden" : "visible";
+ }
+ var names = name.split(separator),
+ out = {};
+ for (var i = 0, ii = names.length; i < ii; i++) {
+ name = names[i];
+ if (name in this.attrs) {
+ out[name] = this.attrs[name];
+ } else if (R.is(this.ca[name], "function")) {
+ out[name] = this.ca[name].def;
+ } else {
+ out[name] = R._availableAttrs[name];
+ }
+ }
+ return ii - 1 ? out : out[names[0]];
+ }
+ if (this.attrs && value == null && R.is(name, "array")) {
+ out = {};
+ for (i = 0, ii = name.length; i < ii; i++) {
+ out[name[i]] = this.attr(name[i]);
+ }
+ return out;
+ }
+ var params;
+ if (value != null) {
+ params = {};
+ params[name] = value;
+ }
+ value == null && R.is(name, "object") && (params = name);
+ for (var key in params) {
+ eve("raphael.attr." + key + "." + this.id, this, params[key], key);
+ }
+ if (params) {
+ var todel = {};
+ for (key in this.ca)
+ if (this.ca[key] && params[has](key) && R.is(this.ca[key], "function") && !this.ca['_invoked' + key]) {
+ this.ca['_invoked' + key] = true; // prevent recursion
+ var par = this.ca[key].apply(this, [].concat(params[key]));
+ delete this.ca['_invoked' + key];
+
+ for (var subkey in par) {
+ if (par[has](subkey)) {
+ params[subkey] = par[subkey];
+ }
+ }
+ this.attrs[key] = params[key];
+ if (par === false) {
+ todel[key] = params[key];
+ delete params[key];
+ }
+ }
+ // this.paper.canvas.style.display = "none";
+ if ('text' in params && this.type == "text") {
+ this.textpath.string = params.text.replace(/ /ig, '\n');
+ }
+ setFillAndStroke(this, params);
+ var follower;
+ for (i = 0, ii = this.followers.length; i < ii; i++) {
+ follower = this.followers[i];
+ (follower.cb && !follower.cb.call(follower.el, params, this)) ||
+ follower.el.attr(params);
+ }
+ for (var subkey in todel) {
+ params[subkey] = todel[subkey];
+ }
+ // this.paper.canvas.style.display = E;
+ }
+ return this;
+ };
+
+ elproto.blur = function(size) {
+ var s = this.node.runtimeStyle,
+ f = s.filter;
+ f = f.replace(blurregexp, E);
+ if (+size !== 0) {
+ this.attrs.blur = size;
+ s.filter = f + S + ms + ".Blur(pixelradius=" + (+size || 1.5) + ")";
+ s.margin = R.format("-{0}px 0 0 -{0}px", round(+size || 1.5));
+ } else {
+ s.filter = f;
+ s.margin = 0;
+ delete this.attrs.blur;
+ }
+ return this;
+ };
+
+ elproto.on = function(eventType, handler) {
+ if (this.removed) {
+ return this;
+ }
+
+ this.node['on'+ eventType] = function() {
+ var evt = R._g.win.event;
+ evt.target = evt.srcElement;
+ handler(evt);
+ };
+ return this;
+ };
+
+ R._engine.getNode = function (el) {
+ var node = el.node || el[0].node;
+ return node.clipRect || node;
+ };
+ R._engine.getLastNode = function (el) {
+ var node = el.node || el[el.length - 1].node;
+ return node.clipRect || node;
+ };
+
+ R._engine.group = function(vml, id, group) {
+ var el = R._g.doc.createElement("div"),
+ p = new Element(el, vml, group);
+
+ el.style.cssText = cssDot;
+
+ id && (el.className = ['red', id, p.id].join('-'));
+ (group || vml).canvas.appendChild(el);
+
+ p.type = 'group';
+ p.canvas = p.node;
+ p.transform = R._engine.group.transform;
+ p.top = null;
+ p.bottom = null;
+
+ return p;
+ };
+
+ R._engine.group.transform = function(tstr) {
+ if (tstr == null) {
+ return this._.transform;
+ }
+
+ var o = this,
+ s = o.node.style,
+ c = o.clip,
+ vbs = o.paper._viewBoxShift,
+ vbt = vbs ? "s" + [vbs.scale, vbs.scale] + "-1-1t" + [vbs.dx, vbs.dy] : E,
+ oldt,
+ matrix,
+ offset,
+ tx,
+ ty;
+
+ if (vbs) {
+ oldt = tstr = Str(tstr).replace(/\.{3}|\u2026/g, o._.transform || E);
+ }
+ R._extractTransform(o, vbt + tstr);
+ matrix = o.matrix;
+ offset = matrix.offset();
+ tx = toFloat(offset[0]) || 0;
+ ty = toFloat(offset[1]) || 0;
+
+ s.left = tx + "px";
+ s.top = ty + "px";
+ s.zoom = (o._.tzoom = matrix.get(0)) + E;
+
+ /* @todo try perform relative group transform, thus avoiding
+ * transform on clipping */
+ c && (s.clip = R.format("rect({1}px {2}px {3}px {0}px)", [
+ c[0] - tx, c[1] - ty, c[2] - tx, c[3] - ty
+ ]));
+
+ return o;
+ };
+
+ R._engine.path = function(pathString, vml, group) {
+ var el = createNode("shape");
+ el.style.cssText = cssDot;
+ el.coordsize = zoom + S + zoom;
+ el.coordorigin = vml.coordorigin;
+ var p = new Element(el, vml, group),
+ attr = {
+ fill: "none",
+ stroke: "#000"
+ };
+
+ pathString && (attr.path = pathString);
+ p.type = "path";
+ p.path = [];
+ p.Path = E;
+ setFillAndStroke(p, attr);
+ (group || vml).canvas.appendChild(el);
+
+ var skew = createNode("skew");
+ skew.on = true;
+ el.appendChild(skew);
+ p.skew = skew;
+ return p;
+ };
+
+ R._engine.rect = function(vml, x, y, w, h, r, group) {
+ var path = R._rectPath(x, y, w, h, r),
+ res = vml.path(path, group),
+ a = res.attrs;
+ res.X = a.x = x;
+ res.Y = a.y = y;
+ res.W = a.width = w;
+ res.H = a.height = h;
+ a.r = r;
+ a.path = path;
+ res.type = "rect";
+ return res;
+ };
+ R._engine.ellipse = function(vml, x, y, rx, ry, group) {
+ var res = vml.path(undefined, group);
+ res.X = x - rx;
+ res.Y = y - ry;
+ res.W = rx * 2;
+ res.H = ry * 2;
+ res.type = "ellipse";
+ setFillAndStroke(res, {
+ cx: x,
+ cy: y,
+ rx: rx,
+ ry: ry
+ });
+ return res;
+ };
+ R._engine.circle = function(vml, x, y, r, group) {
+ var res = vml.path(undefined, group);
+ res.X = x - r;
+ res.Y = y - r;
+ res.W = res.H = r * 2;
+ res.type = "circle";
+ setFillAndStroke(res, {
+ cx: x,
+ cy: y,
+ r: r
+ });
+ return res;
+ };
+ R._engine.image = function(vml, src, x, y, w, h, group) {
+ var path = R._rectPath(x, y, w, h),
+ res = vml.path(path, group).attr({
+ stroke: "none"
+ }),
+ a = res.attrs,
+ node = res.node,
+ fill = node.getElementsByTagName(fillString)[0];
+ a.src = src;
+ res.X = a.x = x;
+ res.Y = a.y = y;
+ res.W = a.width = w;
+ res.H = a.height = h;
+ a.path = path;
+ res.type = "image";
+ fill.parentNode == node && node.removeChild(fill);
+ fill.rotate = true;
+ fill.src = src;
+ fill.type = "tile";
+ res._.fillpos = [x, y];
+ res._.fillsize = [w, h];
+ node.appendChild(fill);
+ setCoords(res, 1, 1, 0, 0, 0);
+ return res;
+ };
+ R._engine.text = function(vml, x, y, text, group) {
+ var el = createNode("shape"),
+ path = createNode("path"),
+ o = createNode("textpath");
+ x = x || 0;
+ y = y || 0;
+ text = text;
+ path.v = R.format("m{0},{1}l{2},{1}", round(x * zoom), round(y * zoom), round(x * zoom) + 1);
+ path.textpathok = true;
+ o.string = Str(text).replace(/ /ig, '\n');
+ o.on = true;
+ el.style.cssText = cssDot;
+ el.coordsize = zoom + S + zoom;
+ el.coordorigin = "0 0";
+ var p = new Element(el, vml, group),
+ attr = {
+ fill: "#000",
+ stroke: "none",
+ text: text
+ };
+
+ p.shape = el;
+ p.path = path;
+ p.textpath = o;
+ p.type = "text";
+ p.attrs.text = Str(text || E);
+ p.attrs.x = x;
+ p.attrs.y = y;
+ p.attrs.w = 1;
+ p.attrs.h = 1;
+ setFillAndStroke(p, attr);
+ el.appendChild(o);
+ el.appendChild(path);
+ (group || vml).canvas.appendChild(el);
+
+ var skew = createNode("skew");
+ skew.on = true;
+ el.appendChild(skew);
+ p.skew = skew;
+ return p;
+ };
+
+ R._engine.setSize = function(width, height) {
+ var cs = this.canvas.style;
+ this.width = width;
+ this.height = height;
+ width == +width && (width += "px");
+ height == +height && (height += "px");
+ cs.width = width;
+ cs.height = height;
+ cs.clip = "rect(0 " + width + " " + height + " 0)";
+ if (this._viewBox) {
+ R._engine.setViewBox.apply(this, this._viewBox);
+ }
+ return this;
+ };
+ R._engine.setViewBox = function(x, y, w, h, fit) {
+ eve("raphael.setViewBox", this, this._viewBox, [x, y, w, h, fit]);
+ var width = this.width,
+ height = this.height,
+ size = 1 / mmax(w / width, h / height),
+ H, W;
+ if (fit) {
+ H = height / h;
+ W = width / w;
+ if (w * H < width) {
+ x -= (width - w * H) / 2 / H;
+ }
+ if (h * W < height) {
+ y -= (height - h * W) / 2 / W;
+ }
+ }
+ this._viewBox = [x, y, w, h, !!fit];
+ this._viewBoxShift = {
+ dx: -x,
+ dy: -y,
+ scale: size
+ };
+ this.forEach(function(el) {
+ el.transform("...");
+ });
+ return this;
+ };
+ var createNode;
+ R._engine.initWin = function(win) {
+ var doc = win.document;
+ doc.createStyleSheet().addRule(".rvml", "behavior:url(#default#VML)");
+ try {
+ !doc.namespaces.rvml && doc.namespaces.add("rvml", "urn:schemas-microsoft-com:vml");
+ createNode = R._createNode = function(tagName, attrs) {
+ var el = doc.createElement(''),
+ prop;
+ for (prop in attrs) {
+ el[prop] = Str(attrs[prop]);
+ }
+ return el;
+ };
+ } catch (e) {
+ createNode = R._createNode = function(tagName, attrs) {
+ var el = doc.createElement('<' + tagName + ' xmlns="urn:schemas-microsoft.com:vml" class="rvml">'),
+ prop;
+ for (prop in attrs) {
+ el[prop] = Str(attrs[prop]);
+ }
+ return el;
+ };
+ }
+ };
+ R._engine.initWin(R._g.win);
+ R._engine.create = function() {
+ var con = R._getContainer.apply(0, arguments),
+ container = con.container,
+ height = con.height,
+ s,
+ width = con.width,
+ x = con.x,
+ y = con.y;
+ if (!container) {
+ throw new Error("VML container not found.");
+ }
+ var res = new R._Paper,
+ c = res.canvas = R._g.doc.createElement("div"),
+ cs = c.style;
+ x = x || 0;
+ y = y || 0;
+ width = width || 512;
+ height = height || 342;
+ res.width = width;
+ res.height = height;
+ width == +width && (width += "px");
+ height == +height && (height += "px");
+ res.coordsize = zoom * 1e3 + S + zoom * 1e3;
+ res.coordorigin = "0 0";
+ res.span = R._g.doc.createElement("span");
+ res.span.style.cssText = "position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;";
+ c.appendChild(res.span);
+ cs.cssText = R.format("top:0;left:0;width:{0};height:{1};display:inline-block;cursor:default;position:relative;clip:rect(0 {0} {1} 0);overflow:hidden", width, height);
+ if (container == 1) {
+ R._g.doc.body.appendChild(c);
+ cs.left = x + "px";
+ cs.top = y + "px";
+ cs.position = "absolute";
+ } else {
+ if (container.firstChild) {
+ container.insertBefore(c, container.firstChild);
+ } else {
+ container.appendChild(c);
+ }
+ }
+ res.renderfix = function() {
+ };
+ return res;
+ };
+ R.prototype.clear = function() {
+ eve("raphael.clear", this);
+ this.canvas.innerHTML = E;
+ this.span = R._g.doc.createElement("span");
+ this.span.style.cssText = "position:absolute;left:-9999em;top:-9999em;padding:0;margin:0;line-height:1;display:inline;";
+ this.canvas.appendChild(this.span);
+ this.bottom = this.top = null;
+ };
+ R.prototype.remove = function() {
+ eve("raphael.remove", this);
+ this.canvas.parentNode.removeChild(this.canvas);
+ for (var i in this) {
+ this[i] = typeof this[i] == "function" ? R._removedFactory(i) : null;
+ }
+ return true;
+ };
+
+ var setproto = R.st;
+ for (var method in elproto)
+ if (elproto[has](method) && !setproto[has](method)) {
+ setproto[method] = (function(methodname) {
+ return function() {
+ var arg = arguments;
+ return this.forEach(function(el) {
+ el[methodname].apply(el, arg);
+ });
+ };
+ })(method);
+ }
+}(window.Raphael);
\ No newline at end of file
diff --git a/source/separator b/source/separator
new file mode 100644
index 0000000..e69de29
diff --git a/tests/basic/baseTestSpec.js b/tests/basic/baseTestSpec.js
new file mode 100644
index 0000000..9ae3a31
--- /dev/null
+++ b/tests/basic/baseTestSpec.js
@@ -0,0 +1,40 @@
+describe('My First Test', function () {
+ var paper,
+ first,
+ second,
+ end;
+
+ it('Raphael constructor exists', function () {
+ expect(Raphael).toBeDefined(true);
+ });
+
+
+ it('Successful paper instantiation', function () {
+ paper = Raphael(0, 0, 400, 400);
+ expect(paper).toEqual(jasmine.any(Raphael));
+ });
+
+ it('Check paper', function () {
+ expect(paper).toBeDefined(true);
+ });
+
+
+ describe('My First Shape', function () {
+
+ it('is a rect', function () {
+ first = paper.rect(0, 0, 40, 40);
+ expect(first).toBe(paper.top);
+ expect(first).toBe(paper.bottom);
+ });
+ });
+
+ describe('My Second Shape', function () {
+ it('is a circle', function () {
+ second = paper.circle(50, 50, 10);
+ expect(second).toBe(paper.top);
+ expect(first).toBe(paper.bottom);
+ });
+ });
+
+});
+
diff --git a/tests/basic/basic.js b/tests/basic/basic.js
new file mode 100644
index 0000000..c30575f
--- /dev/null
+++ b/tests/basic/basic.js
@@ -0,0 +1,19 @@
+describe("RedRaphael", function () {
+ it("should be accessible through window scope", function () {
+ expect(Raphael)
+ .toBeDefined();
+ });
+
+ it("should not dirty the window scope other than exposing itself", function () {
+ expect(arr_diff(initialGlobalKeys,postloadGlobalKeys))
+ .toEqual(["postloadGlobalKeys", "eve", "Raphael"]);
+ });
+
+ describe("Creating paper", function () {
+
+ afterEach(function () {
+
+ });
+ });
+
+})
\ No newline at end of file
diff --git a/tests/basic/specSetup.js b/tests/basic/specSetup.js
new file mode 100644
index 0000000..fe41244
--- /dev/null
+++ b/tests/basic/specSetup.js
@@ -0,0 +1,12 @@
+function arr_diff(a1, a2)
+{
+ var a=[], diff=[];
+ for(var i=0;i
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/index.html b/tests/index.html
new file mode 100644
index 0000000..cb6ee7b
--- /dev/null
+++ b/tests/index.html
@@ -0,0 +1,64 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/minified.html b/tests/minified.html
new file mode 100644
index 0000000..3c3fb09
--- /dev/null
+++ b/tests/minified.html
@@ -0,0 +1,32 @@
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/rect.html b/tests/rect.html
new file mode 100644
index 0000000..4c52022
--- /dev/null
+++ b/tests/rect.html
@@ -0,0 +1,151 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/tests/runner.html b/tests/runner.html
new file mode 100644
index 0000000..9a9a327
--- /dev/null
+++ b/tests/runner.html
@@ -0,0 +1,71 @@
+
+
+
+ Jasmine Spec Runner
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/tests/runner/index.html b/tests/runner/index.html
new file mode 100644
index 0000000..9a9a327
--- /dev/null
+++ b/tests/runner/index.html
@@ -0,0 +1,71 @@
+
+
+
+ Jasmine Spec Runner
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/tests/runner/jasmine-1.3.1/MIT.LICENSE b/tests/runner/jasmine-1.3.1/MIT.LICENSE
new file mode 100644
index 0000000..7c435ba
--- /dev/null
+++ b/tests/runner/jasmine-1.3.1/MIT.LICENSE
@@ -0,0 +1,20 @@
+Copyright (c) 2008-2011 Pivotal Labs
+
+Permission is hereby granted, free of charge, to any person obtaining
+a copy of this software and associated documentation files (the
+"Software"), to deal in the Software without restriction, including
+without limitation the rights to use, copy, modify, merge, publish,
+distribute, sublicense, and/or sell copies of the Software, and to
+permit persons to whom the Software is furnished to do so, subject to
+the following conditions:
+
+The above copyright notice and this permission notice shall be
+included in all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
+EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
+MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
+NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
+LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
+OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
+WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
diff --git a/tests/runner/jasmine-1.3.1/jasmine-html.js b/tests/runner/jasmine-1.3.1/jasmine-html.js
new file mode 100644
index 0000000..543d569
--- /dev/null
+++ b/tests/runner/jasmine-1.3.1/jasmine-html.js
@@ -0,0 +1,681 @@
+jasmine.HtmlReporterHelpers = {};
+
+jasmine.HtmlReporterHelpers.createDom = function(type, attrs, childrenVarArgs) {
+ var el = document.createElement(type);
+
+ for (var i = 2; i < arguments.length; i++) {
+ var child = arguments[i];
+
+ if (typeof child === 'string') {
+ el.appendChild(document.createTextNode(child));
+ } else {
+ if (child) {
+ el.appendChild(child);
+ }
+ }
+ }
+
+ for (var attr in attrs) {
+ if (attr == "className") {
+ el[attr] = attrs[attr];
+ } else {
+ el.setAttribute(attr, attrs[attr]);
+ }
+ }
+
+ return el;
+};
+
+jasmine.HtmlReporterHelpers.getSpecStatus = function(child) {
+ var results = child.results();
+ var status = results.passed() ? 'passed' : 'failed';
+ if (results.skipped) {
+ status = 'skipped';
+ }
+
+ return status;
+};
+
+jasmine.HtmlReporterHelpers.appendToSummary = function(child, childElement) {
+ var parentDiv = this.dom.summary;
+ var parentSuite = (typeof child.parentSuite == 'undefined') ? 'suite' : 'parentSuite';
+ var parent = child[parentSuite];
+
+ if (parent) {
+ if (typeof this.views.suites[parent.id] == 'undefined') {
+ this.views.suites[parent.id] = new jasmine.HtmlReporter.SuiteView(parent, this.dom, this.views);
+ }
+ parentDiv = this.views.suites[parent.id].element;
+ }
+
+ parentDiv.appendChild(childElement);
+};
+
+
+jasmine.HtmlReporterHelpers.addHelpers = function(ctor) {
+ for(var fn in jasmine.HtmlReporterHelpers) {
+ ctor.prototype[fn] = jasmine.HtmlReporterHelpers[fn];
+ }
+};
+
+jasmine.HtmlReporter = function(_doc) {
+ var self = this;
+ var doc = _doc || window.document;
+
+ var reporterView;
+
+ var dom = {};
+
+ // Jasmine Reporter Public Interface
+ self.logRunningSpecs = false;
+
+ self.reportRunnerStarting = function(runner) {
+ var specs = runner.specs() || [];
+
+ if (specs.length == 0) {
+ return;
+ }
+
+ createReporterDom(runner.env.versionString());
+ doc.body.appendChild(dom.reporter);
+ setExceptionHandling();
+
+ reporterView = new jasmine.HtmlReporter.ReporterView(dom);
+ reporterView.addSpecs(specs, self.specFilter);
+ };
+
+ self.reportRunnerResults = function(runner) {
+ reporterView && reporterView.complete();
+ };
+
+ self.reportSuiteResults = function(suite) {
+ reporterView.suiteComplete(suite);
+ };
+
+ self.reportSpecStarting = function(spec) {
+ if (self.logRunningSpecs) {
+ self.log('>> Jasmine Running ' + spec.suite.description + ' ' + spec.description + '...');
+ }
+ };
+
+ self.reportSpecResults = function(spec) {
+ reporterView.specComplete(spec);
+ };
+
+ self.log = function() {
+ var console = jasmine.getGlobal().console;
+ if (console && console.log) {
+ if (console.log.apply) {
+ console.log.apply(console, arguments);
+ } else {
+ console.log(arguments); // ie fix: console.log.apply doesn't exist on ie
+ }
+ }
+ };
+
+ self.specFilter = function(spec) {
+ if (!focusedSpecName()) {
+ return true;
+ }
+
+ return spec.getFullName().indexOf(focusedSpecName()) === 0;
+ };
+
+ return self;
+
+ function focusedSpecName() {
+ var specName;
+
+ (function memoizeFocusedSpec() {
+ if (specName) {
+ return;
+ }
+
+ var paramMap = [];
+ var params = jasmine.HtmlReporter.parameters(doc);
+
+ for (var i = 0; i < params.length; i++) {
+ var p = params[i].split('=');
+ paramMap[decodeURIComponent(p[0])] = decodeURIComponent(p[1]);
+ }
+
+ specName = paramMap.spec;
+ })();
+
+ return specName;
+ }
+
+ function createReporterDom(version) {
+ dom.reporter = self.createDom('div', { id: 'HTMLReporter', className: 'jasmine_reporter' },
+ dom.banner = self.createDom('div', { className: 'banner' },
+ self.createDom('span', { className: 'title' }, "Jasmine "),
+ self.createDom('span', { className: 'version' }, version)),
+
+ dom.symbolSummary = self.createDom('ul', {className: 'symbolSummary'}),
+ dom.alert = self.createDom('div', {className: 'alert'},
+ self.createDom('span', { className: 'exceptions' },
+ self.createDom('label', { className: 'label', 'for': 'no_try_catch' }, 'No try/catch'),
+ self.createDom('input', { id: 'no_try_catch', type: 'checkbox' }))),
+ dom.results = self.createDom('div', {className: 'results'},
+ dom.summary = self.createDom('div', { className: 'summary' }),
+ dom.details = self.createDom('div', { id: 'details' }))
+ );
+ }
+
+ function noTryCatch() {
+ return window.location.search.match(/catch=false/);
+ }
+
+ function searchWithCatch() {
+ var params = jasmine.HtmlReporter.parameters(window.document);
+ var removed = false;
+ var i = 0;
+
+ while (!removed && i < params.length) {
+ if (params[i].match(/catch=/)) {
+ params.splice(i, 1);
+ removed = true;
+ }
+ i++;
+ }
+ if (jasmine.CATCH_EXCEPTIONS) {
+ params.push("catch=false");
+ }
+
+ return params.join("&");
+ }
+
+ function setExceptionHandling() {
+ var chxCatch = document.getElementById('no_try_catch');
+
+ if (noTryCatch()) {
+ chxCatch.setAttribute('checked', true);
+ jasmine.CATCH_EXCEPTIONS = false;
+ }
+ chxCatch.onclick = function() {
+ window.location.search = searchWithCatch();
+ };
+ }
+};
+jasmine.HtmlReporter.parameters = function(doc) {
+ var paramStr = doc.location.search.substring(1);
+ var params = [];
+
+ if (paramStr.length > 0) {
+ params = paramStr.split('&');
+ }
+ return params;
+}
+jasmine.HtmlReporter.sectionLink = function(sectionName) {
+ var link = '?';
+ var params = [];
+
+ if (sectionName) {
+ params.push('spec=' + encodeURIComponent(sectionName));
+ }
+ if (!jasmine.CATCH_EXCEPTIONS) {
+ params.push("catch=false");
+ }
+ if (params.length > 0) {
+ link += params.join("&");
+ }
+
+ return link;
+};
+jasmine.HtmlReporterHelpers.addHelpers(jasmine.HtmlReporter);
+jasmine.HtmlReporter.ReporterView = function(dom) {
+ this.startedAt = new Date();
+ this.runningSpecCount = 0;
+ this.completeSpecCount = 0;
+ this.passedCount = 0;
+ this.failedCount = 0;
+ this.skippedCount = 0;
+
+ this.createResultsMenu = function() {
+ this.resultsMenu = this.createDom('span', {className: 'resultsMenu bar'},
+ this.summaryMenuItem = this.createDom('a', {className: 'summaryMenuItem', href: "#"}, '0 specs'),
+ ' | ',
+ this.detailsMenuItem = this.createDom('a', {className: 'detailsMenuItem', href: "#"}, '0 failing'));
+
+ this.summaryMenuItem.onclick = function() {
+ dom.reporter.className = dom.reporter.className.replace(/ showDetails/g, '');
+ };
+
+ this.detailsMenuItem.onclick = function() {
+ showDetails();
+ };
+ };
+
+ this.addSpecs = function(specs, specFilter) {
+ this.totalSpecCount = specs.length;
+
+ this.views = {
+ specs: {},
+ suites: {}
+ };
+
+ for (var i = 0; i < specs.length; i++) {
+ var spec = specs[i];
+ this.views.specs[spec.id] = new jasmine.HtmlReporter.SpecView(spec, dom, this.views);
+ if (specFilter(spec)) {
+ this.runningSpecCount++;
+ }
+ }
+ };
+
+ this.specComplete = function(spec) {
+ this.completeSpecCount++;
+
+ if (isUndefined(this.views.specs[spec.id])) {
+ this.views.specs[spec.id] = new jasmine.HtmlReporter.SpecView(spec, dom);
+ }
+
+ var specView = this.views.specs[spec.id];
+
+ switch (specView.status()) {
+ case 'passed':
+ this.passedCount++;
+ break;
+
+ case 'failed':
+ this.failedCount++;
+ break;
+
+ case 'skipped':
+ this.skippedCount++;
+ break;
+ }
+
+ specView.refresh();
+ this.refresh();
+ };
+
+ this.suiteComplete = function(suite) {
+ var suiteView = this.views.suites[suite.id];
+ if (isUndefined(suiteView)) {
+ return;
+ }
+ suiteView.refresh();
+ };
+
+ this.refresh = function() {
+
+ if (isUndefined(this.resultsMenu)) {
+ this.createResultsMenu();
+ }
+
+ // currently running UI
+ if (isUndefined(this.runningAlert)) {
+ this.runningAlert = this.createDom('a', { href: jasmine.HtmlReporter.sectionLink(), className: "runningAlert bar" });
+ dom.alert.appendChild(this.runningAlert);
+ }
+ this.runningAlert.innerHTML = "Running " + this.completeSpecCount + " of " + specPluralizedFor(this.totalSpecCount);
+
+ // skipped specs UI
+ if (isUndefined(this.skippedAlert)) {
+ this.skippedAlert = this.createDom('a', { href: jasmine.HtmlReporter.sectionLink(), className: "skippedAlert bar" });
+ }
+
+ this.skippedAlert.innerHTML = "Skipping " + this.skippedCount + " of " + specPluralizedFor(this.totalSpecCount) + " - run all";
+
+ if (this.skippedCount === 1 && isDefined(dom.alert)) {
+ dom.alert.appendChild(this.skippedAlert);
+ }
+
+ // passing specs UI
+ if (isUndefined(this.passedAlert)) {
+ this.passedAlert = this.createDom('span', { href: jasmine.HtmlReporter.sectionLink(), className: "passingAlert bar" });
+ }
+ this.passedAlert.innerHTML = "Passing " + specPluralizedFor(this.passedCount);
+
+ // failing specs UI
+ if (isUndefined(this.failedAlert)) {
+ this.failedAlert = this.createDom('span', {href: "?", className: "failingAlert bar"});
+ }
+ this.failedAlert.innerHTML = "Failing " + specPluralizedFor(this.failedCount);
+
+ if (this.failedCount === 1 && isDefined(dom.alert)) {
+ dom.alert.appendChild(this.failedAlert);
+ dom.alert.appendChild(this.resultsMenu);
+ }
+
+ // summary info
+ this.summaryMenuItem.innerHTML = "" + specPluralizedFor(this.runningSpecCount);
+ this.detailsMenuItem.innerHTML = "" + this.failedCount + " failing";
+ };
+
+ this.complete = function() {
+ dom.alert.removeChild(this.runningAlert);
+
+ this.skippedAlert.innerHTML = "Ran " + this.runningSpecCount + " of " + specPluralizedFor(this.totalSpecCount) + " - run all";
+
+ if (this.failedCount === 0) {
+ dom.alert.appendChild(this.createDom('span', {className: 'passingAlert bar'}, "Passing " + specPluralizedFor(this.passedCount)));
+ } else {
+ showDetails();
+ }
+
+ dom.banner.appendChild(this.createDom('span', {className: 'duration'}, "finished in " + ((new Date().getTime() - this.startedAt.getTime()) / 1000) + "s"));
+ };
+
+ return this;
+
+ function showDetails() {
+ if (dom.reporter.className.search(/showDetails/) === -1) {
+ dom.reporter.className += " showDetails";
+ }
+ }
+
+ function isUndefined(obj) {
+ return typeof obj === 'undefined';
+ }
+
+ function isDefined(obj) {
+ return !isUndefined(obj);
+ }
+
+ function specPluralizedFor(count) {
+ var str = count + " spec";
+ if (count > 1) {
+ str += "s"
+ }
+ return str;
+ }
+
+};
+
+jasmine.HtmlReporterHelpers.addHelpers(jasmine.HtmlReporter.ReporterView);
+
+
+jasmine.HtmlReporter.SpecView = function(spec, dom, views) {
+ this.spec = spec;
+ this.dom = dom;
+ this.views = views;
+
+ this.symbol = this.createDom('li', { className: 'pending' });
+ this.dom.symbolSummary.appendChild(this.symbol);
+
+ this.summary = this.createDom('div', { className: 'specSummary' },
+ this.createDom('a', {
+ className: 'description',
+ href: jasmine.HtmlReporter.sectionLink(this.spec.getFullName()),
+ title: this.spec.getFullName()
+ }, this.spec.description)
+ );
+
+ this.detail = this.createDom('div', { className: 'specDetail' },
+ this.createDom('a', {
+ className: 'description',
+ href: '?spec=' + encodeURIComponent(this.spec.getFullName()),
+ title: this.spec.getFullName()
+ }, this.spec.getFullName())
+ );
+};
+
+jasmine.HtmlReporter.SpecView.prototype.status = function() {
+ return this.getSpecStatus(this.spec);
+};
+
+jasmine.HtmlReporter.SpecView.prototype.refresh = function() {
+ this.symbol.className = this.status();
+
+ switch (this.status()) {
+ case 'skipped':
+ break;
+
+ case 'passed':
+ this.appendSummaryToSuiteDiv();
+ break;
+
+ case 'failed':
+ this.appendSummaryToSuiteDiv();
+ this.appendFailureDetail();
+ break;
+ }
+};
+
+jasmine.HtmlReporter.SpecView.prototype.appendSummaryToSuiteDiv = function() {
+ this.summary.className += ' ' + this.status();
+ this.appendToSummary(this.spec, this.summary);
+};
+
+jasmine.HtmlReporter.SpecView.prototype.appendFailureDetail = function() {
+ this.detail.className += ' ' + this.status();
+
+ var resultItems = this.spec.results().getItems();
+ var messagesDiv = this.createDom('div', { className: 'messages' });
+
+ for (var i = 0; i < resultItems.length; i++) {
+ var result = resultItems[i];
+
+ if (result.type == 'log') {
+ messagesDiv.appendChild(this.createDom('div', {className: 'resultMessage log'}, result.toString()));
+ } else if (result.type == 'expect' && result.passed && !result.passed()) {
+ messagesDiv.appendChild(this.createDom('div', {className: 'resultMessage fail'}, result.message));
+
+ if (result.trace.stack) {
+ messagesDiv.appendChild(this.createDom('div', {className: 'stackTrace'}, result.trace.stack));
+ }
+ }
+ }
+
+ if (messagesDiv.childNodes.length > 0) {
+ this.detail.appendChild(messagesDiv);
+ this.dom.details.appendChild(this.detail);
+ }
+};
+
+jasmine.HtmlReporterHelpers.addHelpers(jasmine.HtmlReporter.SpecView);jasmine.HtmlReporter.SuiteView = function(suite, dom, views) {
+ this.suite = suite;
+ this.dom = dom;
+ this.views = views;
+
+ this.element = this.createDom('div', { className: 'suite' },
+ this.createDom('a', { className: 'description', href: jasmine.HtmlReporter.sectionLink(this.suite.getFullName()) }, this.suite.description)
+ );
+
+ this.appendToSummary(this.suite, this.element);
+};
+
+jasmine.HtmlReporter.SuiteView.prototype.status = function() {
+ return this.getSpecStatus(this.suite);
+};
+
+jasmine.HtmlReporter.SuiteView.prototype.refresh = function() {
+ this.element.className += " " + this.status();
+};
+
+jasmine.HtmlReporterHelpers.addHelpers(jasmine.HtmlReporter.SuiteView);
+
+/* @deprecated Use jasmine.HtmlReporter instead
+ */
+jasmine.TrivialReporter = function(doc) {
+ this.document = doc || document;
+ this.suiteDivs = {};
+ this.logRunningSpecs = false;
+};
+
+jasmine.TrivialReporter.prototype.createDom = function(type, attrs, childrenVarArgs) {
+ var el = document.createElement(type);
+
+ for (var i = 2; i < arguments.length; i++) {
+ var child = arguments[i];
+
+ if (typeof child === 'string') {
+ el.appendChild(document.createTextNode(child));
+ } else {
+ if (child) { el.appendChild(child); }
+ }
+ }
+
+ for (var attr in attrs) {
+ if (attr == "className") {
+ el[attr] = attrs[attr];
+ } else {
+ el.setAttribute(attr, attrs[attr]);
+ }
+ }
+
+ return el;
+};
+
+jasmine.TrivialReporter.prototype.reportRunnerStarting = function(runner) {
+ var showPassed, showSkipped;
+
+ this.outerDiv = this.createDom('div', { id: 'TrivialReporter', className: 'jasmine_reporter' },
+ this.createDom('div', { className: 'banner' },
+ this.createDom('div', { className: 'logo' },
+ this.createDom('span', { className: 'title' }, "Jasmine"),
+ this.createDom('span', { className: 'version' }, runner.env.versionString())),
+ this.createDom('div', { className: 'options' },
+ "Show ",
+ showPassed = this.createDom('input', { id: "__jasmine_TrivialReporter_showPassed__", type: 'checkbox' }),
+ this.createDom('label', { "for": "__jasmine_TrivialReporter_showPassed__" }, " passed "),
+ showSkipped = this.createDom('input', { id: "__jasmine_TrivialReporter_showSkipped__", type: 'checkbox' }),
+ this.createDom('label', { "for": "__jasmine_TrivialReporter_showSkipped__" }, " skipped")
+ )
+ ),
+
+ this.runnerDiv = this.createDom('div', { className: 'runner running' },
+ this.createDom('a', { className: 'run_spec', href: '?' }, "run all"),
+ this.runnerMessageSpan = this.createDom('span', {}, "Running..."),
+ this.finishedAtSpan = this.createDom('span', { className: 'finished-at' }, ""))
+ );
+
+ this.document.body.appendChild(this.outerDiv);
+
+ var suites = runner.suites();
+ for (var i = 0; i < suites.length; i++) {
+ var suite = suites[i];
+ var suiteDiv = this.createDom('div', { className: 'suite' },
+ this.createDom('a', { className: 'run_spec', href: '?spec=' + encodeURIComponent(suite.getFullName()) }, "run"),
+ this.createDom('a', { className: 'description', href: '?spec=' + encodeURIComponent(suite.getFullName()) }, suite.description));
+ this.suiteDivs[suite.id] = suiteDiv;
+ var parentDiv = this.outerDiv;
+ if (suite.parentSuite) {
+ parentDiv = this.suiteDivs[suite.parentSuite.id];
+ }
+ parentDiv.appendChild(suiteDiv);
+ }
+
+ this.startedAt = new Date();
+
+ var self = this;
+ showPassed.onclick = function(evt) {
+ if (showPassed.checked) {
+ self.outerDiv.className += ' show-passed';
+ } else {
+ self.outerDiv.className = self.outerDiv.className.replace(/ show-passed/, '');
+ }
+ };
+
+ showSkipped.onclick = function(evt) {
+ if (showSkipped.checked) {
+ self.outerDiv.className += ' show-skipped';
+ } else {
+ self.outerDiv.className = self.outerDiv.className.replace(/ show-skipped/, '');
+ }
+ };
+};
+
+jasmine.TrivialReporter.prototype.reportRunnerResults = function(runner) {
+ var results = runner.results();
+ var className = (results.failedCount > 0) ? "runner failed" : "runner passed";
+ this.runnerDiv.setAttribute("class", className);
+ //do it twice for IE
+ this.runnerDiv.setAttribute("className", className);
+ var specs = runner.specs();
+ var specCount = 0;
+ for (var i = 0; i < specs.length; i++) {
+ if (this.specFilter(specs[i])) {
+ specCount++;
+ }
+ }
+ var message = "" + specCount + " spec" + (specCount == 1 ? "" : "s" ) + ", " + results.failedCount + " failure" + ((results.failedCount == 1) ? "" : "s");
+ message += " in " + ((new Date().getTime() - this.startedAt.getTime()) / 1000) + "s";
+ this.runnerMessageSpan.replaceChild(this.createDom('a', { className: 'description', href: '?'}, message), this.runnerMessageSpan.firstChild);
+
+ this.finishedAtSpan.appendChild(document.createTextNode("Finished at " + new Date().toString()));
+};
+
+jasmine.TrivialReporter.prototype.reportSuiteResults = function(suite) {
+ var results = suite.results();
+ var status = results.passed() ? 'passed' : 'failed';
+ if (results.totalCount === 0) { // todo: change this to check results.skipped
+ status = 'skipped';
+ }
+ this.suiteDivs[suite.id].className += " " + status;
+};
+
+jasmine.TrivialReporter.prototype.reportSpecStarting = function(spec) {
+ if (this.logRunningSpecs) {
+ this.log('>> Jasmine Running ' + spec.suite.description + ' ' + spec.description + '...');
+ }
+};
+
+jasmine.TrivialReporter.prototype.reportSpecResults = function(spec) {
+ var results = spec.results();
+ var status = results.passed() ? 'passed' : 'failed';
+ if (results.skipped) {
+ status = 'skipped';
+ }
+ var specDiv = this.createDom('div', { className: 'spec ' + status },
+ this.createDom('a', { className: 'run_spec', href: '?spec=' + encodeURIComponent(spec.getFullName()) }, "run"),
+ this.createDom('a', {
+ className: 'description',
+ href: '?spec=' + encodeURIComponent(spec.getFullName()),
+ title: spec.getFullName()
+ }, spec.description));
+
+
+ var resultItems = results.getItems();
+ var messagesDiv = this.createDom('div', { className: 'messages' });
+ for (var i = 0; i < resultItems.length; i++) {
+ var result = resultItems[i];
+
+ if (result.type == 'log') {
+ messagesDiv.appendChild(this.createDom('div', {className: 'resultMessage log'}, result.toString()));
+ } else if (result.type == 'expect' && result.passed && !result.passed()) {
+ messagesDiv.appendChild(this.createDom('div', {className: 'resultMessage fail'}, result.message));
+
+ if (result.trace.stack) {
+ messagesDiv.appendChild(this.createDom('div', {className: 'stackTrace'}, result.trace.stack));
+ }
+ }
+ }
+
+ if (messagesDiv.childNodes.length > 0) {
+ specDiv.appendChild(messagesDiv);
+ }
+
+ this.suiteDivs[spec.suite.id].appendChild(specDiv);
+};
+
+jasmine.TrivialReporter.prototype.log = function() {
+ var console = jasmine.getGlobal().console;
+ if (console && console.log) {
+ if (console.log.apply) {
+ console.log.apply(console, arguments);
+ } else {
+ console.log(arguments); // ie fix: console.log.apply doesn't exist on ie
+ }
+ }
+};
+
+jasmine.TrivialReporter.prototype.getLocation = function() {
+ return this.document.location;
+};
+
+jasmine.TrivialReporter.prototype.specFilter = function(spec) {
+ var paramMap = {};
+ var params = this.getLocation().search.substring(1).split('&');
+ for (var i = 0; i < params.length; i++) {
+ var p = params[i].split('=');
+ paramMap[decodeURIComponent(p[0])] = decodeURIComponent(p[1]);
+ }
+
+ if (!paramMap.spec) {
+ return true;
+ }
+ return spec.getFullName().indexOf(paramMap.spec) === 0;
+};
diff --git a/tests/runner/jasmine-1.3.1/jasmine.css b/tests/runner/jasmine-1.3.1/jasmine.css
new file mode 100644
index 0000000..8c008dc
--- /dev/null
+++ b/tests/runner/jasmine-1.3.1/jasmine.css
@@ -0,0 +1,82 @@
+body { background-color: #eeeeee; padding: 0; margin: 5px; overflow-y: scroll; }
+
+#HTMLReporter { font-size: 11px; font-family: Monaco, "Lucida Console", monospace; line-height: 14px; color: #333333; }
+#HTMLReporter a { text-decoration: none; }
+#HTMLReporter a:hover { text-decoration: underline; }
+#HTMLReporter p, #HTMLReporter h1, #HTMLReporter h2, #HTMLReporter h3, #HTMLReporter h4, #HTMLReporter h5, #HTMLReporter h6 { margin: 0; line-height: 14px; }
+#HTMLReporter .banner, #HTMLReporter .symbolSummary, #HTMLReporter .summary, #HTMLReporter .resultMessage, #HTMLReporter .specDetail .description, #HTMLReporter .alert .bar, #HTMLReporter .stackTrace { padding-left: 9px; padding-right: 9px; }
+#HTMLReporter #jasmine_content { position: fixed; right: 100%; }
+#HTMLReporter .version { color: #aaaaaa; }
+#HTMLReporter .banner { margin-top: 14px; }
+#HTMLReporter .duration { color: #aaaaaa; float: right; }
+#HTMLReporter .symbolSummary { overflow: hidden; *zoom: 1; margin: 14px 0; }
+#HTMLReporter .symbolSummary li { display: block; float: left; height: 7px; width: 14px; margin-bottom: 7px; font-size: 16px; }
+#HTMLReporter .symbolSummary li.passed { font-size: 14px; }
+#HTMLReporter .symbolSummary li.passed:before { color: #5e7d00; content: "\02022"; }
+#HTMLReporter .symbolSummary li.failed { line-height: 9px; }
+#HTMLReporter .symbolSummary li.failed:before { color: #b03911; content: "x"; font-weight: bold; margin-left: -1px; }
+#HTMLReporter .symbolSummary li.skipped { font-size: 14px; }
+#HTMLReporter .symbolSummary li.skipped:before { color: #bababa; content: "\02022"; }
+#HTMLReporter .symbolSummary li.pending { line-height: 11px; }
+#HTMLReporter .symbolSummary li.pending:before { color: #aaaaaa; content: "-"; }
+#HTMLReporter .exceptions { color: #fff; float: right; margin-top: 5px; margin-right: 5px; }
+#HTMLReporter .bar { line-height: 28px; font-size: 14px; display: block; color: #eee; }
+#HTMLReporter .runningAlert { background-color: #666666; }
+#HTMLReporter .skippedAlert { background-color: #aaaaaa; }
+#HTMLReporter .skippedAlert:first-child { background-color: #333333; }
+#HTMLReporter .skippedAlert:hover { text-decoration: none; color: white; text-decoration: underline; }
+#HTMLReporter .passingAlert { background-color: #a6b779; }
+#HTMLReporter .passingAlert:first-child { background-color: #5e7d00; }
+#HTMLReporter .failingAlert { background-color: #cf867e; }
+#HTMLReporter .failingAlert:first-child { background-color: #b03911; }
+#HTMLReporter .results { margin-top: 14px; }
+#HTMLReporter #details { display: none; }
+#HTMLReporter .resultsMenu, #HTMLReporter .resultsMenu a { background-color: #fff; color: #333333; }
+#HTMLReporter.showDetails .summaryMenuItem { font-weight: normal; text-decoration: inherit; }
+#HTMLReporter.showDetails .summaryMenuItem:hover { text-decoration: underline; }
+#HTMLReporter.showDetails .detailsMenuItem { font-weight: bold; text-decoration: underline; }
+#HTMLReporter.showDetails .summary { display: none; }
+#HTMLReporter.showDetails #details { display: block; }
+#HTMLReporter .summaryMenuItem { font-weight: bold; text-decoration: underline; }
+#HTMLReporter .summary { margin-top: 14px; }
+#HTMLReporter .summary .suite .suite, #HTMLReporter .summary .specSummary { margin-left: 14px; }
+#HTMLReporter .summary .specSummary.passed a { color: #5e7d00; }
+#HTMLReporter .summary .specSummary.failed a { color: #b03911; }
+#HTMLReporter .description + .suite { margin-top: 0; }
+#HTMLReporter .suite { margin-top: 14px; }
+#HTMLReporter .suite a { color: #333333; }
+#HTMLReporter #details .specDetail { margin-bottom: 28px; }
+#HTMLReporter #details .specDetail .description { display: block; color: white; background-color: #b03911; }
+#HTMLReporter .resultMessage { padding-top: 14px; color: #333333; }
+#HTMLReporter .resultMessage span.result { display: block; }
+#HTMLReporter .stackTrace { margin: 5px 0 0 0; max-height: 224px; overflow: auto; line-height: 18px; color: #666666; border: 1px solid #ddd; background: white; white-space: pre; }
+
+#TrivialReporter { padding: 8px 13px; position: absolute; top: 0; bottom: 0; left: 0; right: 0; overflow-y: scroll; background-color: white; font-family: "Helvetica Neue Light", "Lucida Grande", "Calibri", "Arial", sans-serif; /*.resultMessage {*/ /*white-space: pre;*/ /*}*/ }
+#TrivialReporter a:visited, #TrivialReporter a { color: #303; }
+#TrivialReporter a:hover, #TrivialReporter a:active { color: blue; }
+#TrivialReporter .run_spec { float: right; padding-right: 5px; font-size: .8em; text-decoration: none; }
+#TrivialReporter .banner { color: #303; background-color: #fef; padding: 5px; }
+#TrivialReporter .logo { float: left; font-size: 1.1em; padding-left: 5px; }
+#TrivialReporter .logo .version { font-size: .6em; padding-left: 1em; }
+#TrivialReporter .runner.running { background-color: yellow; }
+#TrivialReporter .options { text-align: right; font-size: .8em; }
+#TrivialReporter .suite { border: 1px outset gray; margin: 5px 0; padding-left: 1em; }
+#TrivialReporter .suite .suite { margin: 5px; }
+#TrivialReporter .suite.passed { background-color: #dfd; }
+#TrivialReporter .suite.failed { background-color: #fdd; }
+#TrivialReporter .spec { margin: 5px; padding-left: 1em; clear: both; }
+#TrivialReporter .spec.failed, #TrivialReporter .spec.passed, #TrivialReporter .spec.skipped { padding-bottom: 5px; border: 1px solid gray; }
+#TrivialReporter .spec.failed { background-color: #fbb; border-color: red; }
+#TrivialReporter .spec.passed { background-color: #bfb; border-color: green; }
+#TrivialReporter .spec.skipped { background-color: #bbb; }
+#TrivialReporter .messages { border-left: 1px dashed gray; padding-left: 1em; padding-right: 1em; }
+#TrivialReporter .passed { background-color: #cfc; display: none; }
+#TrivialReporter .failed { background-color: #fbb; }
+#TrivialReporter .skipped { color: #777; background-color: #eee; display: none; }
+#TrivialReporter .resultMessage span.result { display: block; line-height: 2em; color: black; }
+#TrivialReporter .resultMessage .mismatch { color: black; }
+#TrivialReporter .stackTrace { white-space: pre; font-size: .8em; margin-left: 10px; max-height: 5em; overflow: auto; border: 1px inset red; padding: 1em; background: #eef; }
+#TrivialReporter .finished-at { padding-left: 1em; font-size: .6em; }
+#TrivialReporter.show-passed .passed, #TrivialReporter.show-skipped .skipped { display: block; }
+#TrivialReporter #jasmine_content { position: fixed; right: 100%; }
+#TrivialReporter .runner { border: 1px solid gray; display: block; margin: 5px 0; padding: 2px 0 2px 10px; }
diff --git a/tests/runner/jasmine-1.3.1/jasmine.js b/tests/runner/jasmine-1.3.1/jasmine.js
new file mode 100644
index 0000000..6b3459b
--- /dev/null
+++ b/tests/runner/jasmine-1.3.1/jasmine.js
@@ -0,0 +1,2600 @@
+var isCommonJS = typeof window == "undefined" && typeof exports == "object";
+
+/**
+ * Top level namespace for Jasmine, a lightweight JavaScript BDD/spec/testing framework.
+ *
+ * @namespace
+ */
+var jasmine = {};
+if (isCommonJS) exports.jasmine = jasmine;
+/**
+ * @private
+ */
+jasmine.unimplementedMethod_ = function() {
+ throw new Error("unimplemented method");
+};
+
+/**
+ * Use jasmine.undefined
instead of undefined
, since undefined
is just
+ * a plain old variable and may be redefined by somebody else.
+ *
+ * @private
+ */
+jasmine.undefined = jasmine.___undefined___;
+
+/**
+ * Show diagnostic messages in the console if set to true
+ *
+ */
+jasmine.VERBOSE = false;
+
+/**
+ * Default interval in milliseconds for event loop yields (e.g. to allow network activity or to refresh the screen with the HTML-based runner). Small values here may result in slow test running. Zero means no updates until all tests have completed.
+ *
+ */
+jasmine.DEFAULT_UPDATE_INTERVAL = 250;
+
+/**
+ * Maximum levels of nesting that will be included when an object is pretty-printed
+ */
+jasmine.MAX_PRETTY_PRINT_DEPTH = 40;
+
+/**
+ * Default timeout interval in milliseconds for waitsFor() blocks.
+ */
+jasmine.DEFAULT_TIMEOUT_INTERVAL = 5000;
+
+/**
+ * By default exceptions thrown in the context of a test are caught by jasmine so that it can run the remaining tests in the suite.
+ * Set to false to let the exception bubble up in the browser.
+ *
+ */
+jasmine.CATCH_EXCEPTIONS = true;
+
+jasmine.getGlobal = function() {
+ function getGlobal() {
+ return this;
+ }
+
+ return getGlobal();
+};
+
+/**
+ * Allows for bound functions to be compared. Internal use only.
+ *
+ * @ignore
+ * @private
+ * @param base {Object} bound 'this' for the function
+ * @param name {Function} function to find
+ */
+jasmine.bindOriginal_ = function(base, name) {
+ var original = base[name];
+ if (original.apply) {
+ return function() {
+ return original.apply(base, arguments);
+ };
+ } else {
+ // IE support
+ return jasmine.getGlobal()[name];
+ }
+};
+
+jasmine.setTimeout = jasmine.bindOriginal_(jasmine.getGlobal(), 'setTimeout');
+jasmine.clearTimeout = jasmine.bindOriginal_(jasmine.getGlobal(), 'clearTimeout');
+jasmine.setInterval = jasmine.bindOriginal_(jasmine.getGlobal(), 'setInterval');
+jasmine.clearInterval = jasmine.bindOriginal_(jasmine.getGlobal(), 'clearInterval');
+
+jasmine.MessageResult = function(values) {
+ this.type = 'log';
+ this.values = values;
+ this.trace = new Error(); // todo: test better
+};
+
+jasmine.MessageResult.prototype.toString = function() {
+ var text = "";
+ for (var i = 0; i < this.values.length; i++) {
+ if (i > 0) text += " ";
+ if (jasmine.isString_(this.values[i])) {
+ text += this.values[i];
+ } else {
+ text += jasmine.pp(this.values[i]);
+ }
+ }
+ return text;
+};
+
+jasmine.ExpectationResult = function(params) {
+ this.type = 'expect';
+ this.matcherName = params.matcherName;
+ this.passed_ = params.passed;
+ this.expected = params.expected;
+ this.actual = params.actual;
+ this.message = this.passed_ ? 'Passed.' : params.message;
+
+ var trace = (params.trace || new Error(this.message));
+ this.trace = this.passed_ ? '' : trace;
+};
+
+jasmine.ExpectationResult.prototype.toString = function () {
+ return this.message;
+};
+
+jasmine.ExpectationResult.prototype.passed = function () {
+ return this.passed_;
+};
+
+/**
+ * Getter for the Jasmine environment. Ensures one gets created
+ */
+jasmine.getEnv = function() {
+ var env = jasmine.currentEnv_ = jasmine.currentEnv_ || new jasmine.Env();
+ return env;
+};
+
+/**
+ * @ignore
+ * @private
+ * @param value
+ * @returns {Boolean}
+ */
+jasmine.isArray_ = function(value) {
+ return jasmine.isA_("Array", value);
+};
+
+/**
+ * @ignore
+ * @private
+ * @param value
+ * @returns {Boolean}
+ */
+jasmine.isString_ = function(value) {
+ return jasmine.isA_("String", value);
+};
+
+/**
+ * @ignore
+ * @private
+ * @param value
+ * @returns {Boolean}
+ */
+jasmine.isNumber_ = function(value) {
+ return jasmine.isA_("Number", value);
+};
+
+/**
+ * @ignore
+ * @private
+ * @param {String} typeName
+ * @param value
+ * @returns {Boolean}
+ */
+jasmine.isA_ = function(typeName, value) {
+ return Object.prototype.toString.apply(value) === '[object ' + typeName + ']';
+};
+
+/**
+ * Pretty printer for expecations. Takes any object and turns it into a human-readable string.
+ *
+ * @param value {Object} an object to be outputted
+ * @returns {String}
+ */
+jasmine.pp = function(value) {
+ var stringPrettyPrinter = new jasmine.StringPrettyPrinter();
+ stringPrettyPrinter.format(value);
+ return stringPrettyPrinter.string;
+};
+
+/**
+ * Returns true if the object is a DOM Node.
+ *
+ * @param {Object} obj object to check
+ * @returns {Boolean}
+ */
+jasmine.isDomNode = function(obj) {
+ return obj.nodeType > 0;
+};
+
+/**
+ * Returns a matchable 'generic' object of the class type. For use in expecations of type when values don't matter.
+ *
+ * @example
+ * // don't care about which function is passed in, as long as it's a function
+ * expect(mySpy).toHaveBeenCalledWith(jasmine.any(Function));
+ *
+ * @param {Class} clazz
+ * @returns matchable object of the type clazz
+ */
+jasmine.any = function(clazz) {
+ return new jasmine.Matchers.Any(clazz);
+};
+
+/**
+ * Returns a matchable subset of a JSON object. For use in expectations when you don't care about all of the
+ * attributes on the object.
+ *
+ * @example
+ * // don't care about any other attributes than foo.
+ * expect(mySpy).toHaveBeenCalledWith(jasmine.objectContaining({foo: "bar"});
+ *
+ * @param sample {Object} sample
+ * @returns matchable object for the sample
+ */
+jasmine.objectContaining = function (sample) {
+ return new jasmine.Matchers.ObjectContaining(sample);
+};
+
+/**
+ * Jasmine Spies are test doubles that can act as stubs, spies, fakes or when used in an expecation, mocks.
+ *
+ * Spies should be created in test setup, before expectations. They can then be checked, using the standard Jasmine
+ * expectation syntax. Spies can be checked if they were called or not and what the calling params were.
+ *
+ * A Spy has the following fields: wasCalled, callCount, mostRecentCall, and argsForCall (see docs).
+ *
+ * Spies are torn down at the end of every spec.
+ *
+ * Note: Do not call new jasmine.Spy() directly - a spy must be created using spyOn, jasmine.createSpy or jasmine.createSpyObj.
+ *
+ * @example
+ * // a stub
+ * var myStub = jasmine.createSpy('myStub'); // can be used anywhere
+ *
+ * // spy example
+ * var foo = {
+ * not: function(bool) { return !bool; }
+ * }
+ *
+ * // actual foo.not will not be called, execution stops
+ * spyOn(foo, 'not');
+
+ // foo.not spied upon, execution will continue to implementation
+ * spyOn(foo, 'not').andCallThrough();
+ *
+ * // fake example
+ * var foo = {
+ * not: function(bool) { return !bool; }
+ * }
+ *
+ * // foo.not(val) will return val
+ * spyOn(foo, 'not').andCallFake(function(value) {return value;});
+ *
+ * // mock example
+ * foo.not(7 == 7);
+ * expect(foo.not).toHaveBeenCalled();
+ * expect(foo.not).toHaveBeenCalledWith(true);
+ *
+ * @constructor
+ * @see spyOn, jasmine.createSpy, jasmine.createSpyObj
+ * @param {String} name
+ */
+jasmine.Spy = function(name) {
+ /**
+ * The name of the spy, if provided.
+ */
+ this.identity = name || 'unknown';
+ /**
+ * Is this Object a spy?
+ */
+ this.isSpy = true;
+ /**
+ * The actual function this spy stubs.
+ */
+ this.plan = function() {
+ };
+ /**
+ * Tracking of the most recent call to the spy.
+ * @example
+ * var mySpy = jasmine.createSpy('foo');
+ * mySpy(1, 2);
+ * mySpy.mostRecentCall.args = [1, 2];
+ */
+ this.mostRecentCall = {};
+
+ /**
+ * Holds arguments for each call to the spy, indexed by call count
+ * @example
+ * var mySpy = jasmine.createSpy('foo');
+ * mySpy(1, 2);
+ * mySpy(7, 8);
+ * mySpy.mostRecentCall.args = [7, 8];
+ * mySpy.argsForCall[0] = [1, 2];
+ * mySpy.argsForCall[1] = [7, 8];
+ */
+ this.argsForCall = [];
+ this.calls = [];
+};
+
+/**
+ * Tells a spy to call through to the actual implemenatation.
+ *
+ * @example
+ * var foo = {
+ * bar: function() { // do some stuff }
+ * }
+ *
+ * // defining a spy on an existing property: foo.bar
+ * spyOn(foo, 'bar').andCallThrough();
+ */
+jasmine.Spy.prototype.andCallThrough = function() {
+ this.plan = this.originalValue;
+ return this;
+};
+
+/**
+ * For setting the return value of a spy.
+ *
+ * @example
+ * // defining a spy from scratch: foo() returns 'baz'
+ * var foo = jasmine.createSpy('spy on foo').andReturn('baz');
+ *
+ * // defining a spy on an existing property: foo.bar() returns 'baz'
+ * spyOn(foo, 'bar').andReturn('baz');
+ *
+ * @param {Object} value
+ */
+jasmine.Spy.prototype.andReturn = function(value) {
+ this.plan = function() {
+ return value;
+ };
+ return this;
+};
+
+/**
+ * For throwing an exception when a spy is called.
+ *
+ * @example
+ * // defining a spy from scratch: foo() throws an exception w/ message 'ouch'
+ * var foo = jasmine.createSpy('spy on foo').andThrow('baz');
+ *
+ * // defining a spy on an existing property: foo.bar() throws an exception w/ message 'ouch'
+ * spyOn(foo, 'bar').andThrow('baz');
+ *
+ * @param {String} exceptionMsg
+ */
+jasmine.Spy.prototype.andThrow = function(exceptionMsg) {
+ this.plan = function() {
+ throw exceptionMsg;
+ };
+ return this;
+};
+
+/**
+ * Calls an alternate implementation when a spy is called.
+ *
+ * @example
+ * var baz = function() {
+ * // do some stuff, return something
+ * }
+ * // defining a spy from scratch: foo() calls the function baz
+ * var foo = jasmine.createSpy('spy on foo').andCall(baz);
+ *
+ * // defining a spy on an existing property: foo.bar() calls an anonymnous function
+ * spyOn(foo, 'bar').andCall(function() { return 'baz';} );
+ *
+ * @param {Function} fakeFunc
+ */
+jasmine.Spy.prototype.andCallFake = function(fakeFunc) {
+ this.plan = fakeFunc;
+ return this;
+};
+
+/**
+ * Resets all of a spy's the tracking variables so that it can be used again.
+ *
+ * @example
+ * spyOn(foo, 'bar');
+ *
+ * foo.bar();
+ *
+ * expect(foo.bar.callCount).toEqual(1);
+ *
+ * foo.bar.reset();
+ *
+ * expect(foo.bar.callCount).toEqual(0);
+ */
+jasmine.Spy.prototype.reset = function() {
+ this.wasCalled = false;
+ this.callCount = 0;
+ this.argsForCall = [];
+ this.calls = [];
+ this.mostRecentCall = {};
+};
+
+jasmine.createSpy = function(name) {
+
+ var spyObj = function() {
+ spyObj.wasCalled = true;
+ spyObj.callCount++;
+ var args = jasmine.util.argsToArray(arguments);
+ spyObj.mostRecentCall.object = this;
+ spyObj.mostRecentCall.args = args;
+ spyObj.argsForCall.push(args);
+ spyObj.calls.push({object: this, args: args});
+ return spyObj.plan.apply(this, arguments);
+ };
+
+ var spy = new jasmine.Spy(name);
+
+ for (var prop in spy) {
+ spyObj[prop] = spy[prop];
+ }
+
+ spyObj.reset();
+
+ return spyObj;
+};
+
+/**
+ * Determines whether an object is a spy.
+ *
+ * @param {jasmine.Spy|Object} putativeSpy
+ * @returns {Boolean}
+ */
+jasmine.isSpy = function(putativeSpy) {
+ return putativeSpy && putativeSpy.isSpy;
+};
+
+/**
+ * Creates a more complicated spy: an Object that has every property a function that is a spy. Used for stubbing something
+ * large in one call.
+ *
+ * @param {String} baseName name of spy class
+ * @param {Array} methodNames array of names of methods to make spies
+ */
+jasmine.createSpyObj = function(baseName, methodNames) {
+ if (!jasmine.isArray_(methodNames) || methodNames.length === 0) {
+ throw new Error('createSpyObj requires a non-empty array of method names to create spies for');
+ }
+ var obj = {};
+ for (var i = 0; i < methodNames.length; i++) {
+ obj[methodNames[i]] = jasmine.createSpy(baseName + '.' + methodNames[i]);
+ }
+ return obj;
+};
+
+/**
+ * All parameters are pretty-printed and concatenated together, then written to the current spec's output.
+ *
+ * Be careful not to leave calls to jasmine.log
in production code.
+ */
+jasmine.log = function() {
+ var spec = jasmine.getEnv().currentSpec;
+ spec.log.apply(spec, arguments);
+};
+
+/**
+ * Function that installs a spy on an existing object's method name. Used within a Spec to create a spy.
+ *
+ * @example
+ * // spy example
+ * var foo = {
+ * not: function(bool) { return !bool; }
+ * }
+ * spyOn(foo, 'not'); // actual foo.not will not be called, execution stops
+ *
+ * @see jasmine.createSpy
+ * @param obj
+ * @param methodName
+ * @return {jasmine.Spy} a Jasmine spy that can be chained with all spy methods
+ */
+var spyOn = function(obj, methodName) {
+ return jasmine.getEnv().currentSpec.spyOn(obj, methodName);
+};
+if (isCommonJS) exports.spyOn = spyOn;
+
+/**
+ * Creates a Jasmine spec that will be added to the current suite.
+ *
+ * // TODO: pending tests
+ *
+ * @example
+ * it('should be true', function() {
+ * expect(true).toEqual(true);
+ * });
+ *
+ * @param {String} desc description of this specification
+ * @param {Function} func defines the preconditions and expectations of the spec
+ */
+var it = function(desc, func) {
+ return jasmine.getEnv().it(desc, func);
+};
+if (isCommonJS) exports.it = it;
+
+/**
+ * Creates a disabled Jasmine spec.
+ *
+ * A convenience method that allows existing specs to be disabled temporarily during development.
+ *
+ * @param {String} desc description of this specification
+ * @param {Function} func defines the preconditions and expectations of the spec
+ */
+var xit = function(desc, func) {
+ return jasmine.getEnv().xit(desc, func);
+};
+if (isCommonJS) exports.xit = xit;
+
+/**
+ * Starts a chain for a Jasmine expectation.
+ *
+ * It is passed an Object that is the actual value and should chain to one of the many
+ * jasmine.Matchers functions.
+ *
+ * @param {Object} actual Actual value to test against and expected value
+ * @return {jasmine.Matchers}
+ */
+var expect = function(actual) {
+ return jasmine.getEnv().currentSpec.expect(actual);
+};
+if (isCommonJS) exports.expect = expect;
+
+/**
+ * Defines part of a jasmine spec. Used in cominbination with waits or waitsFor in asynchrnous specs.
+ *
+ * @param {Function} func Function that defines part of a jasmine spec.
+ */
+var runs = function(func) {
+ jasmine.getEnv().currentSpec.runs(func);
+};
+if (isCommonJS) exports.runs = runs;
+
+/**
+ * Waits a fixed time period before moving to the next block.
+ *
+ * @deprecated Use waitsFor() instead
+ * @param {Number} timeout milliseconds to wait
+ */
+var waits = function(timeout) {
+ jasmine.getEnv().currentSpec.waits(timeout);
+};
+if (isCommonJS) exports.waits = waits;
+
+/**
+ * Waits for the latchFunction to return true before proceeding to the next block.
+ *
+ * @param {Function} latchFunction
+ * @param {String} optional_timeoutMessage
+ * @param {Number} optional_timeout
+ */
+var waitsFor = function(latchFunction, optional_timeoutMessage, optional_timeout) {
+ jasmine.getEnv().currentSpec.waitsFor.apply(jasmine.getEnv().currentSpec, arguments);
+};
+if (isCommonJS) exports.waitsFor = waitsFor;
+
+/**
+ * A function that is called before each spec in a suite.
+ *
+ * Used for spec setup, including validating assumptions.
+ *
+ * @param {Function} beforeEachFunction
+ */
+var beforeEach = function(beforeEachFunction) {
+ jasmine.getEnv().beforeEach(beforeEachFunction);
+};
+if (isCommonJS) exports.beforeEach = beforeEach;
+
+/**
+ * A function that is called after each spec in a suite.
+ *
+ * Used for restoring any state that is hijacked during spec execution.
+ *
+ * @param {Function} afterEachFunction
+ */
+var afterEach = function(afterEachFunction) {
+ jasmine.getEnv().afterEach(afterEachFunction);
+};
+if (isCommonJS) exports.afterEach = afterEach;
+
+/**
+ * Defines a suite of specifications.
+ *
+ * Stores the description and all defined specs in the Jasmine environment as one suite of specs. Variables declared
+ * are accessible by calls to beforeEach, it, and afterEach. Describe blocks can be nested, allowing for specialization
+ * of setup in some tests.
+ *
+ * @example
+ * // TODO: a simple suite
+ *
+ * // TODO: a simple suite with a nested describe block
+ *
+ * @param {String} description A string, usually the class under test.
+ * @param {Function} specDefinitions function that defines several specs.
+ */
+var describe = function(description, specDefinitions) {
+ return jasmine.getEnv().describe(description, specDefinitions);
+};
+if (isCommonJS) exports.describe = describe;
+
+/**
+ * Disables a suite of specifications. Used to disable some suites in a file, or files, temporarily during development.
+ *
+ * @param {String} description A string, usually the class under test.
+ * @param {Function} specDefinitions function that defines several specs.
+ */
+var xdescribe = function(description, specDefinitions) {
+ return jasmine.getEnv().xdescribe(description, specDefinitions);
+};
+if (isCommonJS) exports.xdescribe = xdescribe;
+
+
+// Provide the XMLHttpRequest class for IE 5.x-6.x:
+jasmine.XmlHttpRequest = (typeof XMLHttpRequest == "undefined") ? function() {
+ function tryIt(f) {
+ try {
+ return f();
+ } catch(e) {
+ }
+ return null;
+ }
+
+ var xhr = tryIt(function() {
+ return new ActiveXObject("Msxml2.XMLHTTP.6.0");
+ }) ||
+ tryIt(function() {
+ return new ActiveXObject("Msxml2.XMLHTTP.3.0");
+ }) ||
+ tryIt(function() {
+ return new ActiveXObject("Msxml2.XMLHTTP");
+ }) ||
+ tryIt(function() {
+ return new ActiveXObject("Microsoft.XMLHTTP");
+ });
+
+ if (!xhr) throw new Error("This browser does not support XMLHttpRequest.");
+
+ return xhr;
+} : XMLHttpRequest;
+/**
+ * @namespace
+ */
+jasmine.util = {};
+
+/**
+ * Declare that a child class inherit it's prototype from the parent class.
+ *
+ * @private
+ * @param {Function} childClass
+ * @param {Function} parentClass
+ */
+jasmine.util.inherit = function(childClass, parentClass) {
+ /**
+ * @private
+ */
+ var subclass = function() {
+ };
+ subclass.prototype = parentClass.prototype;
+ childClass.prototype = new subclass();
+};
+
+jasmine.util.formatException = function(e) {
+ var lineNumber;
+ if (e.line) {
+ lineNumber = e.line;
+ }
+ else if (e.lineNumber) {
+ lineNumber = e.lineNumber;
+ }
+
+ var file;
+
+ if (e.sourceURL) {
+ file = e.sourceURL;
+ }
+ else if (e.fileName) {
+ file = e.fileName;
+ }
+
+ var message = (e.name && e.message) ? (e.name + ': ' + e.message) : e.toString();
+
+ if (file && lineNumber) {
+ message += ' in ' + file + ' (line ' + lineNumber + ')';
+ }
+
+ return message;
+};
+
+jasmine.util.htmlEscape = function(str) {
+ if (!str) return str;
+ return str.replace(/&/g, '&')
+ .replace(//g, '>');
+};
+
+jasmine.util.argsToArray = function(args) {
+ var arrayOfArgs = [];
+ for (var i = 0; i < args.length; i++) arrayOfArgs.push(args[i]);
+ return arrayOfArgs;
+};
+
+jasmine.util.extend = function(destination, source) {
+ for (var property in source) destination[property] = source[property];
+ return destination;
+};
+
+/**
+ * Environment for Jasmine
+ *
+ * @constructor
+ */
+jasmine.Env = function() {
+ this.currentSpec = null;
+ this.currentSuite = null;
+ this.currentRunner_ = new jasmine.Runner(this);
+
+ this.reporter = new jasmine.MultiReporter();
+
+ this.updateInterval = jasmine.DEFAULT_UPDATE_INTERVAL;
+ this.defaultTimeoutInterval = jasmine.DEFAULT_TIMEOUT_INTERVAL;
+ this.lastUpdate = 0;
+ this.specFilter = function() {
+ return true;
+ };
+
+ this.nextSpecId_ = 0;
+ this.nextSuiteId_ = 0;
+ this.equalityTesters_ = [];
+
+ // wrap matchers
+ this.matchersClass = function() {
+ jasmine.Matchers.apply(this, arguments);
+ };
+ jasmine.util.inherit(this.matchersClass, jasmine.Matchers);
+
+ jasmine.Matchers.wrapInto_(jasmine.Matchers.prototype, this.matchersClass);
+};
+
+
+jasmine.Env.prototype.setTimeout = jasmine.setTimeout;
+jasmine.Env.prototype.clearTimeout = jasmine.clearTimeout;
+jasmine.Env.prototype.setInterval = jasmine.setInterval;
+jasmine.Env.prototype.clearInterval = jasmine.clearInterval;
+
+/**
+ * @returns an object containing jasmine version build info, if set.
+ */
+jasmine.Env.prototype.version = function () {
+ if (jasmine.version_) {
+ return jasmine.version_;
+ } else {
+ throw new Error('Version not set');
+ }
+};
+
+/**
+ * @returns string containing jasmine version build info, if set.
+ */
+jasmine.Env.prototype.versionString = function() {
+ if (!jasmine.version_) {
+ return "version unknown";
+ }
+
+ var version = this.version();
+ var versionString = version.major + "." + version.minor + "." + version.build;
+ if (version.release_candidate) {
+ versionString += ".rc" + version.release_candidate;
+ }
+ versionString += " revision " + version.revision;
+ return versionString;
+};
+
+/**
+ * @returns a sequential integer starting at 0
+ */
+jasmine.Env.prototype.nextSpecId = function () {
+ return this.nextSpecId_++;
+};
+
+/**
+ * @returns a sequential integer starting at 0
+ */
+jasmine.Env.prototype.nextSuiteId = function () {
+ return this.nextSuiteId_++;
+};
+
+/**
+ * Register a reporter to receive status updates from Jasmine.
+ * @param {jasmine.Reporter} reporter An object which will receive status updates.
+ */
+jasmine.Env.prototype.addReporter = function(reporter) {
+ this.reporter.addReporter(reporter);
+};
+
+jasmine.Env.prototype.execute = function() {
+ this.currentRunner_.execute();
+};
+
+jasmine.Env.prototype.describe = function(description, specDefinitions) {
+ var suite = new jasmine.Suite(this, description, specDefinitions, this.currentSuite);
+
+ var parentSuite = this.currentSuite;
+ if (parentSuite) {
+ parentSuite.add(suite);
+ } else {
+ this.currentRunner_.add(suite);
+ }
+
+ this.currentSuite = suite;
+
+ var declarationError = null;
+ try {
+ specDefinitions.call(suite);
+ } catch(e) {
+ declarationError = e;
+ }
+
+ if (declarationError) {
+ this.it("encountered a declaration exception", function() {
+ throw declarationError;
+ });
+ }
+
+ this.currentSuite = parentSuite;
+
+ return suite;
+};
+
+jasmine.Env.prototype.beforeEach = function(beforeEachFunction) {
+ if (this.currentSuite) {
+ this.currentSuite.beforeEach(beforeEachFunction);
+ } else {
+ this.currentRunner_.beforeEach(beforeEachFunction);
+ }
+};
+
+jasmine.Env.prototype.currentRunner = function () {
+ return this.currentRunner_;
+};
+
+jasmine.Env.prototype.afterEach = function(afterEachFunction) {
+ if (this.currentSuite) {
+ this.currentSuite.afterEach(afterEachFunction);
+ } else {
+ this.currentRunner_.afterEach(afterEachFunction);
+ }
+
+};
+
+jasmine.Env.prototype.xdescribe = function(desc, specDefinitions) {
+ return {
+ execute: function() {
+ }
+ };
+};
+
+jasmine.Env.prototype.it = function(description, func) {
+ var spec = new jasmine.Spec(this, this.currentSuite, description);
+ this.currentSuite.add(spec);
+ this.currentSpec = spec;
+
+ if (func) {
+ spec.runs(func);
+ }
+
+ return spec;
+};
+
+jasmine.Env.prototype.xit = function(desc, func) {
+ return {
+ id: this.nextSpecId(),
+ runs: function() {
+ }
+ };
+};
+
+jasmine.Env.prototype.compareRegExps_ = function(a, b, mismatchKeys, mismatchValues) {
+ if (a.source != b.source)
+ mismatchValues.push("expected pattern /" + b.source + "/ is not equal to the pattern /" + a.source + "/");
+
+ if (a.ignoreCase != b.ignoreCase)
+ mismatchValues.push("expected modifier i was" + (b.ignoreCase ? " " : " not ") + "set and does not equal the origin modifier");
+
+ if (a.global != b.global)
+ mismatchValues.push("expected modifier g was" + (b.global ? " " : " not ") + "set and does not equal the origin modifier");
+
+ if (a.multiline != b.multiline)
+ mismatchValues.push("expected modifier m was" + (b.multiline ? " " : " not ") + "set and does not equal the origin modifier");
+
+ if (a.sticky != b.sticky)
+ mismatchValues.push("expected modifier y was" + (b.sticky ? " " : " not ") + "set and does not equal the origin modifier");
+
+ return (mismatchValues.length === 0);
+};
+
+jasmine.Env.prototype.compareObjects_ = function(a, b, mismatchKeys, mismatchValues) {
+ if (a.__Jasmine_been_here_before__ === b && b.__Jasmine_been_here_before__ === a) {
+ return true;
+ }
+
+ a.__Jasmine_been_here_before__ = b;
+ b.__Jasmine_been_here_before__ = a;
+
+ var hasKey = function(obj, keyName) {
+ return obj !== null && obj[keyName] !== jasmine.undefined;
+ };
+
+ for (var property in b) {
+ if (!hasKey(a, property) && hasKey(b, property)) {
+ mismatchKeys.push("expected has key '" + property + "', but missing from actual.");
+ }
+ }
+ for (property in a) {
+ if (!hasKey(b, property) && hasKey(a, property)) {
+ mismatchKeys.push("expected missing key '" + property + "', but present in actual.");
+ }
+ }
+ for (property in b) {
+ if (property == '__Jasmine_been_here_before__') continue;
+ if (!this.equals_(a[property], b[property], mismatchKeys, mismatchValues)) {
+ mismatchValues.push("'" + property + "' was '" + (b[property] ? jasmine.util.htmlEscape(b[property].toString()) : b[property]) + "' in expected, but was '" + (a[property] ? jasmine.util.htmlEscape(a[property].toString()) : a[property]) + "' in actual.");
+ }
+ }
+
+ if (jasmine.isArray_(a) && jasmine.isArray_(b) && a.length != b.length) {
+ mismatchValues.push("arrays were not the same length");
+ }
+
+ delete a.__Jasmine_been_here_before__;
+ delete b.__Jasmine_been_here_before__;
+ return (mismatchKeys.length === 0 && mismatchValues.length === 0);
+};
+
+jasmine.Env.prototype.equals_ = function(a, b, mismatchKeys, mismatchValues) {
+ mismatchKeys = mismatchKeys || [];
+ mismatchValues = mismatchValues || [];
+
+ for (var i = 0; i < this.equalityTesters_.length; i++) {
+ var equalityTester = this.equalityTesters_[i];
+ var result = equalityTester(a, b, this, mismatchKeys, mismatchValues);
+ if (result !== jasmine.undefined) return result;
+ }
+
+ if (a === b) return true;
+
+ if (a === jasmine.undefined || a === null || b === jasmine.undefined || b === null) {
+ return (a == jasmine.undefined && b == jasmine.undefined);
+ }
+
+ if (jasmine.isDomNode(a) && jasmine.isDomNode(b)) {
+ return a === b;
+ }
+
+ if (a instanceof Date && b instanceof Date) {
+ return a.getTime() == b.getTime();
+ }
+
+ if (a.jasmineMatches) {
+ return a.jasmineMatches(b);
+ }
+
+ if (b.jasmineMatches) {
+ return b.jasmineMatches(a);
+ }
+
+ if (a instanceof jasmine.Matchers.ObjectContaining) {
+ return a.matches(b);
+ }
+
+ if (b instanceof jasmine.Matchers.ObjectContaining) {
+ return b.matches(a);
+ }
+
+ if (jasmine.isString_(a) && jasmine.isString_(b)) {
+ return (a == b);
+ }
+
+ if (jasmine.isNumber_(a) && jasmine.isNumber_(b)) {
+ return (a == b);
+ }
+
+ if (a instanceof RegExp && b instanceof RegExp) {
+ return this.compareRegExps_(a, b, mismatchKeys, mismatchValues);
+ }
+
+ if (typeof a === "object" && typeof b === "object") {
+ return this.compareObjects_(a, b, mismatchKeys, mismatchValues);
+ }
+
+ //Straight check
+ return (a === b);
+};
+
+jasmine.Env.prototype.contains_ = function(haystack, needle) {
+ if (jasmine.isArray_(haystack)) {
+ for (var i = 0; i < haystack.length; i++) {
+ if (this.equals_(haystack[i], needle)) return true;
+ }
+ return false;
+ }
+ return haystack.indexOf(needle) >= 0;
+};
+
+jasmine.Env.prototype.addEqualityTester = function(equalityTester) {
+ this.equalityTesters_.push(equalityTester);
+};
+/** No-op base class for Jasmine reporters.
+ *
+ * @constructor
+ */
+jasmine.Reporter = function() {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.Reporter.prototype.reportRunnerStarting = function(runner) {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.Reporter.prototype.reportRunnerResults = function(runner) {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.Reporter.prototype.reportSuiteResults = function(suite) {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.Reporter.prototype.reportSpecStarting = function(spec) {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.Reporter.prototype.reportSpecResults = function(spec) {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.Reporter.prototype.log = function(str) {
+};
+
+/**
+ * Blocks are functions with executable code that make up a spec.
+ *
+ * @constructor
+ * @param {jasmine.Env} env
+ * @param {Function} func
+ * @param {jasmine.Spec} spec
+ */
+jasmine.Block = function(env, func, spec) {
+ this.env = env;
+ this.func = func;
+ this.spec = spec;
+};
+
+jasmine.Block.prototype.execute = function(onComplete) {
+ if (!jasmine.CATCH_EXCEPTIONS) {
+ this.func.apply(this.spec);
+ }
+ else {
+ try {
+ this.func.apply(this.spec);
+ } catch (e) {
+ this.spec.fail(e);
+ }
+ }
+ onComplete();
+};
+/** JavaScript API reporter.
+ *
+ * @constructor
+ */
+jasmine.JsApiReporter = function() {
+ this.started = false;
+ this.finished = false;
+ this.suites_ = [];
+ this.results_ = {};
+};
+
+jasmine.JsApiReporter.prototype.reportRunnerStarting = function(runner) {
+ this.started = true;
+ var suites = runner.topLevelSuites();
+ for (var i = 0; i < suites.length; i++) {
+ var suite = suites[i];
+ this.suites_.push(this.summarize_(suite));
+ }
+};
+
+jasmine.JsApiReporter.prototype.suites = function() {
+ return this.suites_;
+};
+
+jasmine.JsApiReporter.prototype.summarize_ = function(suiteOrSpec) {
+ var isSuite = suiteOrSpec instanceof jasmine.Suite;
+ var summary = {
+ id: suiteOrSpec.id,
+ name: suiteOrSpec.description,
+ type: isSuite ? 'suite' : 'spec',
+ children: []
+ };
+
+ if (isSuite) {
+ var children = suiteOrSpec.children();
+ for (var i = 0; i < children.length; i++) {
+ summary.children.push(this.summarize_(children[i]));
+ }
+ }
+ return summary;
+};
+
+jasmine.JsApiReporter.prototype.results = function() {
+ return this.results_;
+};
+
+jasmine.JsApiReporter.prototype.resultsForSpec = function(specId) {
+ return this.results_[specId];
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.JsApiReporter.prototype.reportRunnerResults = function(runner) {
+ this.finished = true;
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.JsApiReporter.prototype.reportSuiteResults = function(suite) {
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.JsApiReporter.prototype.reportSpecResults = function(spec) {
+ this.results_[spec.id] = {
+ messages: spec.results().getItems(),
+ result: spec.results().failedCount > 0 ? "failed" : "passed"
+ };
+};
+
+//noinspection JSUnusedLocalSymbols
+jasmine.JsApiReporter.prototype.log = function(str) {
+};
+
+jasmine.JsApiReporter.prototype.resultsForSpecs = function(specIds){
+ var results = {};
+ for (var i = 0; i < specIds.length; i++) {
+ var specId = specIds[i];
+ results[specId] = this.summarizeResult_(this.results_[specId]);
+ }
+ return results;
+};
+
+jasmine.JsApiReporter.prototype.summarizeResult_ = function(result){
+ var summaryMessages = [];
+ var messagesLength = result.messages.length;
+ for (var messageIndex = 0; messageIndex < messagesLength; messageIndex++) {
+ var resultMessage = result.messages[messageIndex];
+ summaryMessages.push({
+ text: resultMessage.type == 'log' ? resultMessage.toString() : jasmine.undefined,
+ passed: resultMessage.passed ? resultMessage.passed() : true,
+ type: resultMessage.type,
+ message: resultMessage.message,
+ trace: {
+ stack: resultMessage.passed && !resultMessage.passed() ? resultMessage.trace.stack : jasmine.undefined
+ }
+ });
+ }
+
+ return {
+ result : result.result,
+ messages : summaryMessages
+ };
+};
+
+/**
+ * @constructor
+ * @param {jasmine.Env} env
+ * @param actual
+ * @param {jasmine.Spec} spec
+ */
+jasmine.Matchers = function(env, actual, spec, opt_isNot) {
+ this.env = env;
+ this.actual = actual;
+ this.spec = spec;
+ this.isNot = opt_isNot || false;
+ this.reportWasCalled_ = false;
+};
+
+// todo: @deprecated as of Jasmine 0.11, remove soon [xw]
+jasmine.Matchers.pp = function(str) {
+ throw new Error("jasmine.Matchers.pp() is no longer supported, please use jasmine.pp() instead!");
+};
+
+// todo: @deprecated Deprecated as of Jasmine 0.10. Rewrite your custom matchers to return true or false. [xw]
+jasmine.Matchers.prototype.report = function(result, failing_message, details) {
+ throw new Error("As of jasmine 0.11, custom matchers must be implemented differently -- please see jasmine docs");
+};
+
+jasmine.Matchers.wrapInto_ = function(prototype, matchersClass) {
+ for (var methodName in prototype) {
+ if (methodName == 'report') continue;
+ var orig = prototype[methodName];
+ matchersClass.prototype[methodName] = jasmine.Matchers.matcherFn_(methodName, orig);
+ }
+};
+
+jasmine.Matchers.matcherFn_ = function(matcherName, matcherFunction) {
+ return function() {
+ var matcherArgs = jasmine.util.argsToArray(arguments);
+ var result = matcherFunction.apply(this, arguments);
+
+ if (this.isNot) {
+ result = !result;
+ }
+
+ if (this.reportWasCalled_) return result;
+
+ var message;
+ if (!result) {
+ if (this.message) {
+ message = this.message.apply(this, arguments);
+ if (jasmine.isArray_(message)) {
+ message = message[this.isNot ? 1 : 0];
+ }
+ } else {
+ var englishyPredicate = matcherName.replace(/[A-Z]/g, function(s) { return ' ' + s.toLowerCase(); });
+ message = "Expected " + jasmine.pp(this.actual) + (this.isNot ? " not " : " ") + englishyPredicate;
+ if (matcherArgs.length > 0) {
+ for (var i = 0; i < matcherArgs.length; i++) {
+ if (i > 0) message += ",";
+ message += " " + jasmine.pp(matcherArgs[i]);
+ }
+ }
+ message += ".";
+ }
+ }
+ var expectationResult = new jasmine.ExpectationResult({
+ matcherName: matcherName,
+ passed: result,
+ expected: matcherArgs.length > 1 ? matcherArgs : matcherArgs[0],
+ actual: this.actual,
+ message: message
+ });
+ this.spec.addMatcherResult(expectationResult);
+ return jasmine.undefined;
+ };
+};
+
+
+
+
+/**
+ * toBe: compares the actual to the expected using ===
+ * @param expected
+ */
+jasmine.Matchers.prototype.toBe = function(expected) {
+ return this.actual === expected;
+};
+
+/**
+ * toNotBe: compares the actual to the expected using !==
+ * @param expected
+ * @deprecated as of 1.0. Use not.toBe() instead.
+ */
+jasmine.Matchers.prototype.toNotBe = function(expected) {
+ return this.actual !== expected;
+};
+
+/**
+ * toEqual: compares the actual to the expected using common sense equality. Handles Objects, Arrays, etc.
+ *
+ * @param expected
+ */
+jasmine.Matchers.prototype.toEqual = function(expected) {
+ return this.env.equals_(this.actual, expected);
+};
+
+/**
+ * toNotEqual: compares the actual to the expected using the ! of jasmine.Matchers.toEqual
+ * @param expected
+ * @deprecated as of 1.0. Use not.toEqual() instead.
+ */
+jasmine.Matchers.prototype.toNotEqual = function(expected) {
+ return !this.env.equals_(this.actual, expected);
+};
+
+/**
+ * Matcher that compares the actual to the expected using a regular expression. Constructs a RegExp, so takes
+ * a pattern or a String.
+ *
+ * @param expected
+ */
+jasmine.Matchers.prototype.toMatch = function(expected) {
+ return new RegExp(expected).test(this.actual);
+};
+
+/**
+ * Matcher that compares the actual to the expected using the boolean inverse of jasmine.Matchers.toMatch
+ * @param expected
+ * @deprecated as of 1.0. Use not.toMatch() instead.
+ */
+jasmine.Matchers.prototype.toNotMatch = function(expected) {
+ return !(new RegExp(expected).test(this.actual));
+};
+
+/**
+ * Matcher that compares the actual to jasmine.undefined.
+ */
+jasmine.Matchers.prototype.toBeDefined = function() {
+ return (this.actual !== jasmine.undefined);
+};
+
+/**
+ * Matcher that compares the actual to jasmine.undefined.
+ */
+jasmine.Matchers.prototype.toBeUndefined = function() {
+ return (this.actual === jasmine.undefined);
+};
+
+/**
+ * Matcher that compares the actual to null.
+ */
+jasmine.Matchers.prototype.toBeNull = function() {
+ return (this.actual === null);
+};
+
+/**
+ * Matcher that compares the actual to NaN.
+ */
+jasmine.Matchers.prototype.toBeNaN = function() {
+ this.message = function() {
+ return [ "Expected " + jasmine.pp(this.actual) + " to be NaN." ];
+ };
+
+ return (this.actual !== this.actual);
+};
+
+/**
+ * Matcher that boolean not-nots the actual.
+ */
+jasmine.Matchers.prototype.toBeTruthy = function() {
+ return !!this.actual;
+};
+
+
+/**
+ * Matcher that boolean nots the actual.
+ */
+jasmine.Matchers.prototype.toBeFalsy = function() {
+ return !this.actual;
+};
+
+
+/**
+ * Matcher that checks to see if the actual, a Jasmine spy, was called.
+ */
+jasmine.Matchers.prototype.toHaveBeenCalled = function() {
+ if (arguments.length > 0) {
+ throw new Error('toHaveBeenCalled does not take arguments, use toHaveBeenCalledWith');
+ }
+
+ if (!jasmine.isSpy(this.actual)) {
+ throw new Error('Expected a spy, but got ' + jasmine.pp(this.actual) + '.');
+ }
+
+ this.message = function() {
+ return [
+ "Expected spy " + this.actual.identity + " to have been called.",
+ "Expected spy " + this.actual.identity + " not to have been called."
+ ];
+ };
+
+ return this.actual.wasCalled;
+};
+
+/** @deprecated Use expect(xxx).toHaveBeenCalled() instead */
+jasmine.Matchers.prototype.wasCalled = jasmine.Matchers.prototype.toHaveBeenCalled;
+
+/**
+ * Matcher that checks to see if the actual, a Jasmine spy, was not called.
+ *
+ * @deprecated Use expect(xxx).not.toHaveBeenCalled() instead
+ */
+jasmine.Matchers.prototype.wasNotCalled = function() {
+ if (arguments.length > 0) {
+ throw new Error('wasNotCalled does not take arguments');
+ }
+
+ if (!jasmine.isSpy(this.actual)) {
+ throw new Error('Expected a spy, but got ' + jasmine.pp(this.actual) + '.');
+ }
+
+ this.message = function() {
+ return [
+ "Expected spy " + this.actual.identity + " to not have been called.",
+ "Expected spy " + this.actual.identity + " to have been called."
+ ];
+ };
+
+ return !this.actual.wasCalled;
+};
+
+/**
+ * Matcher that checks to see if the actual, a Jasmine spy, was called with a set of parameters.
+ *
+ * @example
+ *
+ */
+jasmine.Matchers.prototype.toHaveBeenCalledWith = function() {
+ var expectedArgs = jasmine.util.argsToArray(arguments);
+ if (!jasmine.isSpy(this.actual)) {
+ throw new Error('Expected a spy, but got ' + jasmine.pp(this.actual) + '.');
+ }
+ this.message = function() {
+ var invertedMessage = "Expected spy " + this.actual.identity + " not to have been called with " + jasmine.pp(expectedArgs) + " but it was.";
+ var positiveMessage = "";
+ if (this.actual.callCount === 0) {
+ positiveMessage = "Expected spy " + this.actual.identity + " to have been called with " + jasmine.pp(expectedArgs) + " but it was never called.";
+ } else {
+ positiveMessage = "Expected spy " + this.actual.identity + " to have been called with " + jasmine.pp(expectedArgs) + " but actual calls were " + jasmine.pp(this.actual.argsForCall).replace(/^\[ | \]$/g, '')
+ }
+ return [positiveMessage, invertedMessage];
+ };
+
+ return this.env.contains_(this.actual.argsForCall, expectedArgs);
+};
+
+/** @deprecated Use expect(xxx).toHaveBeenCalledWith() instead */
+jasmine.Matchers.prototype.wasCalledWith = jasmine.Matchers.prototype.toHaveBeenCalledWith;
+
+/** @deprecated Use expect(xxx).not.toHaveBeenCalledWith() instead */
+jasmine.Matchers.prototype.wasNotCalledWith = function() {
+ var expectedArgs = jasmine.util.argsToArray(arguments);
+ if (!jasmine.isSpy(this.actual)) {
+ throw new Error('Expected a spy, but got ' + jasmine.pp(this.actual) + '.');
+ }
+
+ this.message = function() {
+ return [
+ "Expected spy not to have been called with " + jasmine.pp(expectedArgs) + " but it was",
+ "Expected spy to have been called with " + jasmine.pp(expectedArgs) + " but it was"
+ ];
+ };
+
+ return !this.env.contains_(this.actual.argsForCall, expectedArgs);
+};
+
+/**
+ * Matcher that checks that the expected item is an element in the actual Array.
+ *
+ * @param {Object} expected
+ */
+jasmine.Matchers.prototype.toContain = function(expected) {
+ return this.env.contains_(this.actual, expected);
+};
+
+/**
+ * Matcher that checks that the expected item is NOT an element in the actual Array.
+ *
+ * @param {Object} expected
+ * @deprecated as of 1.0. Use not.toContain() instead.
+ */
+jasmine.Matchers.prototype.toNotContain = function(expected) {
+ return !this.env.contains_(this.actual, expected);
+};
+
+jasmine.Matchers.prototype.toBeLessThan = function(expected) {
+ return this.actual < expected;
+};
+
+jasmine.Matchers.prototype.toBeGreaterThan = function(expected) {
+ return this.actual > expected;
+};
+
+/**
+ * Matcher that checks that the expected item is equal to the actual item
+ * up to a given level of decimal precision (default 2).
+ *
+ * @param {Number} expected
+ * @param {Number} precision, as number of decimal places
+ */
+jasmine.Matchers.prototype.toBeCloseTo = function(expected, precision) {
+ if (!(precision === 0)) {
+ precision = precision || 2;
+ }
+ return Math.abs(expected - this.actual) < (Math.pow(10, -precision) / 2);
+};
+
+/**
+ * Matcher that checks that the expected exception was thrown by the actual.
+ *
+ * @param {String} [expected]
+ */
+jasmine.Matchers.prototype.toThrow = function(expected) {
+ var result = false;
+ var exception;
+ if (typeof this.actual != 'function') {
+ throw new Error('Actual is not a function');
+ }
+ try {
+ this.actual();
+ } catch (e) {
+ exception = e;
+ }
+ if (exception) {
+ result = (expected === jasmine.undefined || this.env.equals_(exception.message || exception, expected.message || expected));
+ }
+
+ var not = this.isNot ? "not " : "";
+
+ this.message = function() {
+ if (exception && (expected === jasmine.undefined || !this.env.equals_(exception.message || exception, expected.message || expected))) {
+ return ["Expected function " + not + "to throw", expected ? expected.message || expected : "an exception", ", but it threw", exception.message || exception].join(' ');
+ } else {
+ return "Expected function to throw an exception.";
+ }
+ };
+
+ return result;
+};
+
+jasmine.Matchers.Any = function(expectedClass) {
+ this.expectedClass = expectedClass;
+};
+
+jasmine.Matchers.Any.prototype.jasmineMatches = function(other) {
+ if (this.expectedClass == String) {
+ return typeof other == 'string' || other instanceof String;
+ }
+
+ if (this.expectedClass == Number) {
+ return typeof other == 'number' || other instanceof Number;
+ }
+
+ if (this.expectedClass == Function) {
+ return typeof other == 'function' || other instanceof Function;
+ }
+
+ if (this.expectedClass == Object) {
+ return typeof other == 'object';
+ }
+
+ return other instanceof this.expectedClass;
+};
+
+jasmine.Matchers.Any.prototype.jasmineToString = function() {
+ return '';
+};
+
+jasmine.Matchers.ObjectContaining = function (sample) {
+ this.sample = sample;
+};
+
+jasmine.Matchers.ObjectContaining.prototype.jasmineMatches = function(other, mismatchKeys, mismatchValues) {
+ mismatchKeys = mismatchKeys || [];
+ mismatchValues = mismatchValues || [];
+
+ var env = jasmine.getEnv();
+
+ var hasKey = function(obj, keyName) {
+ return obj != null && obj[keyName] !== jasmine.undefined;
+ };
+
+ for (var property in this.sample) {
+ if (!hasKey(other, property) && hasKey(this.sample, property)) {
+ mismatchKeys.push("expected has key '" + property + "', but missing from actual.");
+ }
+ else if (!env.equals_(this.sample[property], other[property], mismatchKeys, mismatchValues)) {
+ mismatchValues.push("'" + property + "' was '" + (other[property] ? jasmine.util.htmlEscape(other[property].toString()) : other[property]) + "' in expected, but was '" + (this.sample[property] ? jasmine.util.htmlEscape(this.sample[property].toString()) : this.sample[property]) + "' in actual.");
+ }
+ }
+
+ return (mismatchKeys.length === 0 && mismatchValues.length === 0);
+};
+
+jasmine.Matchers.ObjectContaining.prototype.jasmineToString = function () {
+ return "";
+};
+// Mock setTimeout, clearTimeout
+// Contributed by Pivotal Computer Systems, www.pivotalsf.com
+
+jasmine.FakeTimer = function() {
+ this.reset();
+
+ var self = this;
+ self.setTimeout = function(funcToCall, millis) {
+ self.timeoutsMade++;
+ self.scheduleFunction(self.timeoutsMade, funcToCall, millis, false);
+ return self.timeoutsMade;
+ };
+
+ self.setInterval = function(funcToCall, millis) {
+ self.timeoutsMade++;
+ self.scheduleFunction(self.timeoutsMade, funcToCall, millis, true);
+ return self.timeoutsMade;
+ };
+
+ self.clearTimeout = function(timeoutKey) {
+ self.scheduledFunctions[timeoutKey] = jasmine.undefined;
+ };
+
+ self.clearInterval = function(timeoutKey) {
+ self.scheduledFunctions[timeoutKey] = jasmine.undefined;
+ };
+
+};
+
+jasmine.FakeTimer.prototype.reset = function() {
+ this.timeoutsMade = 0;
+ this.scheduledFunctions = {};
+ this.nowMillis = 0;
+};
+
+jasmine.FakeTimer.prototype.tick = function(millis) {
+ var oldMillis = this.nowMillis;
+ var newMillis = oldMillis + millis;
+ this.runFunctionsWithinRange(oldMillis, newMillis);
+ this.nowMillis = newMillis;
+};
+
+jasmine.FakeTimer.prototype.runFunctionsWithinRange = function(oldMillis, nowMillis) {
+ var scheduledFunc;
+ var funcsToRun = [];
+ for (var timeoutKey in this.scheduledFunctions) {
+ scheduledFunc = this.scheduledFunctions[timeoutKey];
+ if (scheduledFunc != jasmine.undefined &&
+ scheduledFunc.runAtMillis >= oldMillis &&
+ scheduledFunc.runAtMillis <= nowMillis) {
+ funcsToRun.push(scheduledFunc);
+ this.scheduledFunctions[timeoutKey] = jasmine.undefined;
+ }
+ }
+
+ if (funcsToRun.length > 0) {
+ funcsToRun.sort(function(a, b) {
+ return a.runAtMillis - b.runAtMillis;
+ });
+ for (var i = 0; i < funcsToRun.length; ++i) {
+ try {
+ var funcToRun = funcsToRun[i];
+ this.nowMillis = funcToRun.runAtMillis;
+ funcToRun.funcToCall();
+ if (funcToRun.recurring) {
+ this.scheduleFunction(funcToRun.timeoutKey,
+ funcToRun.funcToCall,
+ funcToRun.millis,
+ true);
+ }
+ } catch(e) {
+ }
+ }
+ this.runFunctionsWithinRange(oldMillis, nowMillis);
+ }
+};
+
+jasmine.FakeTimer.prototype.scheduleFunction = function(timeoutKey, funcToCall, millis, recurring) {
+ this.scheduledFunctions[timeoutKey] = {
+ runAtMillis: this.nowMillis + millis,
+ funcToCall: funcToCall,
+ recurring: recurring,
+ timeoutKey: timeoutKey,
+ millis: millis
+ };
+};
+
+/**
+ * @namespace
+ */
+jasmine.Clock = {
+ defaultFakeTimer: new jasmine.FakeTimer(),
+
+ reset: function() {
+ jasmine.Clock.assertInstalled();
+ jasmine.Clock.defaultFakeTimer.reset();
+ },
+
+ tick: function(millis) {
+ jasmine.Clock.assertInstalled();
+ jasmine.Clock.defaultFakeTimer.tick(millis);
+ },
+
+ runFunctionsWithinRange: function(oldMillis, nowMillis) {
+ jasmine.Clock.defaultFakeTimer.runFunctionsWithinRange(oldMillis, nowMillis);
+ },
+
+ scheduleFunction: function(timeoutKey, funcToCall, millis, recurring) {
+ jasmine.Clock.defaultFakeTimer.scheduleFunction(timeoutKey, funcToCall, millis, recurring);
+ },
+
+ useMock: function() {
+ if (!jasmine.Clock.isInstalled()) {
+ var spec = jasmine.getEnv().currentSpec;
+ spec.after(jasmine.Clock.uninstallMock);
+
+ jasmine.Clock.installMock();
+ }
+ },
+
+ installMock: function() {
+ jasmine.Clock.installed = jasmine.Clock.defaultFakeTimer;
+ },
+
+ uninstallMock: function() {
+ jasmine.Clock.assertInstalled();
+ jasmine.Clock.installed = jasmine.Clock.real;
+ },
+
+ real: {
+ setTimeout: jasmine.getGlobal().setTimeout,
+ clearTimeout: jasmine.getGlobal().clearTimeout,
+ setInterval: jasmine.getGlobal().setInterval,
+ clearInterval: jasmine.getGlobal().clearInterval
+ },
+
+ assertInstalled: function() {
+ if (!jasmine.Clock.isInstalled()) {
+ throw new Error("Mock clock is not installed, use jasmine.Clock.useMock()");
+ }
+ },
+
+ isInstalled: function() {
+ return jasmine.Clock.installed == jasmine.Clock.defaultFakeTimer;
+ },
+
+ installed: null
+};
+jasmine.Clock.installed = jasmine.Clock.real;
+
+//else for IE support
+jasmine.getGlobal().setTimeout = function(funcToCall, millis) {
+ if (jasmine.Clock.installed.setTimeout.apply) {
+ return jasmine.Clock.installed.setTimeout.apply(this, arguments);
+ } else {
+ return jasmine.Clock.installed.setTimeout(funcToCall, millis);
+ }
+};
+
+jasmine.getGlobal().setInterval = function(funcToCall, millis) {
+ if (jasmine.Clock.installed.setInterval.apply) {
+ return jasmine.Clock.installed.setInterval.apply(this, arguments);
+ } else {
+ return jasmine.Clock.installed.setInterval(funcToCall, millis);
+ }
+};
+
+jasmine.getGlobal().clearTimeout = function(timeoutKey) {
+ if (jasmine.Clock.installed.clearTimeout.apply) {
+ return jasmine.Clock.installed.clearTimeout.apply(this, arguments);
+ } else {
+ return jasmine.Clock.installed.clearTimeout(timeoutKey);
+ }
+};
+
+jasmine.getGlobal().clearInterval = function(timeoutKey) {
+ if (jasmine.Clock.installed.clearTimeout.apply) {
+ return jasmine.Clock.installed.clearInterval.apply(this, arguments);
+ } else {
+ return jasmine.Clock.installed.clearInterval(timeoutKey);
+ }
+};
+
+/**
+ * @constructor
+ */
+jasmine.MultiReporter = function() {
+ this.subReporters_ = [];
+};
+jasmine.util.inherit(jasmine.MultiReporter, jasmine.Reporter);
+
+jasmine.MultiReporter.prototype.addReporter = function(reporter) {
+ this.subReporters_.push(reporter);
+};
+
+(function() {
+ var functionNames = [
+ "reportRunnerStarting",
+ "reportRunnerResults",
+ "reportSuiteResults",
+ "reportSpecStarting",
+ "reportSpecResults",
+ "log"
+ ];
+ for (var i = 0; i < functionNames.length; i++) {
+ var functionName = functionNames[i];
+ jasmine.MultiReporter.prototype[functionName] = (function(functionName) {
+ return function() {
+ for (var j = 0; j < this.subReporters_.length; j++) {
+ var subReporter = this.subReporters_[j];
+ if (subReporter[functionName]) {
+ subReporter[functionName].apply(subReporter, arguments);
+ }
+ }
+ };
+ })(functionName);
+ }
+})();
+/**
+ * Holds results for a set of Jasmine spec. Allows for the results array to hold another jasmine.NestedResults
+ *
+ * @constructor
+ */
+jasmine.NestedResults = function() {
+ /**
+ * The total count of results
+ */
+ this.totalCount = 0;
+ /**
+ * Number of passed results
+ */
+ this.passedCount = 0;
+ /**
+ * Number of failed results
+ */
+ this.failedCount = 0;
+ /**
+ * Was this suite/spec skipped?
+ */
+ this.skipped = false;
+ /**
+ * @ignore
+ */
+ this.items_ = [];
+};
+
+/**
+ * Roll up the result counts.
+ *
+ * @param result
+ */
+jasmine.NestedResults.prototype.rollupCounts = function(result) {
+ this.totalCount += result.totalCount;
+ this.passedCount += result.passedCount;
+ this.failedCount += result.failedCount;
+};
+
+/**
+ * Adds a log message.
+ * @param values Array of message parts which will be concatenated later.
+ */
+jasmine.NestedResults.prototype.log = function(values) {
+ this.items_.push(new jasmine.MessageResult(values));
+};
+
+/**
+ * Getter for the results: message & results.
+ */
+jasmine.NestedResults.prototype.getItems = function() {
+ return this.items_;
+};
+
+/**
+ * Adds a result, tracking counts (total, passed, & failed)
+ * @param {jasmine.ExpectationResult|jasmine.NestedResults} result
+ */
+jasmine.NestedResults.prototype.addResult = function(result) {
+ if (result.type != 'log') {
+ if (result.items_) {
+ this.rollupCounts(result);
+ } else {
+ this.totalCount++;
+ if (result.passed()) {
+ this.passedCount++;
+ } else {
+ this.failedCount++;
+ }
+ }
+ }
+ this.items_.push(result);
+};
+
+/**
+ * @returns {Boolean} True if everything below passed
+ */
+jasmine.NestedResults.prototype.passed = function() {
+ return this.passedCount === this.totalCount;
+};
+/**
+ * Base class for pretty printing for expectation results.
+ */
+jasmine.PrettyPrinter = function() {
+ this.ppNestLevel_ = 0;
+};
+
+/**
+ * Formats a value in a nice, human-readable string.
+ *
+ * @param value
+ */
+jasmine.PrettyPrinter.prototype.format = function(value) {
+ this.ppNestLevel_++;
+ try {
+ if (value === jasmine.undefined) {
+ this.emitScalar('undefined');
+ } else if (value === null) {
+ this.emitScalar('null');
+ } else if (value === jasmine.getGlobal()) {
+ this.emitScalar('');
+ } else if (value.jasmineToString) {
+ this.emitScalar(value.jasmineToString());
+ } else if (typeof value === 'string') {
+ this.emitString(value);
+ } else if (jasmine.isSpy(value)) {
+ this.emitScalar("spy on " + value.identity);
+ } else if (value instanceof RegExp) {
+ this.emitScalar(value.toString());
+ } else if (typeof value === 'function') {
+ this.emitScalar('Function');
+ } else if (typeof value.nodeType === 'number') {
+ this.emitScalar('HTMLNode');
+ } else if (value instanceof Date) {
+ this.emitScalar('Date(' + value + ')');
+ } else if (value.__Jasmine_been_here_before__) {
+ this.emitScalar('');
+ } else if (jasmine.isArray_(value) || typeof value == 'object') {
+ value.__Jasmine_been_here_before__ = true;
+ if (jasmine.isArray_(value)) {
+ this.emitArray(value);
+ } else {
+ this.emitObject(value);
+ }
+ delete value.__Jasmine_been_here_before__;
+ } else {
+ this.emitScalar(value.toString());
+ }
+ } finally {
+ this.ppNestLevel_--;
+ }
+};
+
+jasmine.PrettyPrinter.prototype.iterateObject = function(obj, fn) {
+ for (var property in obj) {
+ if (!obj.hasOwnProperty(property)) continue;
+ if (property == '__Jasmine_been_here_before__') continue;
+ fn(property, obj.__lookupGetter__ ? (obj.__lookupGetter__(property) !== jasmine.undefined &&
+ obj.__lookupGetter__(property) !== null) : false);
+ }
+};
+
+jasmine.PrettyPrinter.prototype.emitArray = jasmine.unimplementedMethod_;
+jasmine.PrettyPrinter.prototype.emitObject = jasmine.unimplementedMethod_;
+jasmine.PrettyPrinter.prototype.emitScalar = jasmine.unimplementedMethod_;
+jasmine.PrettyPrinter.prototype.emitString = jasmine.unimplementedMethod_;
+
+jasmine.StringPrettyPrinter = function() {
+ jasmine.PrettyPrinter.call(this);
+
+ this.string = '';
+};
+jasmine.util.inherit(jasmine.StringPrettyPrinter, jasmine.PrettyPrinter);
+
+jasmine.StringPrettyPrinter.prototype.emitScalar = function(value) {
+ this.append(value);
+};
+
+jasmine.StringPrettyPrinter.prototype.emitString = function(value) {
+ this.append("'" + value + "'");
+};
+
+jasmine.StringPrettyPrinter.prototype.emitArray = function(array) {
+ if (this.ppNestLevel_ > jasmine.MAX_PRETTY_PRINT_DEPTH) {
+ this.append("Array");
+ return;
+ }
+
+ this.append('[ ');
+ for (var i = 0; i < array.length; i++) {
+ if (i > 0) {
+ this.append(', ');
+ }
+ this.format(array[i]);
+ }
+ this.append(' ]');
+};
+
+jasmine.StringPrettyPrinter.prototype.emitObject = function(obj) {
+ if (this.ppNestLevel_ > jasmine.MAX_PRETTY_PRINT_DEPTH) {
+ this.append("Object");
+ return;
+ }
+
+ var self = this;
+ this.append('{ ');
+ var first = true;
+
+ this.iterateObject(obj, function(property, isGetter) {
+ if (first) {
+ first = false;
+ } else {
+ self.append(', ');
+ }
+
+ self.append(property);
+ self.append(' : ');
+ if (isGetter) {
+ self.append('');
+ } else {
+ self.format(obj[property]);
+ }
+ });
+
+ this.append(' }');
+};
+
+jasmine.StringPrettyPrinter.prototype.append = function(value) {
+ this.string += value;
+};
+jasmine.Queue = function(env) {
+ this.env = env;
+
+ // parallel to blocks. each true value in this array means the block will
+ // get executed even if we abort
+ this.ensured = [];
+ this.blocks = [];
+ this.running = false;
+ this.index = 0;
+ this.offset = 0;
+ this.abort = false;
+};
+
+jasmine.Queue.prototype.addBefore = function(block, ensure) {
+ if (ensure === jasmine.undefined) {
+ ensure = false;
+ }
+
+ this.blocks.unshift(block);
+ this.ensured.unshift(ensure);
+};
+
+jasmine.Queue.prototype.add = function(block, ensure) {
+ if (ensure === jasmine.undefined) {
+ ensure = false;
+ }
+
+ this.blocks.push(block);
+ this.ensured.push(ensure);
+};
+
+jasmine.Queue.prototype.insertNext = function(block, ensure) {
+ if (ensure === jasmine.undefined) {
+ ensure = false;
+ }
+
+ this.ensured.splice((this.index + this.offset + 1), 0, ensure);
+ this.blocks.splice((this.index + this.offset + 1), 0, block);
+ this.offset++;
+};
+
+jasmine.Queue.prototype.start = function(onComplete) {
+ this.running = true;
+ this.onComplete = onComplete;
+ this.next_();
+};
+
+jasmine.Queue.prototype.isRunning = function() {
+ return this.running;
+};
+
+jasmine.Queue.LOOP_DONT_RECURSE = true;
+
+jasmine.Queue.prototype.next_ = function() {
+ var self = this;
+ var goAgain = true;
+
+ while (goAgain) {
+ goAgain = false;
+
+ if (self.index < self.blocks.length && !(this.abort && !this.ensured[self.index])) {
+ var calledSynchronously = true;
+ var completedSynchronously = false;
+
+ var onComplete = function () {
+ if (jasmine.Queue.LOOP_DONT_RECURSE && calledSynchronously) {
+ completedSynchronously = true;
+ return;
+ }
+
+ if (self.blocks[self.index].abort) {
+ self.abort = true;
+ }
+
+ self.offset = 0;
+ self.index++;
+
+ var now = new Date().getTime();
+ if (self.env.updateInterval && now - self.env.lastUpdate > self.env.updateInterval) {
+ self.env.lastUpdate = now;
+ self.env.setTimeout(function() {
+ self.next_();
+ }, 0);
+ } else {
+ if (jasmine.Queue.LOOP_DONT_RECURSE && completedSynchronously) {
+ goAgain = true;
+ } else {
+ self.next_();
+ }
+ }
+ };
+ self.blocks[self.index].execute(onComplete);
+
+ calledSynchronously = false;
+ if (completedSynchronously) {
+ onComplete();
+ }
+
+ } else {
+ self.running = false;
+ if (self.onComplete) {
+ self.onComplete();
+ }
+ }
+ }
+};
+
+jasmine.Queue.prototype.results = function() {
+ var results = new jasmine.NestedResults();
+ for (var i = 0; i < this.blocks.length; i++) {
+ if (this.blocks[i].results) {
+ results.addResult(this.blocks[i].results());
+ }
+ }
+ return results;
+};
+
+
+/**
+ * Runner
+ *
+ * @constructor
+ * @param {jasmine.Env} env
+ */
+jasmine.Runner = function(env) {
+ var self = this;
+ self.env = env;
+ self.queue = new jasmine.Queue(env);
+ self.before_ = [];
+ self.after_ = [];
+ self.suites_ = [];
+};
+
+jasmine.Runner.prototype.execute = function() {
+ var self = this;
+ if (self.env.reporter.reportRunnerStarting) {
+ self.env.reporter.reportRunnerStarting(this);
+ }
+ self.queue.start(function () {
+ self.finishCallback();
+ });
+};
+
+jasmine.Runner.prototype.beforeEach = function(beforeEachFunction) {
+ beforeEachFunction.typeName = 'beforeEach';
+ this.before_.splice(0,0,beforeEachFunction);
+};
+
+jasmine.Runner.prototype.afterEach = function(afterEachFunction) {
+ afterEachFunction.typeName = 'afterEach';
+ this.after_.splice(0,0,afterEachFunction);
+};
+
+
+jasmine.Runner.prototype.finishCallback = function() {
+ this.env.reporter.reportRunnerResults(this);
+};
+
+jasmine.Runner.prototype.addSuite = function(suite) {
+ this.suites_.push(suite);
+};
+
+jasmine.Runner.prototype.add = function(block) {
+ if (block instanceof jasmine.Suite) {
+ this.addSuite(block);
+ }
+ this.queue.add(block);
+};
+
+jasmine.Runner.prototype.specs = function () {
+ var suites = this.suites();
+ var specs = [];
+ for (var i = 0; i < suites.length; i++) {
+ specs = specs.concat(suites[i].specs());
+ }
+ return specs;
+};
+
+jasmine.Runner.prototype.suites = function() {
+ return this.suites_;
+};
+
+jasmine.Runner.prototype.topLevelSuites = function() {
+ var topLevelSuites = [];
+ for (var i = 0; i < this.suites_.length; i++) {
+ if (!this.suites_[i].parentSuite) {
+ topLevelSuites.push(this.suites_[i]);
+ }
+ }
+ return topLevelSuites;
+};
+
+jasmine.Runner.prototype.results = function() {
+ return this.queue.results();
+};
+/**
+ * Internal representation of a Jasmine specification, or test.
+ *
+ * @constructor
+ * @param {jasmine.Env} env
+ * @param {jasmine.Suite} suite
+ * @param {String} description
+ */
+jasmine.Spec = function(env, suite, description) {
+ if (!env) {
+ throw new Error('jasmine.Env() required');
+ }
+ if (!suite) {
+ throw new Error('jasmine.Suite() required');
+ }
+ var spec = this;
+ spec.id = env.nextSpecId ? env.nextSpecId() : null;
+ spec.env = env;
+ spec.suite = suite;
+ spec.description = description;
+ spec.queue = new jasmine.Queue(env);
+
+ spec.afterCallbacks = [];
+ spec.spies_ = [];
+
+ spec.results_ = new jasmine.NestedResults();
+ spec.results_.description = description;
+ spec.matchersClass = null;
+};
+
+jasmine.Spec.prototype.getFullName = function() {
+ return this.suite.getFullName() + ' ' + this.description + '.';
+};
+
+
+jasmine.Spec.prototype.results = function() {
+ return this.results_;
+};
+
+/**
+ * All parameters are pretty-printed and concatenated together, then written to the spec's output.
+ *
+ * Be careful not to leave calls to jasmine.log
in production code.
+ */
+jasmine.Spec.prototype.log = function() {
+ return this.results_.log(arguments);
+};
+
+jasmine.Spec.prototype.runs = function (func) {
+ var block = new jasmine.Block(this.env, func, this);
+ this.addToQueue(block);
+ return this;
+};
+
+jasmine.Spec.prototype.addToQueue = function (block) {
+ if (this.queue.isRunning()) {
+ this.queue.insertNext(block);
+ } else {
+ this.queue.add(block);
+ }
+};
+
+/**
+ * @param {jasmine.ExpectationResult} result
+ */
+jasmine.Spec.prototype.addMatcherResult = function(result) {
+ this.results_.addResult(result);
+};
+
+jasmine.Spec.prototype.expect = function(actual) {
+ var positive = new (this.getMatchersClass_())(this.env, actual, this);
+ positive.not = new (this.getMatchersClass_())(this.env, actual, this, true);
+ return positive;
+};
+
+/**
+ * Waits a fixed time period before moving to the next block.
+ *
+ * @deprecated Use waitsFor() instead
+ * @param {Number} timeout milliseconds to wait
+ */
+jasmine.Spec.prototype.waits = function(timeout) {
+ var waitsFunc = new jasmine.WaitsBlock(this.env, timeout, this);
+ this.addToQueue(waitsFunc);
+ return this;
+};
+
+/**
+ * Waits for the latchFunction to return true before proceeding to the next block.
+ *
+ * @param {Function} latchFunction
+ * @param {String} optional_timeoutMessage
+ * @param {Number} optional_timeout
+ */
+jasmine.Spec.prototype.waitsFor = function(latchFunction, optional_timeoutMessage, optional_timeout) {
+ var latchFunction_ = null;
+ var optional_timeoutMessage_ = null;
+ var optional_timeout_ = null;
+
+ for (var i = 0; i < arguments.length; i++) {
+ var arg = arguments[i];
+ switch (typeof arg) {
+ case 'function':
+ latchFunction_ = arg;
+ break;
+ case 'string':
+ optional_timeoutMessage_ = arg;
+ break;
+ case 'number':
+ optional_timeout_ = arg;
+ break;
+ }
+ }
+
+ var waitsForFunc = new jasmine.WaitsForBlock(this.env, optional_timeout_, latchFunction_, optional_timeoutMessage_, this);
+ this.addToQueue(waitsForFunc);
+ return this;
+};
+
+jasmine.Spec.prototype.fail = function (e) {
+ var expectationResult = new jasmine.ExpectationResult({
+ passed: false,
+ message: e ? jasmine.util.formatException(e) : 'Exception',
+ trace: { stack: e.stack }
+ });
+ this.results_.addResult(expectationResult);
+};
+
+jasmine.Spec.prototype.getMatchersClass_ = function() {
+ return this.matchersClass || this.env.matchersClass;
+};
+
+jasmine.Spec.prototype.addMatchers = function(matchersPrototype) {
+ var parent = this.getMatchersClass_();
+ var newMatchersClass = function() {
+ parent.apply(this, arguments);
+ };
+ jasmine.util.inherit(newMatchersClass, parent);
+ jasmine.Matchers.wrapInto_(matchersPrototype, newMatchersClass);
+ this.matchersClass = newMatchersClass;
+};
+
+jasmine.Spec.prototype.finishCallback = function() {
+ this.env.reporter.reportSpecResults(this);
+};
+
+jasmine.Spec.prototype.finish = function(onComplete) {
+ this.removeAllSpies();
+ this.finishCallback();
+ if (onComplete) {
+ onComplete();
+ }
+};
+
+jasmine.Spec.prototype.after = function(doAfter) {
+ if (this.queue.isRunning()) {
+ this.queue.add(new jasmine.Block(this.env, doAfter, this), true);
+ } else {
+ this.afterCallbacks.unshift(doAfter);
+ }
+};
+
+jasmine.Spec.prototype.execute = function(onComplete) {
+ var spec = this;
+ if (!spec.env.specFilter(spec)) {
+ spec.results_.skipped = true;
+ spec.finish(onComplete);
+ return;
+ }
+
+ this.env.reporter.reportSpecStarting(this);
+
+ spec.env.currentSpec = spec;
+
+ spec.addBeforesAndAftersToQueue();
+
+ spec.queue.start(function () {
+ spec.finish(onComplete);
+ });
+};
+
+jasmine.Spec.prototype.addBeforesAndAftersToQueue = function() {
+ var runner = this.env.currentRunner();
+ var i;
+
+ for (var suite = this.suite; suite; suite = suite.parentSuite) {
+ for (i = 0; i < suite.before_.length; i++) {
+ this.queue.addBefore(new jasmine.Block(this.env, suite.before_[i], this));
+ }
+ }
+ for (i = 0; i < runner.before_.length; i++) {
+ this.queue.addBefore(new jasmine.Block(this.env, runner.before_[i], this));
+ }
+ for (i = 0; i < this.afterCallbacks.length; i++) {
+ this.queue.add(new jasmine.Block(this.env, this.afterCallbacks[i], this), true);
+ }
+ for (suite = this.suite; suite; suite = suite.parentSuite) {
+ for (i = 0; i < suite.after_.length; i++) {
+ this.queue.add(new jasmine.Block(this.env, suite.after_[i], this), true);
+ }
+ }
+ for (i = 0; i < runner.after_.length; i++) {
+ this.queue.add(new jasmine.Block(this.env, runner.after_[i], this), true);
+ }
+};
+
+jasmine.Spec.prototype.explodes = function() {
+ throw 'explodes function should not have been called';
+};
+
+jasmine.Spec.prototype.spyOn = function(obj, methodName, ignoreMethodDoesntExist) {
+ if (obj == jasmine.undefined) {
+ throw "spyOn could not find an object to spy upon for " + methodName + "()";
+ }
+
+ if (!ignoreMethodDoesntExist && obj[methodName] === jasmine.undefined) {
+ throw methodName + '() method does not exist';
+ }
+
+ if (!ignoreMethodDoesntExist && obj[methodName] && obj[methodName].isSpy) {
+ throw new Error(methodName + ' has already been spied upon');
+ }
+
+ var spyObj = jasmine.createSpy(methodName);
+
+ this.spies_.push(spyObj);
+ spyObj.baseObj = obj;
+ spyObj.methodName = methodName;
+ spyObj.originalValue = obj[methodName];
+
+ obj[methodName] = spyObj;
+
+ return spyObj;
+};
+
+jasmine.Spec.prototype.removeAllSpies = function() {
+ for (var i = 0; i < this.spies_.length; i++) {
+ var spy = this.spies_[i];
+ spy.baseObj[spy.methodName] = spy.originalValue;
+ }
+ this.spies_ = [];
+};
+
+/**
+ * Internal representation of a Jasmine suite.
+ *
+ * @constructor
+ * @param {jasmine.Env} env
+ * @param {String} description
+ * @param {Function} specDefinitions
+ * @param {jasmine.Suite} parentSuite
+ */
+jasmine.Suite = function(env, description, specDefinitions, parentSuite) {
+ var self = this;
+ self.id = env.nextSuiteId ? env.nextSuiteId() : null;
+ self.description = description;
+ self.queue = new jasmine.Queue(env);
+ self.parentSuite = parentSuite;
+ self.env = env;
+ self.before_ = [];
+ self.after_ = [];
+ self.children_ = [];
+ self.suites_ = [];
+ self.specs_ = [];
+};
+
+jasmine.Suite.prototype.getFullName = function() {
+ var fullName = this.description;
+ for (var parentSuite = this.parentSuite; parentSuite; parentSuite = parentSuite.parentSuite) {
+ fullName = parentSuite.description + ' ' + fullName;
+ }
+ return fullName;
+};
+
+jasmine.Suite.prototype.finish = function(onComplete) {
+ this.env.reporter.reportSuiteResults(this);
+ this.finished = true;
+ if (typeof(onComplete) == 'function') {
+ onComplete();
+ }
+};
+
+jasmine.Suite.prototype.beforeEach = function(beforeEachFunction) {
+ beforeEachFunction.typeName = 'beforeEach';
+ this.before_.unshift(beforeEachFunction);
+};
+
+jasmine.Suite.prototype.afterEach = function(afterEachFunction) {
+ afterEachFunction.typeName = 'afterEach';
+ this.after_.unshift(afterEachFunction);
+};
+
+jasmine.Suite.prototype.results = function() {
+ return this.queue.results();
+};
+
+jasmine.Suite.prototype.add = function(suiteOrSpec) {
+ this.children_.push(suiteOrSpec);
+ if (suiteOrSpec instanceof jasmine.Suite) {
+ this.suites_.push(suiteOrSpec);
+ this.env.currentRunner().addSuite(suiteOrSpec);
+ } else {
+ this.specs_.push(suiteOrSpec);
+ }
+ this.queue.add(suiteOrSpec);
+};
+
+jasmine.Suite.prototype.specs = function() {
+ return this.specs_;
+};
+
+jasmine.Suite.prototype.suites = function() {
+ return this.suites_;
+};
+
+jasmine.Suite.prototype.children = function() {
+ return this.children_;
+};
+
+jasmine.Suite.prototype.execute = function(onComplete) {
+ var self = this;
+ this.queue.start(function () {
+ self.finish(onComplete);
+ });
+};
+jasmine.WaitsBlock = function(env, timeout, spec) {
+ this.timeout = timeout;
+ jasmine.Block.call(this, env, null, spec);
+};
+
+jasmine.util.inherit(jasmine.WaitsBlock, jasmine.Block);
+
+jasmine.WaitsBlock.prototype.execute = function (onComplete) {
+ if (jasmine.VERBOSE) {
+ this.env.reporter.log('>> Jasmine waiting for ' + this.timeout + ' ms...');
+ }
+ this.env.setTimeout(function () {
+ onComplete();
+ }, this.timeout);
+};
+/**
+ * A block which waits for some condition to become true, with timeout.
+ *
+ * @constructor
+ * @extends jasmine.Block
+ * @param {jasmine.Env} env The Jasmine environment.
+ * @param {Number} timeout The maximum time in milliseconds to wait for the condition to become true.
+ * @param {Function} latchFunction A function which returns true when the desired condition has been met.
+ * @param {String} message The message to display if the desired condition hasn't been met within the given time period.
+ * @param {jasmine.Spec} spec The Jasmine spec.
+ */
+jasmine.WaitsForBlock = function(env, timeout, latchFunction, message, spec) {
+ this.timeout = timeout || env.defaultTimeoutInterval;
+ this.latchFunction = latchFunction;
+ this.message = message;
+ this.totalTimeSpentWaitingForLatch = 0;
+ jasmine.Block.call(this, env, null, spec);
+};
+jasmine.util.inherit(jasmine.WaitsForBlock, jasmine.Block);
+
+jasmine.WaitsForBlock.TIMEOUT_INCREMENT = 10;
+
+jasmine.WaitsForBlock.prototype.execute = function(onComplete) {
+ if (jasmine.VERBOSE) {
+ this.env.reporter.log('>> Jasmine waiting for ' + (this.message || 'something to happen'));
+ }
+ var latchFunctionResult;
+ try {
+ latchFunctionResult = this.latchFunction.apply(this.spec);
+ } catch (e) {
+ this.spec.fail(e);
+ onComplete();
+ return;
+ }
+
+ if (latchFunctionResult) {
+ onComplete();
+ } else if (this.totalTimeSpentWaitingForLatch >= this.timeout) {
+ var message = 'timed out after ' + this.timeout + ' msec waiting for ' + (this.message || 'something to happen');
+ this.spec.fail({
+ name: 'timeout',
+ message: message
+ });
+
+ this.abort = true;
+ onComplete();
+ } else {
+ this.totalTimeSpentWaitingForLatch += jasmine.WaitsForBlock.TIMEOUT_INCREMENT;
+ var self = this;
+ this.env.setTimeout(function() {
+ self.execute(onComplete);
+ }, jasmine.WaitsForBlock.TIMEOUT_INCREMENT);
+ }
+};
+
+jasmine.version_= {
+ "major": 1,
+ "minor": 3,
+ "build": 1,
+ "revision": 1354556913
+};