[Pkg-javascript-commits] [libjs-jsxc] 01/16: Imported Upstream version 1.0.0

Praveen Arimbrathodiyil praveen at moszumanska.debian.org
Sat Jan 10 18:20:34 UTC 2015


This is an automated email from the git hooks/post-receive script.

praveen pushed a commit to branch master
in repository libjs-jsxc.

commit 9d64423680503d8d6e6eeefcc95cb28bd196be24
Author: Praveen Arimbrathodiyil <praveen at debian.org>
Date:   Sat Jan 10 22:33:10 2015 +0530

    Imported Upstream version 1.0.0
---
 LICENSE                                      |    21 +
 img/XMPP_logo.png                            |   Bin 0 -> 9572 bytes
 img/XMPP_logo.svg                            |   105 +
 img/camera_icon_grey.svg                     |    34 +
 img/camera_icon_white.svg                    |    72 +
 img/edit.png                                 |   Bin 0 -> 610 bytes
 img/emotions/angel.png                       |   Bin 0 -> 2077 bytes
 img/emotions/angry.png                       |   Bin 0 -> 1952 bytes
 img/emotions/beer.png                        |   Bin 0 -> 1694 bytes
 img/emotions/biggrin.png                     |   Bin 0 -> 1846 bytes
 img/emotions/crysad.png                      |   Bin 0 -> 1971 bytes
 img/emotions/dead.png                        |   Bin 0 -> 1896 bytes
 img/emotions/devil.png                       |   Bin 0 -> 1868 bytes
 img/emotions/dontknow.png                    |   Bin 0 -> 1881 bytes
 img/emotions/doubt.png                       |   Bin 0 -> 1907 bytes
 img/emotions/grin.png                        |   Bin 0 -> 1821 bytes
 img/emotions/kiss.png                        |   Bin 0 -> 1980 bytes
 img/emotions/kissing.png                     |   Bin 0 -> 2036 bytes
 img/emotions/lol.png                         |   Bin 0 -> 1836 bytes
 img/emotions/love.png                        |   Bin 0 -> 1832 bytes
 img/emotions/music.png                       |   Bin 0 -> 2404 bytes
 img/emotions/nerv.png                        |   Bin 0 -> 1897 bytes
 img/emotions/rose.png                        |   Bin 0 -> 1639 bytes
 img/emotions/sad.png                         |   Bin 0 -> 1910 bytes
 img/emotions/sick.png                        |   Bin 0 -> 1976 bytes
 img/emotions/smile.png                       |   Bin 0 -> 1919 bytes
 img/emotions/strange.png                     |   Bin 0 -> 1860 bytes
 img/emotions/sunglassess.png                 |   Bin 0 -> 1920 bytes
 img/emotions/surprised.png                   |   Bin 0 -> 1877 bytes
 img/emotions/thumbsdown.png                  |   Bin 0 -> 1525 bytes
 img/emotions/thumbsup.png                    |   Bin 0 -> 1650 bytes
 img/emotions/tired.png                       |   Bin 0 -> 2104 bytes
 img/emotions/tonguesmile.png                 |   Bin 0 -> 1934 bytes
 img/emotions/weird.png                       |   Bin 0 -> 1904 bytes
 img/emotions/wink.png                        |   Bin 0 -> 1919 bytes
 img/emotions/xp.png                          |   Bin 0 -> 1977 bytes
 img/emotions/yesyes.png                      |   Bin 0 -> 1869 bytes
 img/emotions/zip.png                         |   Bin 0 -> 1975 bytes
 img/gear_grey.svg                            |    87 +
 img/gear_white.svg                           |    86 +
 img/loading.gif                              |   Bin 0 -> 2767 bytes
 img/padlock_close_green.svg                  |    86 +
 img/padlock_close_grey.svg                   |    94 +
 img/padlock_close_orange.svg                 |    94 +
 img/padlock_open.svg                         |    39 +
 img/padlock_open_grey.svg                    |    66 +
 img/padlock_open_white.svg                   |    64 +
 img/smiley.png                               |   Bin 0 -> 560 bytes
 img/speech_balloon_white.svg                 |    66 +
 jsxc.css                                     |  1243 ++
 jsxc.js                                      |  6925 ++++++++++
 jsxc.lib.js                                  |  5871 +++++++++
 jsxc.lib.webrtc.js                           |  1051 ++
 jsxc.min.js                                  |    30 +
 jsxc.min.js.map                              |     1 +
 jsxc.webrtc.css                              |   379 +
 lib/dsa-ww.js                                |    50 +
 lib/jquery.colorbox-min.js                   |     7 +
 lib/jquery.fullscreen.js                     |    88 +
 lib/jquery.min.js                            |     4 +
 lib/jquery.slimscroll.js                     |   474 +
 lib/jquery.ui.min.js                         |     7 +
 lib/jsxc.dep.js                              | 16644 +++++++++++++++++++++++++
 lib/jsxc.dep.min.js                          |    67 +
 lib/jsxc.dep.min.js.map                      |     1 +
 lib/otr/build/dep/bigint.js                  |  1705 +++
 lib/otr/build/dep/crypto.js                  |  2434 ++++
 lib/otr/build/dep/eventemitter.js            |   455 +
 lib/otr/build/dep/salsa20.js                 |   254 +
 lib/otr/build/dsa-webworker.js               |    52 +
 lib/otr/build/otr.js                         |  2631 ++++
 lib/otr/build/otr.min.js                     |    13 +
 lib/otr/build/sm-webworker.js                |    60 +
 lib/otr/lib/const.js                         |    55 +
 lib/otr/lib/dsa-webworker.js                 |    52 +
 lib/otr/lib/dsa.js                           |   405 +
 lib/otr/lib/helpers.js                       |   349 +
 lib/otr/lib/sm-webworker.js                  |    60 +
 lib/otr/vendor/bigint.js                     |  1705 +++
 lib/otr/vendor/crypto.js                     |  2434 ++++
 lib/otr/vendor/eventemitter.js               |   455 +
 lib/otr/vendor/salsa20.js                    |   254 +
 lib/strophe.caps.js                          |   264 +
 lib/strophe.disco.js                         |   232 +
 lib/strophe.jingle/strophe.jingle.adapter.js |   386 +
 lib/strophe.jingle/strophe.jingle.js         |   266 +
 lib/strophe.jingle/strophe.jingle.sdp.js     |   822 ++
 lib/strophe.jingle/strophe.jingle.session.js |   914 ++
 lib/strophe.js                               |  5153 ++++++++
 lib/strophe.muc.js                           |  1020 ++
 lib/strophe.vcard.js                         |    66 +
 sound/Ping1.mp3                              |   Bin 0 -> 56841 bytes
 sound/Rotary-Phone6.mp3                      |   Bin 0 -> 114624 bytes
 sound/credential                             |    11 +
 sound/incomingMessage.wav                    |   Bin 0 -> 100452 bytes
 95 files changed, 56263 insertions(+)

diff --git a/LICENSE b/LICENSE
new file mode 100644
index 0000000..9ca7f2a
--- /dev/null
+++ b/LICENSE
@@ -0,0 +1,21 @@
+The MIT License (MIT)
+
+Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org>
+
+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.
\ No newline at end of file
diff --git a/img/XMPP_logo.png b/img/XMPP_logo.png
new file mode 100644
index 0000000..783da61
Binary files /dev/null and b/img/XMPP_logo.png differ
diff --git a/img/XMPP_logo.svg b/img/XMPP_logo.svg
new file mode 100644
index 0000000..87f683d
--- /dev/null
+++ b/img/XMPP_logo.svg
@@ -0,0 +1,105 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Generator: Adobe Illustrator 13.0.2, SVG Export Plug-In . SVG Version: 6.00 Build 14948)  -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   id="Layer_1"
+   xml:space="preserve"
+   height="147.22015"
+   viewBox="0 0 192.89149 147.21771"
+   width="192.89587"
+   version="1.1"
+   y="0px"
+   x="0px"
+   enable-background="new 0 0 176.486 181.437"
+   inkscape:version="0.91pre2 r"
+   sodipodi:docname="XMPP_logo.svg"><metadata
+     id="metadata41"><rdf:RDF><cc:Work
+         rdf:about=""><dc:format>image/svg+xml</dc:format><dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" /><dc:title></dc:title></cc:Work></rdf:RDF></metadata><defs
+     id="defs39" /><sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="640"
+     inkscape:window-height="480"
+     id="namedview37"
+     showgrid="false"
+     fit-margin-top="10"
+     fit-margin-left="10"
+     fit-margin-right="10"
+     fit-margin-bottom="10"
+     inkscape:zoom="1.3007054"
+     inkscape:cx="71.332001"
+     inkscape:cy="72.501142"
+     inkscape:window-x="75"
+     inkscape:window-y="34"
+     inkscape:window-maximized="0"
+     inkscape:current-layer="Layer_1" /><linearGradient
+     id="SVGID_1_"
+     y2="0.00048828"
+     gradientUnits="userSpaceOnUse"
+     x2="-1807.2"
+     gradientTransform="translate(1925.9953,9.9988343)"
+     y1="125.86"
+     x1="-1807.2"><stop
+       stop-color="#1b3967"
+       offset=".011"
+       id="stop4" /><stop
+       stop-color="#13b5ea"
+       offset=".467"
+       id="stop6" /><stop
+       stop-color="#002b5c"
+       offset=".9945"
+       id="stop8" /></linearGradient><path
+     d="m 146.28531,24.187835 c 0.077,1.313 -1.786,0.968 -1.786,2.293 0,38.551 -44.720002,96.831035 -89.847002,108.190035 l 0,1.182 C 114.60931,130.34287 181.38231,69.052835 182.89231,10.002834 l -36.6,14.189001 z"
+     id="path10"
+     style="fill:url(#SVGID_1_)"
+     inkscape:connector-curvature="0" /><path
+     d="m 130.22531,27.958835 c 0.077,1.313 0.121,2.633 0.121,3.958 0,38.551 -30.700002,90.497035 -75.827002,101.860035 l 0,1.637 c 59.065002,-3.823 105.810002,-63.023035 105.810002,-109.200035 0,-2.375 -0.125,-4.729 -0.371,-7.056 l -29.73,8.796 z"
+     id="path12"
+     inkscape:connector-curvature="0"
+     style="fill:#e96d1f" /><linearGradient
+     id="SVGID_2_"
+     y2="1.279e-13"
+     gradientUnits="userSpaceOnUse"
+     x2="-1073.2"
+     gradientTransform="matrix(-1,0,0,1,-998.20465,9.9988343)"
+     y1="126.85"
+     x1="-1073.2"><stop
+       stop-color="#1b3967"
+       offset=".011"
+       id="stop15" /><stop
+       stop-color="#13b5ea"
+       offset=".467"
+       id="stop17" /><stop
+       stop-color="#002b5c"
+       offset=".9945"
+       id="stop19" /></linearGradient><path
+     d="m 46.594308,24.187835 c -0.077,1.313 1.787,0.968 1.787,2.293 0,38.551 46.558,97.366035 91.688002,108.730035 l 0,1.639 C 80.116308,131.32987 11.509308,69.049835 9.9993079,9.9998343 L 46.598308,24.188835 Z"
+     id="path21"
+     style="fill:url(#SVGID_2_)"
+     inkscape:connector-curvature="0" /><path
+     d="m 64.726308,28.930835 c -0.076,1.313 -0.12,2.63 -0.12,3.957 0,38.551 30.699,90.497035 75.827002,101.860035 l 0,1.639 C 81.389308,133.59687 34.623308,73.362835 34.623308,27.186835 c 0,-2.375 0.128,-4.729 0.371,-7.056 l 29.73,8.798 z"
+     id="path23"
+     inkscape:connector-curvature="0"
+     style="fill:#a0ce67" /><path
+     d="m 34.708308,19.581835 7.617,2.722 c -0.041,0.962 -0.066,2.254 -0.066,3.225 0,41.219 37.271,98.204035 87.272002,107.120035 3.245,1.088 7.538,2.077 10.932,2.931 l 0,1.638 C 75.209308,131.65787 29.363308,65.351835 34.703308,19.577835 Z"
+     id="path25"
+     inkscape:connector-curvature="0"
+     style="fill:#439639" /><path
+     d="m 160.33531,18.758835 -7.833,2.625 c 0.041,0.963 0.191,2.203 0.191,3.173 0,41.219 -37.272,98.205035 -87.274002,107.120035 -3.243,1.089 -7.538,2.077 -10.93,2.932 l 0,1.639 C 122.83331,127.58787 165.66931,64.528835 160.32931,18.757835 Z"
+     id="path27"
+     inkscape:connector-curvature="0"
+     style="fill:#d9541e" /></svg>
\ No newline at end of file
diff --git a/img/camera_icon_grey.svg b/img/camera_icon_grey.svg
new file mode 100644
index 0000000..0228619
--- /dev/null
+++ b/img/camera_icon_grey.svg
@@ -0,0 +1,34 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   version="1.1"
+   id="svg2"
+   width="20"
+   height="20">
+  <metadata
+     id="metadata10">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs8" />
+  <g
+     id="layer1"
+     transform="matrix(0.03788594,0,0,0.03788594,-0.41340793,-2.6310716)">
+    <path
+       id="path3906"
+       style="fill:#808080;stroke:none"
+       d="m 361.65148,313.51768 177.16074,-131.78516 0,302.22999 L 358.97338,350.18517 Z M 10.911922,182.40634 l 378.372678,0 0,302.65582 -378.372678,0 z" />
+  </g>
+</svg>
diff --git a/img/camera_icon_white.svg b/img/camera_icon_white.svg
new file mode 100644
index 0000000..da48b07
--- /dev/null
+++ b/img/camera_icon_white.svg
@@ -0,0 +1,72 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="535.00458"
+   height="510.4368"
+   id="svg3896"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   sodipodi:docname="camera_icon_white.svg">
+  <defs
+     id="defs3898" />
+  <sodipodi:namedview
+     id="base"
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1.0"
+     inkscape:pageopacity="0.0"
+     inkscape:pageshadow="2"
+     inkscape:zoom="0.98994949"
+     inkscape:cx="353.91934"
+     inkscape:cy="233.8456"
+     inkscape:document-units="px"
+     inkscape:current-layer="layer1"
+     showgrid="false"
+     fit-margin-top="0"
+     fit-margin-left="0"
+     fit-margin-right="0"
+     fit-margin-bottom="0"
+     inkscape:window-width="1855"
+     inkscape:window-height="1056"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="1" />
+  <metadata
+     id="metadata3901">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <g
+     inkscape:label="Layer 1"
+     inkscape:groupmode="layer"
+     id="layer1"
+     transform="translate(-5.7142771,-72.362177)">
+    <rect
+       style="fill:#ffffff;stroke:none;stroke-width:1.6172694;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+       id="rect3904"
+       width="378.37268"
+       height="302.65582"
+       x="10.911922"
+       y="182.40634" />
+    <path
+       style="fill:#ffffff;stroke:none"
+       d="m 361.65148,313.51768 177.16074,-131.78516 0,302.22999 -179.83884,-133.77734 z"
+       id="path3906"
+       inkscape:connector-curvature="0" />
+  </g>
+</svg>
diff --git a/img/edit.png b/img/edit.png
new file mode 100644
index 0000000..442372d
Binary files /dev/null and b/img/edit.png differ
diff --git a/img/emotions/angel.png b/img/emotions/angel.png
new file mode 100644
index 0000000..2d1a750
Binary files /dev/null and b/img/emotions/angel.png differ
diff --git a/img/emotions/angry.png b/img/emotions/angry.png
new file mode 100644
index 0000000..6091f9d
Binary files /dev/null and b/img/emotions/angry.png differ
diff --git a/img/emotions/beer.png b/img/emotions/beer.png
new file mode 100644
index 0000000..4ca1dd7
Binary files /dev/null and b/img/emotions/beer.png differ
diff --git a/img/emotions/biggrin.png b/img/emotions/biggrin.png
new file mode 100644
index 0000000..44e3d67
Binary files /dev/null and b/img/emotions/biggrin.png differ
diff --git a/img/emotions/crysad.png b/img/emotions/crysad.png
new file mode 100644
index 0000000..74d4c2d
Binary files /dev/null and b/img/emotions/crysad.png differ
diff --git a/img/emotions/dead.png b/img/emotions/dead.png
new file mode 100644
index 0000000..7dc19f8
Binary files /dev/null and b/img/emotions/dead.png differ
diff --git a/img/emotions/devil.png b/img/emotions/devil.png
new file mode 100644
index 0000000..6fd9533
Binary files /dev/null and b/img/emotions/devil.png differ
diff --git a/img/emotions/dontknow.png b/img/emotions/dontknow.png
new file mode 100644
index 0000000..7f95f8f
Binary files /dev/null and b/img/emotions/dontknow.png differ
diff --git a/img/emotions/doubt.png b/img/emotions/doubt.png
new file mode 100644
index 0000000..46ca83c
Binary files /dev/null and b/img/emotions/doubt.png differ
diff --git a/img/emotions/grin.png b/img/emotions/grin.png
new file mode 100644
index 0000000..6206f66
Binary files /dev/null and b/img/emotions/grin.png differ
diff --git a/img/emotions/kiss.png b/img/emotions/kiss.png
new file mode 100644
index 0000000..ca292b1
Binary files /dev/null and b/img/emotions/kiss.png differ
diff --git a/img/emotions/kissing.png b/img/emotions/kissing.png
new file mode 100644
index 0000000..4cd401c
Binary files /dev/null and b/img/emotions/kissing.png differ
diff --git a/img/emotions/lol.png b/img/emotions/lol.png
new file mode 100644
index 0000000..21eff88
Binary files /dev/null and b/img/emotions/lol.png differ
diff --git a/img/emotions/love.png b/img/emotions/love.png
new file mode 100644
index 0000000..a0738ab
Binary files /dev/null and b/img/emotions/love.png differ
diff --git a/img/emotions/music.png b/img/emotions/music.png
new file mode 100644
index 0000000..9693d3d
Binary files /dev/null and b/img/emotions/music.png differ
diff --git a/img/emotions/nerv.png b/img/emotions/nerv.png
new file mode 100644
index 0000000..3f8c97e
Binary files /dev/null and b/img/emotions/nerv.png differ
diff --git a/img/emotions/rose.png b/img/emotions/rose.png
new file mode 100644
index 0000000..8f79602
Binary files /dev/null and b/img/emotions/rose.png differ
diff --git a/img/emotions/sad.png b/img/emotions/sad.png
new file mode 100644
index 0000000..96e1311
Binary files /dev/null and b/img/emotions/sad.png differ
diff --git a/img/emotions/sick.png b/img/emotions/sick.png
new file mode 100644
index 0000000..dc6bef2
Binary files /dev/null and b/img/emotions/sick.png differ
diff --git a/img/emotions/smile.png b/img/emotions/smile.png
new file mode 100644
index 0000000..c57c836
Binary files /dev/null and b/img/emotions/smile.png differ
diff --git a/img/emotions/strange.png b/img/emotions/strange.png
new file mode 100644
index 0000000..394a322
Binary files /dev/null and b/img/emotions/strange.png differ
diff --git a/img/emotions/sunglassess.png b/img/emotions/sunglassess.png
new file mode 100644
index 0000000..cc4df59
Binary files /dev/null and b/img/emotions/sunglassess.png differ
diff --git a/img/emotions/surprised.png b/img/emotions/surprised.png
new file mode 100644
index 0000000..28ab3b6
Binary files /dev/null and b/img/emotions/surprised.png differ
diff --git a/img/emotions/thumbsdown.png b/img/emotions/thumbsdown.png
new file mode 100644
index 0000000..d9809cb
Binary files /dev/null and b/img/emotions/thumbsdown.png differ
diff --git a/img/emotions/thumbsup.png b/img/emotions/thumbsup.png
new file mode 100644
index 0000000..0e3f1b4
Binary files /dev/null and b/img/emotions/thumbsup.png differ
diff --git a/img/emotions/tired.png b/img/emotions/tired.png
new file mode 100644
index 0000000..d229f4d
Binary files /dev/null and b/img/emotions/tired.png differ
diff --git a/img/emotions/tonguesmile.png b/img/emotions/tonguesmile.png
new file mode 100644
index 0000000..cfd4ea1
Binary files /dev/null and b/img/emotions/tonguesmile.png differ
diff --git a/img/emotions/weird.png b/img/emotions/weird.png
new file mode 100644
index 0000000..4775a9d
Binary files /dev/null and b/img/emotions/weird.png differ
diff --git a/img/emotions/wink.png b/img/emotions/wink.png
new file mode 100644
index 0000000..4f89797
Binary files /dev/null and b/img/emotions/wink.png differ
diff --git a/img/emotions/xp.png b/img/emotions/xp.png
new file mode 100644
index 0000000..e83ad50
Binary files /dev/null and b/img/emotions/xp.png differ
diff --git a/img/emotions/yesyes.png b/img/emotions/yesyes.png
new file mode 100644
index 0000000..dfb978e
Binary files /dev/null and b/img/emotions/yesyes.png differ
diff --git a/img/emotions/zip.png b/img/emotions/zip.png
new file mode 100644
index 0000000..0994932
Binary files /dev/null and b/img/emotions/zip.png differ
diff --git a/img/gear_grey.svg b/img/gear_grey.svg
new file mode 100644
index 0000000..871de5d
--- /dev/null
+++ b/img/gear_grey.svg
@@ -0,0 +1,87 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="272.70593"
+   height="275.19449"
+   id="svg2"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   sodipodi:docname="gear.svg">
+  <sodipodi:namedview
+     id="base"
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1.0"
+     inkscape:pageopacity="0.0"
+     inkscape:pageshadow="2"
+     inkscape:zoom="2.8"
+     inkscape:cx="146.88159"
+     inkscape:cy="131.06634"
+     inkscape:document-units="px"
+     inkscape:current-layer="layer1"
+     showgrid="false"
+     inkscape:window-width="1111"
+     inkscape:window-height="840"
+     inkscape:window-x="191"
+     inkscape:window-y="34"
+     inkscape:window-maximized="0"
+     fit-margin-top="0"
+     fit-margin-left="0"
+     fit-margin-right="0"
+     fit-margin-bottom="0" />
+  <defs
+     id="defs4">
+    <clipPath
+       clipPathUnits="userSpaceOnUse"
+       id="clipPath4617">
+      <path
+         sodipodi:type="arc"
+         style="fill:#b3b3b3;stroke:none"
+         id="path4619"
+         sodipodi:cx="141.42857"
+         sodipodi:cy="140.90877"
+         sodipodi:rx="124.28571"
+         sodipodi:ry="124.28571"
+         d="m 265.71429,140.90877 a 124.28571,124.28571 0 1 1 -248.57143,0 124.28571,124.28571 0 1 1 248.57143,0 z"
+         transform="translate(170.48656,275.14888)" />
+    </clipPath>
+  </defs>
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <g
+     inkscape:label="Layer 1"
+     inkscape:groupmode="layer"
+     id="layer1"
+     transform="translate(-175.56216,-278.4604)">
+    <path
+       style="fill:#b3b3b3;stroke:none"
+       d="m 322.90077,552.37609 c -0.8025,-1.03265 -4.1298,-6.77812 -7.394,-12.76771 -5.66062,-10.38686 -5.9349,-11.45971 -5.9349,-23.21429 0,-6.77827 -0.16071,-12.33471 -0.35714,-12.34764 -0.19643,-0.0129 -7.10714,-1.25351 -15.35714,-2.75686 -8.25,-1.50332 -15.42054,-2.29632 -15.93453,-1.7622 -0.51399,0.53413 -2.65068,5.852 -4.7482,11.8175 -3.29679,9.3763 -4.91271,11.81339 -11.92261,17.98135 -13.16776,11.58617 -12.78024,11.47418 -21.89865,6.32807 l -7.91867,-4.46902 2.59473,-13.70492 c  [...]
+       id="path4584"
+       inkscape:connector-curvature="0"
+       clip-path="url(#clipPath4617)" />
+    <path
+       style="fill:#999999;stroke:none"
+       d="m 137.40915,254.8325 c -1.26388,-3.15876 -2.30675,-9.74805 -2.31748,-14.64286 -0.03,-13.68249 -0.66793,-14.93253 -8.47507,-16.60686 -3.72822,-0.79957 -10.89569,-1.87094 -15.92771,-2.38086 l -9.14914,-0.92711 -4.027767,10.62916 c -3.904909,10.30494 -9.421742,18.24315 -12.678473,18.24315 -1.736606,0 -17.384447,-8.69884 -21.68079,-12.05264 -4.20131,-3.27961 -1.931667,-10.19323 6.980559,-21.26363 l 7.845084,-9.74484 -6.385824,-8.1123 c -3.512203,-4.46177 -8.245923,-10.30856 -10.519 [...]
+       id="path2985"
+       inkscape:connector-curvature="0"
+       transform="translate(175.56216,278.4604)" />
+  </g>
+</svg>
diff --git a/img/gear_white.svg b/img/gear_white.svg
new file mode 100644
index 0000000..abf677c
--- /dev/null
+++ b/img/gear_white.svg
@@ -0,0 +1,86 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="272.70593"
+   height="275.19449"
+   id="svg2"
+   version="1.1"
+   inkscape:version="0.48.3.1 r9886"
+   sodipodi:docname="gear.svg">
+  <sodipodi:namedview
+     id="base"
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1.0"
+     inkscape:pageopacity="0.0"
+     inkscape:pageshadow="2"
+     inkscape:zoom="0.7"
+     inkscape:cx="159.11118"
+     inkscape:cy="75.475936"
+     inkscape:document-units="px"
+     inkscape:current-layer="layer1"
+     showgrid="false"
+     inkscape:window-width="1111"
+     inkscape:window-height="840"
+     inkscape:window-x="191"
+     inkscape:window-y="28"
+     inkscape:window-maximized="0"
+     fit-margin-top="0"
+     fit-margin-left="0"
+     fit-margin-right="0"
+     fit-margin-bottom="0" />
+  <defs
+     id="defs4">
+    <clipPath
+       clipPathUnits="userSpaceOnUse"
+       id="clipPath4617">
+      <path
+         sodipodi:type="arc"
+         style="fill:#b3b3b3;stroke:none"
+         id="path4619"
+         sodipodi:cx="141.42857"
+         sodipodi:cy="140.90877"
+         sodipodi:rx="124.28571"
+         sodipodi:ry="124.28571"
+         d="m 265.71429,140.90877 a 124.28571,124.28571 0 1 1 -248.57143,0 124.28571,124.28571 0 1 1 248.57143,0 z"
+         transform="translate(170.48656,275.14888)" />
+    </clipPath>
+  </defs>
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <g
+     inkscape:label="Layer 1"
+     inkscape:groupmode="layer"
+     id="layer1"
+     transform="translate(-175.56216,-278.4604)">
+    <path
+       style="fill:#b3b3b3;stroke:none"
+       d="m 322.90077,552.37609 c -0.8025,-1.03265 -4.1298,-6.77812 -7.394,-12.76771 -5.66062,-10.38686 -5.9349,-11.45971 -5.9349,-23.21429 0,-6.77827 -0.16071,-12.33471 -0.35714,-12.34764 -0.19643,-0.0129 -7.10714,-1.25351 -15.35714,-2.75686 -8.25,-1.50332 -15.42054,-2.29632 -15.93453,-1.7622 -0.51399,0.53413 -2.65068,5.852 -4.7482,11.8175 -3.29679,9.3763 -4.91271,11.81339 -11.92261,17.98135 -13.16776,11.58617 -12.78024,11.47418 -21.89865,6.32807 l -7.91867,-4.46902 2.59473,-13.70492 c  [...]
+       id="path4584"
+       inkscape:connector-curvature="0"
+       clip-path="url(#clipPath4617)" />
+    <path
+       style="fill:#ffffff;stroke:none"
+       d="m 137.40915,254.8325 c -1.26388,-3.15876 -2.30675,-9.74805 -2.31748,-14.64286 -0.03,-13.68249 -0.66793,-14.93253 -8.47507,-16.60686 -3.72822,-0.79957 -10.89569,-1.87094 -15.92771,-2.38086 l -9.14914,-0.92711 -4.027767,10.62916 c -3.904909,10.30494 -9.421742,18.24315 -12.678473,18.24315 -1.736606,0 -17.384447,-8.69884 -21.68079,-12.05264 -4.20131,-3.27961 -1.931667,-10.19323 6.980559,-21.26363 l 7.845084,-9.74484 -6.385824,-8.1123 c -3.512203,-4.46177 -8.245923,-10.30856 -10.519 [...]
+       id="path2985"
+       inkscape:connector-curvature="0"
+       transform="translate(175.56216,278.4604)" />
+  </g>
+</svg>
diff --git a/img/loading.gif b/img/loading.gif
new file mode 100644
index 0000000..f864d5f
Binary files /dev/null and b/img/loading.gif differ
diff --git a/img/padlock_close_green.svg b/img/padlock_close_green.svg
new file mode 100644
index 0000000..8a4ac0a
--- /dev/null
+++ b/img/padlock_close_green.svg
@@ -0,0 +1,86 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   id="svg3041"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   width="15"
+   height="15"
+   sodipodi:docname="padlock_close_green.svg">
+  <metadata
+     id="metadata3047">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs3045" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1855"
+     inkscape:window-height="1056"
+     id="namedview3043"
+     showgrid="false"
+     inkscape:zoom="32"
+     inkscape:cx="12.823367"
+     inkscape:cy="8.9449238"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg3041"
+     borderlayer="true" />
+  <rect
+     style="fill:#44aa00;stroke:#44aa00;stroke-width:0.92522794"
+     id="rect3051"
+     width="10.232394"
+     height="6.4687548"
+     x="2.5534275"
+     y="8.074255" />
+  <path
+     style="fill:none;fill-opacity:1;stroke:#44aa00;stroke-width:2.08827519;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+     id="path4101"
+     sodipodi:type="arc"
+     sodipodi:cx="7.6713805"
+     sodipodi:cy="6.7928491"
+     sodipodi:rx="4.534133"
+     sodipodi:ry="4.5301714"
+     sodipodi:start="3.1415927"
+     sodipodi:end="0"
+     d="M 3.1372476,6.7928489 A 4.534133,4.5301714 0 0 1 7.6713806,2.2626777 4.534133,4.5301714 0 0 1 12.205513,6.7928491"
+     sodipodi:open="true" />
+  <rect
+     style="fill:#44aa00;fill-opacity:1;stroke:none;stroke-width:1.39526081;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+     id="rect4153"
+     width="2.0906768"
+     height="1.0984892"
+     x="2.0913804"
+     y="6.7554431" />
+  <rect
+     y="6.7632556"
+     x="11.157786"
+     height="1.0984892"
+     width="2.0906768"
+     id="rect4155"
+     style="fill:#44aa00;fill-opacity:1;stroke:none;stroke-width:1.39526081;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
+</svg>
diff --git a/img/padlock_close_grey.svg b/img/padlock_close_grey.svg
new file mode 100644
index 0000000..a09d4c6
--- /dev/null
+++ b/img/padlock_close_grey.svg
@@ -0,0 +1,94 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   id="svg3041"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   width="15"
+   height="15"
+   sodipodi:docname="padlock_close_grey.svg">
+  <metadata
+     id="metadata3047">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs3045" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1855"
+     inkscape:window-height="1056"
+     id="namedview3043"
+     showgrid="false"
+     inkscape:zoom="32"
+     inkscape:cx="12.792117"
+     inkscape:cy="8.9449238"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg3041"
+     borderlayer="true" />
+  <rect
+     style="fill:#808080;stroke:#808080;stroke-width:0.92522794"
+     id="rect3051"
+     width="10.232394"
+     height="6.4687548"
+     x="2.5534275"
+     y="8.074255" />
+  <a
+     id="a4157"
+     style="fill:none;stroke:#808080">
+    <path
+       sodipodi:open="true"
+       d="M 3.1372476,6.7928489 A 4.534133,4.5301714 0 0 1 7.6713806,2.2626777 4.534133,4.5301714 0 0 1 12.205513,6.7928491"
+       sodipodi:end="0"
+       sodipodi:start="3.1415927"
+       sodipodi:ry="4.5301714"
+       sodipodi:rx="4.534133"
+       sodipodi:cy="6.7928491"
+       sodipodi:cx="7.6713805"
+       sodipodi:type="arc"
+       id="path4101"
+       style="fill:none;fill-opacity:1;stroke:#808080;stroke-width:2.08827519;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
+  </a>
+  <g
+     id="g4170"
+     style="fill:#808080">
+    <rect
+       y="6.7554431"
+       x="2.0913804"
+       height="1.0984892"
+       width="2.0906768"
+       id="rect4153"
+       style="fill:#808080;fill-opacity:1;stroke:none;stroke-width:1.39526081;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
+    <rect
+       style="fill:#808080;fill-opacity:1;stroke:none;stroke-width:1.39526081;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+       id="rect4155"
+       width="2.0906768"
+       height="1.0984892"
+       x="11.157786"
+       y="6.7632556" />
+  </g>
+</svg>
diff --git a/img/padlock_close_orange.svg b/img/padlock_close_orange.svg
new file mode 100644
index 0000000..3338356
--- /dev/null
+++ b/img/padlock_close_orange.svg
@@ -0,0 +1,94 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   id="svg3041"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   width="15"
+   height="15"
+   sodipodi:docname="padlock_close_orange.svg">
+  <metadata
+     id="metadata3047">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs3045" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1855"
+     inkscape:window-height="1056"
+     id="namedview3043"
+     showgrid="false"
+     inkscape:zoom="32"
+     inkscape:cx="12.792117"
+     inkscape:cy="8.9449238"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg3041"
+     borderlayer="true" />
+  <rect
+     style="fill:#ff7f2a;stroke:#ff7f2a;stroke-width:0.92522794"
+     id="rect3051"
+     width="10.232394"
+     height="6.4687548"
+     x="2.5534275"
+     y="8.074255" />
+  <a
+     id="a4157"
+     style="fill:none;stroke:#ff7f2a">
+    <path
+       sodipodi:open="true"
+       d="M 3.1372476,6.7928489 A 4.534133,4.5301714 0 0 1 7.6713806,2.2626777 4.534133,4.5301714 0 0 1 12.205513,6.7928491"
+       sodipodi:end="0"
+       sodipodi:start="3.1415927"
+       sodipodi:ry="4.5301714"
+       sodipodi:rx="4.534133"
+       sodipodi:cy="6.7928491"
+       sodipodi:cx="7.6713805"
+       sodipodi:type="arc"
+       id="path4101"
+       style="fill:none;fill-opacity:1;stroke:#ff7f2a;stroke-width:2.08827519;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
+  </a>
+  <g
+     id="g4170"
+     style="fill:#ff7f2a">
+    <rect
+       y="6.7554431"
+       x="2.0913804"
+       height="1.0984892"
+       width="2.0906768"
+       id="rect4153"
+       style="fill:#ff7f2a;fill-opacity:1;stroke:none;stroke-width:1.39526081;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1" />
+    <rect
+       style="fill:#ff7f2a;fill-opacity:1;stroke:none;stroke-width:1.39526081;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+       id="rect4155"
+       width="2.0906768"
+       height="1.0984892"
+       x="11.157786"
+       y="6.7632556" />
+  </g>
+</svg>
diff --git a/img/padlock_open.svg b/img/padlock_open.svg
new file mode 100644
index 0000000..60aabe1
--- /dev/null
+++ b/img/padlock_open.svg
@@ -0,0 +1,39 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   version="1.1"
+   width="15"
+   height="15"
+   id="svg3041">
+  <metadata
+     id="metadata3047">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs3045" />
+  <rect
+     width="11.059322"
+     height="6.9915257"
+     x="2.1610172"
+     y="8.0084743"
+     id="rect3051"
+     style="fill:#999999;stroke:none" />
+  <path
+     d="m 3.082628,5.0211864 c 0,0 -0.127119,-4.06779652 4.449152,-4.06779652 4.576268,0 4.703386,3.94067782 4.703386,3.94067782 l 0.06356,5.5932193 0,0"
+     id="path3848"
+     style="fill:none;stroke:#999999;stroke-width:1.89999998;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none" />
+</svg>
diff --git a/img/padlock_open_grey.svg b/img/padlock_open_grey.svg
new file mode 100644
index 0000000..769837b
--- /dev/null
+++ b/img/padlock_open_grey.svg
@@ -0,0 +1,66 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   id="svg3041"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   width="15"
+   height="15"
+   sodipodi:docname="padlock_open-white.svg">
+  <metadata
+     id="metadata3047">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs3045" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1855"
+     inkscape:window-height="1056"
+     id="namedview3043"
+     showgrid="false"
+     inkscape:zoom="44.500587"
+     inkscape:cx="6.292871"
+     inkscape:cy="10.986979"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg3041"
+     borderlayer="true" />
+  <rect
+     style="fill:#808080;stroke:#808080;stroke-width:0.92522794"
+     id="rect3051"
+     width="10.232394"
+     height="6.4687548"
+     x="2.5534275"
+     y="8.074255" />
+  <path
+     style="color:#000000;font-style:normal;font-variant:normal;font-weight:normal;font-stretch:normal;font-size:medium;line-height:normal;font-family:sans-serif;text-indent:0;text-align:start;text-decoration:none;text-decoration-line:none;text-decoration-color:currentColor;letter-spacing:normal;word-spacing:normal;text-transform:none;direction:ltr;block-progression:tb;writing-mode:lr-tb;baseline-shift:baseline;text-anchor:start;clip-rule:nonzero;display:inline;overflow:visible;visibilit [...]
+     d="M 7.5078125,0.00556144 C 5.0260054,0.00556144 3.491603,0.79834464 2.75,1.9469676 2.008397,3.0955907 2.0878906,4.3496954 2.0878906,5.0250926 l 1.9042969,0 c 0,-0.7329344 0.013699,-1.5125503 0.3574219,-2.0449218 0.3437231,-0.5323715 1.0433343,-1.0703125 3.1582031,-1.0703125 2.0264219,0 2.9669635,0.6265545 3.4326915,1.3827484 0.465728,0.7561939 0.400333,1.7142789 0.400333,1.7142789 0.04517,1.7732263 -0.06599,4.3031738 0.0049,5.5357844 l 0.05274,0.898438 0.898437,0 0.945445,-0.0031 0 [...]
+     id="path3848"
+     inkscape:connector-curvature="0"
+     sodipodi:nodetypes="ssccssscccccccs" />
+</svg>
diff --git a/img/padlock_open_white.svg b/img/padlock_open_white.svg
new file mode 100644
index 0000000..0d0baed
--- /dev/null
+++ b/img/padlock_open_white.svg
@@ -0,0 +1,64 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   id="svg3041"
+   version="1.1"
+   inkscape:version="0.48.3.1 r9886"
+   width="15"
+   height="15"
+   sodipodi:docname="padlock_open-white.svg">
+  <metadata
+     id="metadata3047">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs3045" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1444"
+     inkscape:window-height="787"
+     id="namedview3043"
+     showgrid="false"
+     inkscape:zoom="44.500587"
+     inkscape:cx="9.2134583"
+     inkscape:cy="8.0232356"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="0"
+     inkscape:current-layer="svg3041" />
+  <rect
+     style="fill:#ffffff;stroke:#ffffff"
+     id="rect3051"
+     width="11.059322"
+     height="6.9915257"
+     x="2.1610172"
+     y="8.0084743" />
+  <path
+     style="fill:none;stroke:#ffffff;stroke-width:1.89999998000000003;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-opacity:1;stroke-dasharray:none"
+     d="m 3.082628,5.0211864 c 0,0 -0.127119,-4.06779652 4.449152,-4.06779652 4.576268,0 4.703386,3.94067782 4.703386,3.94067782 l 0.06356,5.5932193 0,0"
+     id="path3848"
+     inkscape:connector-curvature="0" />
+</svg>
diff --git a/img/smiley.png b/img/smiley.png
new file mode 100644
index 0000000..9a72067
Binary files /dev/null and b/img/smiley.png differ
diff --git a/img/speech_balloon_white.svg b/img/speech_balloon_white.svg
new file mode 100644
index 0000000..465f4a8
--- /dev/null
+++ b/img/speech_balloon_white.svg
@@ -0,0 +1,66 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<!-- Created with Inkscape (http://www.inkscape.org/) -->
+
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="25"
+   height="25"
+   id="svg2"
+   version="1.1"
+   inkscape:version="0.48+devel r"
+   viewBox="0 0 25 25.000001"
+   sodipodi:docname="speech_balloon_white.svg">
+  <defs
+     id="defs4" />
+  <sodipodi:namedview
+     id="base"
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1.0"
+     inkscape:pageopacity="0.0"
+     inkscape:pageshadow="2"
+     inkscape:zoom="22.4"
+     inkscape:cx="25.186628"
+     inkscape:cy="14.479999"
+     inkscape:document-units="px"
+     inkscape:current-layer="layer1"
+     showgrid="false"
+     fit-margin-top="0"
+     fit-margin-left="0"
+     fit-margin-right="0"
+     fit-margin-bottom="0"
+     inkscape:window-width="1855"
+     inkscape:window-height="1056"
+     inkscape:window-x="65"
+     inkscape:window-y="24"
+     inkscape:window-maximized="1" />
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <g
+     inkscape:label="Layer 1"
+     inkscape:groupmode="layer"
+     id="layer1"
+     transform="translate(-228.24219,-320.66798)">
+    <path
+       style="fill:#ffffff;fill-opacity:1;stroke:#000000;stroke-width:0;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+       d="m 240.77627,324.02578 a 12.489309,7.9254426 0 0 0 -12.48944,7.92538 12.489309,7.9254426 0 0 0 5.26146,6.45637 c -0.17862,1.91196 -1.06119,3.95363 -3.14604,4.64437 2.54121,-0.11444 5.60026,-1.93057 7.47209,-3.40137 a 12.489309,7.9254426 0 0 0 2.90193,0.22616 12.489309,7.9254426 0 0 0 12.48928,-7.92553 12.489309,7.9254426 0 0 0 -12.48928,-7.92538 z"
+       id="path4101"
+       inkscape:connector-curvature="0" />
+  </g>
+</svg>
diff --git a/jsxc.css b/jsxc.css
new file mode 100644
index 0000000..7692f9b
--- /dev/null
+++ b/jsxc.css
@@ -0,0 +1,1243 @@
+.jsxc_online,.jsxc_chat {
+	border-left: 4px solid #00BFFF !important;
+}
+
+.jsxc_away,.jsxc_xa {
+	border-left: 4px solid orange !important;
+}
+
+.jsxc_dnd {
+	border-left: 4px solid red !important;
+}
+
+.jsxc_offline {
+	border-left: 4px solid #A4A4A4 !important;
+}
+
+.jsxc_hidden {
+	display: none;
+}
+
+.jsxc_invalid {
+	border: 2px solid red !important;
+}
+
+p.jsxc_right {
+	text-align: right;
+}
+
+p.jsxc_center {
+	text-align: center;
+}
+
+li.jsxc_sep {
+	border-top: 1px solid black;
+}
+
+/*############################ 
+ *########## Dialog ########## 
+ *############################*/
+#jsxc_dialog {
+	padding: 10px;
+	min-width: 300px;
+}
+
+#jsxc_dialog ul {
+    list-style: none;
+}
+
+#jsxc_dialog ul, #jsxc_dialog li {
+    margin: 0px;
+    padding: 0px;
+}
+
+#jsxc_dialog p {
+	margin-bottom: 1em;
+}
+
+#jsxc_dialog h3 {
+	font-size: 120%;
+	font-weight: bold;
+	margin-bottom: 10px;
+	margin-top: 20px;
+}
+
+#jsxc_dialog p input {
+	margin-bottom: 5px;
+	width: 60%;
+	outline: none;
+}
+
+#jsxc_dialog p input[type="submit"] {
+	width: auto;
+}
+
+#jsxc_dialog p label {
+	display: block;
+	width: 30%;
+	float: left;
+	padding: 6px 0px;
+}
+
+#jsxc_dialog .jsxc_right {
+	margin-top: 20px;
+}
+
+#jsxc_dialog p.jsxc_maxWidth {
+	max-width: 500px;
+}
+
+#jsxc_dialog fieldset {
+	border: 2px solid #ccc;
+}
+
+#jsxc_dialog legend {
+	padding: 0px 10px;
+	font-weight: bold;
+}
+
+#jsxc_dialog input {
+	outline: none;
+}
+
+#jsxc_dialog input:invalid {
+	border:1px solid red;
+}
+
+#jsxc_dialog .jsxc_fieldset {
+	width: 280px;
+	padding: 10px;
+	border-radius: 3px;
+}
+
+#jsxc_dialog .jsxc_fieldset label {
+	display: inline-block;
+	width: 100px;
+}
+
+#jsxc_dialog .jsxc_fieldset input[type="Number"] {
+   display: inline-block;
+   width: 50px;
+}
+
+#jsxc_dialog input[readonly] {
+	background-color: #efefef;
+}
+
+/*############################ 
+ *########## Overall ######### 
+ *############################*/
+li .jsxc_name {
+	overflow: hidden;
+	height: 30px;
+	cursor: pointer;
+	text-overflow: ellipsis;
+	white-space: nowrap;
+}
+li .jsxc_name:hover {
+   color: #FAFAFA;
+}
+
+.jsxc_avatar,ul#jsxc_buddylist li:hover .jsxc_avatar {
+	width: 25px;
+	height: 25px;
+	line-height: 25px;
+	margin-top: 2px;
+	margin-right: 4px;
+	background-color: #a3a3a3;
+	border-radius: 2px;
+	float: left;
+	text-align: center;
+	font-weight: bold;
+	font-size: 30px;
+	color: #b1b1b1;
+	overflow: hidden;
+	position: relative;
+	font-family: Arial;
+	background-size: cover;
+	background-position: center center;
+}
+
+.jsxc_avatar img {
+	display: block;
+	width: 25px;
+	height: 25px;
+	position: absolute;
+	top: 0px;
+	left: 0px;
+}
+
+.jsxc_spot {
+	display: inline-block;
+   width: 10px;
+   height: 10px;
+   background-color: #fff;
+   border-radius: 5px;
+   text-indent: -99999em;
+   margin-top: 3px;
+   margin-right: 5px;
+   border: 1px solid #000 !important;
+   line-height: 100%;
+   cursor: pointer;
+}
+
+.jsxc_spot.jsxc_online, .jsxc_spot.jsxc_chat {
+   background-color: #00BFFF;
+}
+
+.jsxc_spot.jsxc_away, .jsxc_spot.jsxc_xa {
+   background-color: orange;
+}
+
+.jsxc_spot.jsxc_dnd {
+   background-color: red;
+}
+
+.jsxc_spot.jsxc_offline {
+   background-color: #A4A4A4;
+}
+
+/*############################ 
+ *########## Roster ########## 
+ *############################*/
+div#jsxc_roster {
+	position: fixed;
+	top: 0px;
+	bottom: 0px;
+	right: 0px;
+	width: 200px;
+	overflow: visible;
+	border-left: 4px solid #A4A4A4;
+	z-index: 80;
+	margin-left: 10px;
+	box-shadow: 0px 0px 7px #000000;
+	background-color: #383C43;
+}
+
+div#jsxc_roster .slimScrollDiv {
+	left: -4px;
+	margin-bottom: 30px;
+	z-index: 40;
+}
+
+div#jsxc_roster .jsxc_wait {
+	position: absolute;
+	top: 0px;
+	left: 0px;
+	bottom: 0px;
+	width: 160px;
+	padding: 20px;
+	background-color: #fff;
+	z-index: 60;
+}
+
+div#jsxc_roster .jsxc_wait img {
+	padding: 5px;
+}
+
+div#jsxc_roster .jsxc_wait h3 {
+	margin-bottom: 5px;
+	font-size: 1.13em;
+	font-weight: bold;
+}
+
+/** Input field for alias renaming */
+#jsxc_roster input {
+	position: absolute;
+	top: 50%;
+	left: 31px;
+	width: 157px;
+	height: 18px;
+	margin-top: -11px;
+	padding: 2px;
+	border: 0px;
+	background-image: none;
+	background-color: #fff;
+	border-radius: 3px;
+	-moz-box-shadow: inset 0 0 5px #888;
+	-webkit-box-shadow: inset 0 0 5px #888;
+	box-shadow: inner 0 0 5px #888;
+	outline: none;
+}
+
+#jsxc_roster .jsxc_expand input {
+   left: 51px;
+   width: 137px;
+}
+
+div#jsxc_roster p {
+	color: #939393;
+	padding: 10px;
+}
+
+div#jsxc_roster p a {
+	color: #b1b1b1;
+	text-decoration: underline;
+}
+
+div#jsxc_roster>.jsxc_bottom {
+	position: absolute;
+	left: -4px;
+	bottom: 0px;
+	height: 30px;
+	width: 200px;
+	border-left: 4px solid #A4A4A4;
+	border-top: 1px solid #939393;
+	padding-left: 0px;
+	background-color: #282323;
+	color: #939393;
+	cursor: pointer;
+	line-height: 30px;
+	z-index: 50;
+}
+
+div#jsxc_roster>.jsxc_bottom>div:hover {
+	color: #FAFAFA;
+}
+
+div#jsxc_roster>.jsxc_bottom>div>span {
+	cursor: pointer;
+}
+
+div#jsxc_roster>.jsxc_bottom ul {
+	width: 190px;
+	position: absolute;
+	bottom: 30px;
+	left: 0px;
+	display: none;
+	border-top: 1px solid #939393;
+	padding-left: 10px;
+	background-color: #282323;
+	margin: 0;
+}
+
+div#jsxc_roster>.jsxc_bottom li {
+	height: 30px;
+	background-color: #282323;
+	color: #939393;
+	cursor: pointer;
+	width: 180px;
+	padding-left: 10px;
+	line-height: 30px;
+	text-align: left;
+	overflow: hidden;
+	text-overflow: ellipsis;
+	white-space: nowrap;
+}
+
+div#jsxc_roster>.jsxc_bottom li:hover:not(.jsxc_disabled ) {
+	color: #fff;
+	background-color: #3F8FBA;
+}
+
+div#jsxc_roster>.jsxc_bottom li.jsxc_disabled {
+	color: #595959;
+	cursor: default;
+}
+
+div#jsxc_roster>.jsxc_bottom ul li:last-child {
+	border-bottom: 1px solid #939393;
+}
+
+#jsxc_menu,#jsxc_avatar,#jsxc_notice {
+	height: 30px;
+	width: 30px;
+}
+
+/** own avatar */
+#jsxc_avatar {
+	float: left;
+	margin: 0px 2px;
+	width: 25px;
+}
+
+#jsxc_menu {
+	cursor: pointer;
+	float: right;
+	text-align: center;
+	width: 25px;
+}
+
+#jsxc_menu>span {
+	opacity: 0.5;
+	display: block;
+	width: 25px;
+	height: 30px;
+	background-image: url('img/gear_white.svg');
+	background-size: 18px 18px;
+	background-repeat: no-repeat;
+	background-position: center center;
+}
+
+#jsxc_menu:hover>span {
+	opacity: 1;
+}
+
+#jsxc_notice {
+	float: right;
+	text-align: center;
+	line-height: 30px;
+}
+
+#jsxc_notice>span:empty {
+	display: none;
+}
+
+#jsxc_notice span {
+	background-color: #fbc14c;
+	border-radius: 11px;
+	color: black;
+	font-size: 80%;
+	padding: 2px;
+	position: relative;
+	-webkit-animation: bounce 2s 1s infinite;
+	animation: bounce 2s 1s infinite;
+}
+
+ at -webkit-keyframes bounce {
+   0% {
+      bottom: 0;
+      -webkit-animation-timing-function: ease-in;
+   }
+   12% {
+      bottom: 5px; 
+      -webkit-animation-timing-function: ease-out;
+   }
+   25% {
+      bottom: 0px;
+      -webkit-animation-timing-function: ease-in;
+   }
+   37% {
+      bottom: 5px;
+      -webkit-animation-timing-function: ease-out;
+   }
+   50% {
+      bottom: 0px;
+      -webkit-animation-timing-function: ease-in;
+   }
+   100% {
+      bottom: 0px;
+   }
+}
+
+ at keyframes bounce {
+   0% {
+      bottom: 0;
+      animation-timing-function: ease-in;
+   }
+   12% {
+      bottom: 5px; 
+      animation-timing-function: ease-out;
+   }
+   25% {
+      bottom: 0px;
+      animation-timing-function: ease-in;
+   }
+   37% {
+      bottom: 5px;
+      animation-timing-function: ease-out;
+   }
+   50% {
+      bottom: 0px;
+      animation-timing-function: ease-in;
+   }
+   100% {
+      bottom: 0px;
+   }
+}
+
+
+#jsxc_presence {
+	cursor: pointer;
+	padding-left: 2px;
+	overflow: hidden;
+}
+
+/** List of all buddies inside the roster */
+ul#jsxc_buddylist {
+	list-style: none;
+	padding: 0px;
+	margin: 0px;
+	width: 204px;
+}
+
+ul#jsxc_buddylist li {
+	padding: 0px;
+	margin: 0px;
+	height: 30px;
+	border-bottom: 1px solid #939393;
+	cursor: pointer;
+	width: 200px;
+	position: relative;
+	color: #939393;
+	font-family: Arial;
+	line-height: 30px;
+	padding-left: 3px;
+	z-index: 85;
+	overflow: hidden;
+}
+
+ul#jsxc_buddylist .jsxc_name {
+	padding-right: 30px;
+}
+
+ul#jsxc_buddylist li .jsxc_name:before {
+	content: ' ';
+	width: 0px;
+	height: 0px;
+	display: inline-block;
+	opacity: 0.0;
+	-webkit-transition: width 0.5s, opacity 0.5s;
+   transition: width 0.5s, opacity 0.5s;
+}
+
+ul#jsxc_buddylist li.jsxc_unreadMsg .jsxc_name:before {
+   width: 12px;
+   height: 12px;
+   border-radius: 6px;
+   background-color: orange;
+   margin-right: 3px;
+   opacity: 1.0;
+   -webkit-transition: width 0.5s, opacity 0.5s;
+   transition: width 0.5s, opacity 0.5s;
+}
+
+ul#jsxc_buddylist li.jsxc_expand {
+	height: 54px;
+}
+
+ul#jsxc_buddylist li.jsxc_expand .jsxc_avatar, ul#jsxc_buddylist li.jsxc_expand .jsxc_avatar img {
+	width: 48px;
+	height: 48px;
+}
+
+#jsxc_buddylist .jsxc_control {
+   position:absolute;
+   width: 30px;
+   height: 30px;
+   top: 0px;
+   right: 0px;
+   cursor: pointer;
+}
+
+#jsxc_buddylist .jsxc_control:after {
+	content: '▾';
+	position: absolute;
+	left: 50%;
+	top: 15px;
+	width: 16px;
+	height: 16px;
+	line-height: 16px;
+	text-align: center;
+	margin-left: -8px;
+	margin-top: -8px;
+	border-radius: 8px;
+	color: #c1c1c1;
+	background-color: rgba(150, 150, 150, 0.5);
+	opacity: 0.6;
+	-webkit-transition: -webkit-transform 0.5s;
+   transition: transform 0.5s;
+}
+
+#jsxc_buddylist .jsxc_control:hover:after {
+   color: #fff;
+	opacity: 1.0;
+}
+
+#jsxc_buddylist li.jsxc_expand .jsxc_control:after {
+	/* content: '▴'; */
+	-ms-transform:rotate(180deg); /* IE 9 */
+   -webkit-transform:rotate(180deg);
+	transform:rotate(180deg);
+}
+
+#jsxc_buddylist .jsxc_options {
+    height: 20px;
+    float: left;
+    border-radius:2px;
+    background-color: rgba(150, 150, 150, 0.5);
+}
+#jsxc_buddylist .jsxc_options > div {
+    height: 20px;
+    width: 20px;
+    float: left;
+    margin-right:0px 1px;
+    background-repeat: no-repeat;
+    background-position: center center;
+    opacity: 0.6;
+    cursor: pointer;
+}
+#jsxc_buddylist .jsxc_options > div:hover {
+	opacity: 1;
+}
+#jsxc_buddylist .jsxc_chaticon {	
+	background-image: url('img/speech_balloon_white.svg');
+	background-size: 15px 15px;
+}
+#jsxc_buddylist .jsxc_videoicon { 
+   background-image: url('img/camera_icon_white.svg');
+   background-size: 13px 13px;
+}
+#jsxc_buddylist .jsxc_vcardicon {
+    font-size: 15px;
+    line-height: 20px;
+    text-align: center;
+    color: #fff;
+    font-weight: bold;
+    font-family: serif;
+}
+
+ul#jsxc_buddylist li:hover {
+	
+}
+
+ul#jsxc_buddylist .jsxc_oneway {
+	opacity: 0.7;
+}
+.jsxc_oneway .jsxc_avatar {
+	-webkit-filter: grayscale(100%);
+   -moz-filter: grayscale(100%);
+   -ms-filter: grayscale(100%);
+   -o-filter: grayscale(100%);
+   filter: grayscale(100%);
+}
+ul#jsxc_buddylist .jsxc_right {
+	float:left;
+}
+ul#jsxc_buddylist .jsxc_right {
+	float: right;
+	margin-right: 6px;
+}
+
+ul#jsxc_buddylist .jsxc_right div {
+	font-weight: bold;
+	text-align: center;
+	font-size: 13px;
+	line-height: 20px;
+	color: #fff;
+}
+
+ul#jsxc_buddylist li[data-type!="chat"] .jsxc_right {
+	display: none;
+}
+
+ul#jsxc_buddylist .jsxc_right div:hover {
+	opacity: 1.0;
+}
+
+ul#jsxc_buddylist .jsxc_rename {
+
+}
+
+ul#jsxc_buddylist .jsxc_delete {
+
+}
+
+#jsxc_roster .jsxc_avatar {
+   position: relative;
+   cursor: pointer;
+}
+#jsxc_roster .jsxc_avatar img {
+   cursor: pointer;
+}
+
+div#jsxc_toggleRoster {
+	width: 14px;
+	height: 100%;
+	position: absolute;
+	left: -14px !important;
+	top: 0px;
+	z-index: 110;
+	background-color: transparent;
+	cursor: pointer;
+}
+
+#jsxc_toggleRoster:hover {
+	background-color: #a4a4a4;
+	opacity: 0.5;
+}
+
+/*############################ 
+ *###### Window List ######### 
+ *############################*/
+div#jsxc_windowList {
+	position: fixed;
+	bottom: 0px;
+	right: 210px;
+	left: 0px;
+	z-index: 50;
+	clip: rect(-10000px, 10000px, 30px, 30px);
+}
+
+div#jsxc_windowList>ul {
+	list-style: none;
+	padding: 0px;
+	margin: 0px;
+	position: absolute;
+	bottom: 0px;
+	right: 0px;
+	height: 30px;
+	overflow: visible;
+	white-space: nowrap;
+	-webkit-transition: right 0.5s;
+	transition: right 0.5s;
+}
+
+div#jsxc_windowList>ul>li {
+	padding: 0px;
+	margin: 0px;
+	display: inline-block;
+	width: 250px;
+	background-color: #282323;
+	height: 30px;
+	position: relative;
+	overflow: visible;
+	margin-right: 5px;
+	cursor: pointer;
+	white-space: normal;
+}
+
+div#jsxc_windowList>ul>li.jsxc_min {
+	width: 200px !important;
+}
+
+#jsxc_windowListSB {
+	position: fixed;
+	left: 0px;
+	bottom: 0px;
+	width: 30px;
+	height: 30px;
+}
+#jsxc_windowListSB > div {
+	box-sizing: border-box;
+	width: 14px;
+	height: 100%;
+	background-color: #D3D3D3;
+	color: #999;
+	text-align: center;
+	line-height: 30px;
+	float: left;
+	cursor: pointer;
+	-webkit-touch-callout: none;
+	-webkit-user-select: none;
+	-khtml-user-select: none;
+	-moz-user-select: none;
+	-ms-user-select: none;
+	user-select: none;
+}
+#jsxc_windowListSB > div:hover {
+	background-color: #c3c3c3;
+}
+#jsxc_windowListSB > div.jsxc_disabled {
+	background-color: #f1f1f1 !important;
+	color: #d1d1d1;
+	cursor: default !important;
+	display: none;
+}
+
+/*############################ 
+ *######### Window ########### 
+ *############################*/
+div.jsxc_bar {
+	background-color: #282323;
+	cursor: pointer;
+	height: 30px;
+	line-height: 26px;
+	padding: 2px;
+	color: #939393;
+	width: 100%;
+	box-sizing: border-box;
+	overflow: hidden;
+}
+
+div.jsxc_cycle {
+	position: absolute;
+	bottom: 9px;
+	right: 5px;
+	height: 12px;
+	width: 12px;
+	border-radius: 6px;
+}
+
+div#jsxc_windowList>ul>li.jsxc_unreadMsg .jsxc_name {
+	font-style: italic;
+}
+
+div#jsxc_windowList>ul>li.jsxc_unreadMsg .jsxc_cycle {
+	background-color: orange;
+}
+
+/** Contains text area */
+div.jsxc_window {
+	position: absolute;
+	bottom: 0px;
+	left: 0px;
+	right: 0px;
+	height: auto;
+	background-color: #A4A4A4;
+	z-index: 80;
+	cursor: default;
+}
+
+div.jsxc_window .jsxc_emoticons {
+	height: 30px;
+	width: 30px;
+	position: absolute;
+	top: 240px;
+	/*bottom: 0px;*/
+	right: 0px;
+	cursor: pointer;
+}
+
+div.jsxc_window .jsxc_emoticons:after {
+	content: ' ';
+	background-image: url('img/smiley.png');
+	background-position: center center;
+	background-repeat: no-repeat;
+	height: 30px;
+	width: 30px;
+	opacity: 0.3;
+	position: absolute;
+	top: 0px;
+	left: 0px;
+}
+
+div.jsxc_window .jsxc_emoticons:hover:after {
+	opacity: 0.5;
+}
+
+div.jsxc_window .jsxc_emoticons ul {
+	display: none;
+	position: absolute;
+	bottom: 33px;
+	right: 3px;
+	width: 210px;
+	background-color: rgba(40, 35, 35, 0.8);
+	padding: 3px;
+	border-radius: 3px;
+	z-index: 200;
+	list-style-type: none;
+}
+
+div.jsxc_window .jsxc_emoticons ul:after {
+	content: '';
+	position: absolute;
+	border-left: 5px solid transparent;
+	border-right: 5px solid transparent;
+	border-top: 6px solid rgba(40, 35, 35, 0.8);
+	display: block;
+	width: 0;
+	z-index: 1;
+	right: 6px;
+	bottom: -6px;
+}
+
+div.jsxc_window .jsxc_emoticons li {
+	float: right;
+	cursor: pointer;
+}
+
+div.jsxc_window .jsxc_emoticons div {
+	cursor: pointer;
+	border-radius: 3px;
+	background-size: 30px 30px;
+	width: 30px;
+	height: 30px;
+}
+
+div.jsxc_window .jsxc_emoticons div:hover {
+	background-color: rgba(255, 255, 255, 0.8);
+}
+
+/** groups elements, which will be hidden, if the window is minimized. */
+.jsxc_window .jsxc_fade {
+	position: relative;
+}
+
+.jsxc_window .jsxc_gradient { 
+	display:none;
+	position: absolute;
+	top: 0px;
+	left: 0px;
+	right: 0px;
+	height: 15px;
+	z-index: 100;
+	background: -moz-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+		rgba(40, 35, 35, 0) 100%); /* FF3.6+ */
+	background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(164,
+		164, 164, 1)), color-stop(100%, rgba(40, 35, 35, 0)));
+	/* Chrome,Safari4+ */
+	background: -webkit-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+		rgba(40, 35, 35, 0) 100%); /* Chrome10+,Safari5.1+ */
+	background: -o-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+		rgba(40, 35, 35, 0) 100%); /* Opera 11.10+ */
+	background: -ms-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+		rgba(40, 35, 35, 0) 100%); /* IE10+ */
+	background: linear-gradient(to bottom, #a4a4a4 0%, rgba(40, 35, 35, 0)
+		100%); /* W3C */
+	filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#a4a4a4',
+		endColorstr='#00282323', GradientType=0); /* IE6-9 */
+}
+
+div.jsxc_window .jsxc_avatar {
+	margin-top: 1px;
+}
+
+div.jsxc_window .jsxc_textarea {
+	width: 100%;
+	overflow: hidden;
+	padding: 3px;
+}
+
+div.jsxc_window .slimScrollDiv {
+	margin: 0px 0px 6px 0px;
+	left: auto !important;
+   top: auto !important;
+}
+
+div.jsxc_chatmessage {
+	margin: 3px;
+	padding: 4px;
+	padding-right: 10px;
+	max-width: 76%;
+	word-wrap: break-word;
+	background-color: #fff;
+	position: relative;
+	outline: none;
+	clear: both;
+}
+
+div.jsxc_chatmessage a {
+	color: blue;
+	text-decoration: underline;
+}
+
+div.jsxc_chatmessage img {
+	width: 19px;
+	height: 19px;
+	background-size: 19px 19px;
+}
+
+/** incoming message */
+div.jsxc_in {
+	float: left;
+	position: relative;
+	margin-left: 10px;
+	background: #f1f7fd;
+	border: solid 1px #a6b8c9;
+	-webkit-border-radius: 3px;
+	-moz-border-radius: 3px;
+	border-radius: 3px;
+}
+
+div.jsxc_in:after {
+	content: '';
+	position: absolute;
+	border-style: solid;
+	border-width: 5px 6px 5px 0;
+	border-color: transparent #f1f7fd;
+	display: block;
+	width: 0;
+	z-index: 1;
+	left: -6px;
+	bottom: 10px;
+}
+
+div.jsxc_out.jsxc_received:before {
+	content: '✓';
+	position: absolute;
+	bottom: 2px;
+	right: 2px;
+	font-size: 12px;
+	line-height: 12px;
+	color: green;
+}
+
+/** outgoing message */
+div.jsxc_out {
+	float: right;
+	position: relative;
+	margin-right: 10px;
+	background: #e6ffd1;
+	border: solid 1px #b4d89f;
+	-webkit-border-radius: 3px;
+	-moz-border-radius: 3px;
+	border-radius: 3px;
+}
+
+div.jsxc_out:after {
+	content: '';
+	position: absolute;
+	border-style: solid;
+	border-width: 5px 0 5px 6px;
+	border-color: transparent #e6ffd1;
+	display: block;
+	width: 0;
+	z-index: 1;
+	right: -6px;
+	bottom: 10px;
+}
+
+/** system message */
+div.jsxc_sys {
+	width: auto;
+	max-width: none;
+	padding-right: 4px;
+	box-sizing: border-box;
+	margin-right: 3px;
+	border-radius: 3px;
+}
+
+.jsxc_window input.jsxc_textinput {
+	width: 100%;
+	height: 30px;
+	margin: 0px;
+	padding: 3px;
+	padding-right: 40px;
+	outline: none;
+	border-radius: 0px;
+	box-sizing: border-box;
+	text-align: right;
+}
+
+.jsxc_window ::-webkit-input-placeholder {
+	color: #000;
+	opacity: 0.3;
+}
+
+.jsxc_window ::-moz-placeholder {
+	color: #000;
+	opacity: 0.3;
+} /* firefox 19+ */
+.jsxc_window :-ms-input-placeholder {
+	color: rgba(0, 0, 0, 0.4);
+} /* ie */
+.jsxc_window input:-moz-placeholder {
+	color: #000;
+	opacity: 0.3;
+}
+
+div.jsxc_window .jsxc_tools {
+	height: 26px;
+	float: right;
+}
+
+.jsxc_min .jsxc_tools {
+	display: none;
+}
+
+div.jsxc_window .jsxc_tools > div {
+	width: 17px;
+   height: 26px;
+   display: block;
+   float: left;
+   color: #939393;
+   font-family: Arial;
+   line-height: 26px;
+   cursor: pointer;
+   text-align: center;
+}
+
+div.jsxc_settings {
+	background-image: url('img/gear_grey.svg');
+	background-size: 15px 15px;
+	background-repeat: no-repeat;
+	background-position: center center;
+	overflow: visible;
+}
+
+div.jsxc_settings:hover,div.jsxc_settings.hover {
+	background-image: url('img/gear_white.svg');
+}
+
+div.jsxc_settings ul {
+	list-style: none;
+	padding: 0px;
+	margin: 0px;
+	position: absolute;
+	top: 30px;
+	left: 0px;
+	box-shadow: 0px 5px 5px -5px #000;
+	z-index: 110;
+	overflow: visible;
+	background-color: #282323;
+	display: none;
+}
+
+div.jsxc_settings li {
+	width: 170px;
+	margin-left: 20px;
+	padding-left: 10px;
+	height: 30px;
+	line-height: 30px;
+	text-align: left;
+	cursor: pointer;
+}
+
+div.jsxc_settings li:first-child {
+	border-top: 1px solid #ccc;
+}
+
+div.jsxc_settings li:hover:not(.jsxc_disabled ){
+	color: #fff;
+	background-color: #3F8FBA;
+}
+
+div.jsxc_settings li.jsxc_disabled {
+	color: #595959;
+	cursor: default;
+}
+
+.jsxc_tools > div.jsxc_disabled {
+   opacity: 0.5;
+   cursor: default !important;
+}
+
+div.jsxc_transfer, div.jsxc_transfer.jsxc_disabled:hover {
+	background-image: url('img/padlock_open_grey.svg');
+	background-repeat: no-repeat;
+	background-position: center center;
+	background-size: 12px 12px;
+}
+
+div.jsxc_transfer:hover {
+	background-image: url('img/padlock_open_white.svg');
+}
+
+div.jsxc_transfer.jsxc_fin {
+	background-image: url('img/padlock_close_grey.svg');
+}
+
+div.jsxc_transfer.jsxc_enc {
+	background-image: url('img/padlock_close_orange.svg');
+}
+
+div.jsxc_transfer.jsxc_enc.jsxc_trust {
+	background-image: url('img/padlock_close_green.svg');
+}
+
+.jsxc_window .jsxc_close {
+	font-size: 20px;
+}
+
+.jsxc_window .jsxc_close:hover {
+	color: #FAFAFA;
+}
+
+#colorbox,#cboxWrapper {
+	outline: none;
+}
+
+/*############################ 
+ *######### Other ############ 
+ *############################*/
+ul.jsxc_vCard {
+	min-width: 400px;
+}
+
+ul.jsxc_vCard ul {
+	margin-left: 20px;
+}
+
+ul.jsxc_vCard li {
+	cursor: default !important;
+}
+
+ul.jsxc_vCard strong {
+	
+}
+
+img.jsxc_vCard {
+	float: right;
+	max-width: 200px;
+	max-height: 200px;
+	border: 5px solid white;
+	border-radius: 2px;
+}
+
+.jsxc_window .ui-resizable-w {
+	left: 0px;
+}
+
+.jsxc_window .ui-resizable-nw {
+   top: 0px;
+   left: 0px;
+   width: 15px;
+   height: 15px;
+   z-index: 95 !important;
+}
+
+.jsxc_window .ui-resizable-n {
+   position: absolute;
+   top: 0px;
+   left: 0px;
+   right: 0px;
+   height: 15px;
+   z-index: 100;
+   background: -moz-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+      rgba(40, 35, 35, 0) 100%); /* FF3.6+ */
+   background: -webkit-gradient(linear, left top, left bottom, color-stop(0%, rgba(164,
+      164, 164, 1)), color-stop(100%, rgba(40, 35, 35, 0)));
+   /* Chrome,Safari4+ */
+   background: -webkit-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+      rgba(40, 35, 35, 0) 100%); /* Chrome10+,Safari5.1+ */
+   background: -o-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+      rgba(40, 35, 35, 0) 100%); /* Opera 11.10+ */
+   background: -ms-linear-gradient(top, rgba(164, 164, 164, 1) 0%,
+      rgba(40, 35, 35, 0) 100%); /* IE10+ */
+   background: linear-gradient(to bottom, #a4a4a4 0%, rgba(40, 35, 35, 0)
+      100%); /* W3C */
+   filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#a4a4a4',
+      endColorstr='#00282323', GradientType=0); /* IE6-9 */
+}
+
+.jsxc_emoticon {
+  width: 19px;
+  height: 19px;
+  background-size: 19px 19px !important;
+  border: 0px;
+}
+
+.jsxc_angel { background: url('img/emotions/angel.png') }
+
+.jsxc_angry { background: url('img/emotions/angry.png') }
+
+.jsxc_smile { background: url('img/emotions/smile.png') }
+
+.jsxc_grin { background: url('img/emotions/grin.png') }
+
+.jsxc_sad { background: url('img/emotions/sad.png') }
+
+.jsxc_wink { background: url('img/emotions/wink.png') }
+
+.jsxc_tonguesmile { background: url('img/emotions/tonguesmile.png') }
+
+.jsxc_surprised { background: url('img/emotions/surprised.png') }
+
+.jsxc_kiss { background: url('img/emotions/kiss.png') }
+
+.jsxc_sunglassess { background: url('img/emotions/sunglassess.png') }
+
+.jsxc_crysad { background: url('img/emotions/crysad.png') }
+
+.jsxc_doubt { background: url('img/emotions/doubt.png') }
+
+.jsxc_zip { background: url('img/emotions/zip.png') }
+
+.jsxc_thumbsup { background: url('img/emotions/thumbsup.png') }
+
+.jsxc_thumbsdown { background: url('img/emotions/thumbsdown.png') }
+
+.jsxc_beer { background: url('img/emotions/beer.png') }
+
+.jsxc_devil { background: url('img/emotions/devil.png') }
+
+.jsxc_kissing { background: url('img/emotions/kissing.png') }
+
+.jsxc_rose { background: url('img/emotions/rose.png') }
+
+.jsxc_music { background: url('img/emotions/music.png') }
+
+.jsxc_love { background: url('img/emotions/love.png') }
+
+.jsxc_tired { background: url('img/emotions/tired.png') }
diff --git a/jsxc.js b/jsxc.js
new file mode 100644
index 0000000..084c893
--- /dev/null
+++ b/jsxc.js
@@ -0,0 +1,6925 @@
+/*! This file is concatenated for the browser. */
+
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org> <br>
+ * Released under the MIT license
+ * 
+ * Please see http://www.jsxc.org/
+ * 
+ * @author Klaus Herberth <klaus at jsxc.org>
+ * @version 1.0.0
+ * @license MIT
+ */
+
+var jsxc;
+
+(function($) {
+   "use strict";
+
+   /**
+    * JavaScript Xmpp Chat namespace
+    * 
+    * @namespace jsxc
+    */
+   jsxc = {
+      /** Version of jsxc */
+      version: '1.0.0',
+
+      /** True if i'm the master */
+      master: false,
+
+      /** True if the role allocation is finished */
+      role_allocation: false,
+
+      /** Timeout for keepalive */
+      to: null,
+
+      /** Timeout after normal keepalive starts */
+      toBusy: null,
+
+      /** Timeout for notification */
+      toNotification: null,
+
+      /** Timeout delay for notification */
+      toNotificationDelay: 500,
+
+      /** Interval for keep-alive */
+      keepalive: null,
+
+      /** True if last activity was 10 min ago */
+      restore: false,
+
+      /** True if restore is complete */
+      restoreCompleted: false,
+
+      /** True if login through form */
+      triggeredFromForm: false,
+
+      /** True if login through box */
+      triggeredFromBox: false,
+
+      /** True if logout through element click */
+      triggeredFromElement: false,
+
+      /** True if logout through logout click */
+      triggeredFromLogout: false,
+
+      /** last values which we wrote into localstorage (IE workaround) */
+      ls: [],
+
+      /**
+       * storage event is even fired if I write something into storage (IE
+       * workaround) 0: conform, 1: not conform, 2: not shure
+       */
+      storageNotConform: null,
+
+      /** Timeout for storageNotConform test */
+      toSNC: null,
+
+      /** My bar id */
+      bid: null,
+
+      /** Some constants */
+      CONST: {
+         NOTIFICATION_DEFAULT: 'default',
+         NOTIFICATION_GRANTED: 'granted',
+         NOTIFICATION_DENIED: 'denied',
+         STATUS: [ 'offline', 'dnd', 'xa', 'away', 'chat', 'online' ],
+         SOUNDS: {
+            MSG: 'incomingMessage.wav',
+            CALL: 'Rotary-Phone6.mp3',
+            NOTICE: 'Ping1.mp3'
+         },
+         REGEX: {
+            JID: new RegExp('\\b[^"&\'\\/:<>@\\s]+@[\\w-_.]+\\b', 'ig'),
+            URL: new RegExp(/((?:https?:\/\/|www\.|([\w\-]+\.[a-zA-Z]{2,3})(?=\b))(?:(?:[\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*\([\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*\)([\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*[\-A-Za-z0-9+&@#\/%=~_|])?)|(?:[\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*[\-A-Za-z0-9+&@#\/%=~_|]))?)/gi)
+         }
+      },
+
+      /**
+       * Write debug message to console and to log.
+       * 
+       * @memberOf jsxc
+       * @param {String} msg Debug message
+       * @param {Object} data
+       * @param {String} Could be warn|error|null
+       */
+      debug: function(msg, data, level) {
+         if (level) {
+            msg = '[' + level + '] ' + msg;
+         }
+
+         if (data) {
+            if (jsxc.storage.getItem('debug') === true) {
+               console.log(msg, data);
+            }
+
+            // try to convert data to string
+            var d;
+            try {
+               // clone html snippet
+               d = $("<span>").prepend($(data).clone()).html();
+            } catch (err) {
+               try {
+                  d = JSON.stringify(data);
+               } catch (err2) {
+                  d = 'see js console';
+               }
+            }
+
+            jsxc.log = jsxc.log + msg + ': ' + d + '\n';
+         } else {
+            console.log(msg);
+            jsxc.log = jsxc.log + msg + '\n';
+         }
+      },
+
+      /**
+       * Write warn message.
+       * 
+       * @memberOf jsxc
+       * @param {String} msg Warn message
+       * @param {Object} data
+       */
+      warn: function(msg, data) {
+         jsxc.debug(msg, data, 'WARN');
+      },
+
+      /**
+       * Write error message.
+       * 
+       * @memberOf jsxc
+       * @param {String} msg Error message
+       * @param {Object} data
+       */
+      error: function(msg, data) {
+         jsxc.debug(msg, data, 'ERROR');
+      },
+
+      /** debug log */
+      log: '',
+
+      /**
+       * Starts the action
+       * 
+       * @memberOf jsxc
+       * @param {object} options
+       */
+      init: function(options) {
+
+         if (options) {
+            // override default options
+            $.extend(jsxc.options, options);
+         }
+
+         /**
+          * Getter method for options. Saved options will override default one.
+          * 
+          * @param {string} key option key
+          * @returns default or saved option value
+          */
+         jsxc.options.get = function(key) {
+            var local = jsxc.storage.getUserItem('options') || {};
+
+            return local[key] || jsxc.options[key];
+         };
+
+         /**
+          * Setter method for options. Will write into localstorage.
+          * 
+          * @param {string} key option key
+          * @param {object} value option value
+          */
+         jsxc.options.set = function(key, value) {
+            jsxc.storage.updateItem('options', key, value, true);
+         };
+
+         jsxc.storageNotConform = jsxc.storage.getItem('storageNotConform');
+         if (jsxc.storageNotConform === null) {
+            jsxc.storageNotConform = 2;
+         }
+
+         // detect language
+         var lang;
+         if (jsxc.storage.getItem('lang') !== null) {
+            lang = jsxc.storage.getItem('lang');
+         } else if (jsxc.options.autoLang && navigator.language) {
+            lang = navigator.language.substr(0, 2);
+         } else {
+            lang = jsxc.options.defaultLang;
+         }
+
+         // set language
+         jsxc.l = jsxc.l10n.en;
+         $.extend(jsxc.l, jsxc.l10n[lang]);
+
+         // Check localStorage
+         if (typeof (localStorage) === 'undefined') {
+            jsxc.debug("Browser doesn't support localStorage.");
+            return;
+         }
+
+         // Register event listener for the storage event
+         window.addEventListener('storage', jsxc.storage.onStorage, false);
+
+         var lastActivity = jsxc.storage.getItem('lastActivity') || 0;
+
+         if ((new Date()).getTime() - lastActivity < jsxc.options.loginTimeout) {
+            jsxc.restore = true;
+         }
+
+         // Check if we have to establish a new connection
+         if (!jsxc.storage.getItem('rid') || !jsxc.storage.getItem('sid') || !jsxc.restore) {
+
+            // Looking for a login form
+            if (!jsxc.options.loginForm.form || !(jsxc.el_exists(jsxc.options.loginForm.form) && jsxc.el_exists(jsxc.options.loginForm.jid) && jsxc.el_exists(jsxc.options.loginForm.pass))) {
+
+               if (jsxc.options.displayRosterMinimized()) {
+                  // Show minimized roster
+                  jsxc.storage.setUserItem('roster', 'hidden');
+                  jsxc.gui.roster.init();
+                  jsxc.gui.roster.noConnection();
+               }
+
+               return;
+            }
+
+            if (typeof jsxc.options.formFound === 'function') {
+               jsxc.options.formFound.call();
+            }
+
+            // create jquery object
+            var form = jsxc.options.loginForm.form = $(jsxc.options.loginForm.form);
+            var events = form.data('events') || {
+               submit: []
+            };
+            var submits = [];
+
+            // save attached submit events and remove them. Will be reattached
+            // in jsxc.submitLoginForm
+            $.each(events.submit, function(index, val) {
+               submits.push(val.handler);
+            });
+
+            form.data('submits', submits);
+            form.off('submit');
+
+            // Add jsxc login action to form
+            form.submit(function() {
+
+               var settings = jsxc.prepareLogin();
+
+               if (settings !== false && (settings.xmpp.onlogin === "true" || settings.xmpp.onlogin === true)) {
+                  jsxc.triggeredFromForm = true;
+
+                  jsxc.xmpp.login();
+
+                  // Trigger submit in jsxc.xmpp.connected()
+                  return false;
+               }
+
+               return true;
+            });
+
+         } else {
+
+            // Restore old connection
+
+            jsxc.bid = jsxc.jidToBid(jsxc.storage.getItem('jid'));
+
+            jsxc.gui.init();
+
+            // Looking for logout element
+            if (jsxc.options.logoutElement !== null && jsxc.options.logoutElement.length > 0) {
+               jsxc.options.logoutElement.one('click', function() {
+                  jsxc.options.logoutElement = $(this);
+                  jsxc.triggeredFromLogout = true;
+                  return jsxc.xmpp.logout();
+               });
+            }
+
+            if (typeof (jsxc.storage.getItem('alive')) === 'undefined' || !jsxc.restore) {
+               jsxc.onMaster();
+            } else {
+               jsxc.checkMaster();
+            }
+         }
+      },
+
+      /**
+       * Load settings and prepare jid.
+       * 
+       * @memberOf jsxc
+       * @returns Loaded settings
+       */
+      prepareLogin: function() {
+         var username = $(jsxc.options.loginForm.jid).val();
+         var password = $(jsxc.options.loginForm.pass).val();
+
+         if (typeof jsxc.options.loadSettings !== 'function') {
+            jsxc.error('No loadSettings function given. Abort.');
+            return;
+         }
+
+         jsxc.gui.showWaitAlert(jsxc.l.Logging_in);
+
+         var settings = jsxc.options.loadSettings.call(this, username, password);
+
+         if (settings === false || settings === null || typeof settings === 'undefined') {
+            jsxc.warn('No settings provided');
+
+            return false;
+         }
+
+         if (typeof settings.xmpp.username === 'string') {
+            username = settings.xmpp.username;
+         }
+
+         var resource = (settings.xmpp.resource) ? '/' + settings.xmpp.resource : '';
+         var domain = settings.xmpp.domain;
+         var jid;
+
+         if (username.match(/@(.*)$/)) {
+            jid = (username.match(/\/(.*)$/)) ? username : username + resource;
+         } else {
+            jid = username + '@' + domain + resource;
+         }
+
+         if (typeof jsxc.options.loginForm.preJid === 'function') {
+            jid = jsxc.options.loginForm.preJid(jid);
+         }
+
+         jsxc.bid = jsxc.jidToBid(jid);
+
+         settings.xmpp.username = jid.split('@')[0];
+         settings.xmpp.domain = jid.split('@')[1].split('/')[0];
+         settings.xmpp.resource = jid.split('@')[1].split('/')[1] || "";
+
+         $.each(settings, function(key, val) {
+            jsxc.options.set(key, val);
+         });
+
+         jsxc.options.xmpp.jid = jid;
+         jsxc.options.xmpp.password = password;
+
+         return settings;
+      },
+
+      /**
+       * Called if the script is a slave
+       */
+      onSlave: function() {
+         jsxc.debug('I am the slave.');
+
+         jsxc.role_allocation = true;
+
+         jsxc.restoreRoster();
+         jsxc.restoreWindows();
+         jsxc.restoreCompleted = true;
+
+         $(document).trigger('restoreCompleted.jsxc');
+      },
+
+      /**
+       * Called if the script is the master
+       */
+      onMaster: function() {
+         jsxc.debug('I am master.');
+
+         jsxc.master = true;
+
+         // Init local storage
+         jsxc.storage.setItem('alive', 0);
+         jsxc.storage.setItem('alive_busy', 0);
+         if (!jsxc.storage.getUserItem('windowlist')) {
+            jsxc.storage.setUserItem('windowlist', []);
+         }
+
+         // Sending keepalive signal
+         jsxc.startKeepAlive();
+
+         // create or load DSA key and call _onMaster
+         jsxc.otr.createDSA();
+      },
+
+      /**
+       * Second half of the onMaster routine
+       */
+      _onMaster: function() {
+
+         // create otr objects, if we lost the master
+         if (jsxc.role_allocation) {
+            $.each(jsxc.storage.getUserItem('windowlist'), function(index, val) {
+               jsxc.otr.create(val);
+            });
+         }
+
+         jsxc.role_allocation = true;
+
+         if (jsxc.restore && !jsxc.restoreCompleted) {
+            jsxc.restoreRoster();
+            jsxc.restoreWindows();
+            jsxc.restoreCompleted = true;
+
+            $(document).trigger('restoreCompleted.jsxc');
+         }
+
+         // Prepare notifications
+         if (jsxc.restore) {
+            var noti = jsxc.storage.getUserItem('notification') || 2;
+            if (jsxc.options.notification && noti > 0 && jsxc.notification.hasSupport()) {
+               if (jsxc.notification.hasPermission()) {
+                  jsxc.notification.init();
+               } else {
+                  jsxc.notification.prepareRequest();
+               }
+            } else {
+               // No support => disable
+               jsxc.options.notification = false;
+            }
+         }
+
+         jsxc.xmpp.login();
+      },
+
+      /**
+       * Checks if there is a master
+       */
+      checkMaster: function() {
+         jsxc.debug('check master');
+
+         jsxc.to = window.setTimeout(jsxc.onMaster, 1000);
+         jsxc.storage.ink('alive');
+      },
+
+      /**
+       * Start sending keep-alive signal
+       */
+      startKeepAlive: function() {
+         jsxc.keepalive = window.setInterval(jsxc.keepAlive, jsxc.options.timeout - 1000);
+      },
+
+      /**
+       * Sends the keep-alive signal to signal that the master is still there.
+       */
+      keepAlive: function() {
+         jsxc.storage.ink('alive');
+
+         if (jsxc.role_allocation) {
+            jsxc.storage.setItem('lastActivity', (new Date()).getTime());
+         }
+      },
+
+      /**
+       * Send one keep-alive signal with higher timeout, and than resume with
+       * normal signal
+       */
+      keepBusyAlive: function() {
+         if (jsxc.toBusy) {
+            window.clearTimeout(jsxc.toBusy);
+         }
+
+         if (jsxc.keepalive) {
+            window.clearInterval(jsxc.keepalive);
+         }
+
+         jsxc.storage.ink('alive_busy');
+         jsxc.toBusy = window.setTimeout(jsxc.startKeepAlive, jsxc.options.busyTimeout - 1000);
+      },
+
+      /**
+       * Generates a random integer number between 0 and max
+       * 
+       * @param {Integer} max
+       * @return {Integer} random integer between 0 and max
+       */
+      random: function(max) {
+         return Math.floor(Math.random() * max);
+      },
+
+      /**
+       * Checks if there is a element with the given selector
+       * 
+       * @param {String} selector jQuery selector
+       * @return {Boolean}
+       */
+      el_exists: function(selector) {
+         return $(selector).length > 0;
+      },
+
+      /**
+       * Creates a CSS compatible string from a JID
+       * 
+       * @param {type} jid Valid Jabber ID
+       * @returns {String} css Compatible string
+       */
+      jidToCid: function(jid) {
+         jsxc.warn('jsxc.jidToCid is deprecated!');
+
+         var cid = Strophe.getBareJidFromJid(jid).replace('@', '-').replace(/\./g, '-').toLowerCase();
+
+         return cid;
+      },
+
+      /**
+       * Create comparable bar jid.
+       * 
+       * @memberOf jsxc
+       * @param jid
+       * @returns comparable bar jid
+       */
+      jidToBid: function(jid) {
+         return Strophe.getBareJidFromJid(jid).toLowerCase();
+      },
+
+      /**
+       * Restore roster
+       */
+      restoreRoster: function() {
+         var buddies = jsxc.storage.getUserItem('buddylist');
+
+         if (!buddies || buddies.length === 0) {
+            jsxc.debug('No saved buddylist.');
+
+            jsxc.gui.roster.empty();
+
+            return;
+         }
+
+         $.each(buddies, function(index, value) {
+            jsxc.gui.roster.add(value);
+         });
+
+         $(document).trigger('cloaded.roster.jsxc');
+      },
+
+      /**
+       * Restore all windows
+       */
+      restoreWindows: function() {
+         var windows = jsxc.storage.getUserItem('windowlist');
+
+         if (windows === null) {
+            return;
+         }
+
+         $.each(windows, function(index, bid) {
+            var window = jsxc.storage.getUserItem('window', bid);
+
+            if (!window) {
+               jsxc.debug('Associated window-element is missing: ' + bid);
+               return true;
+            }
+
+            jsxc.gui.window.init(bid);
+
+            if (!window.minimize) {
+               jsxc.gui.window.show(bid);
+            } else {
+               jsxc.gui.window.hide(bid);
+            }
+
+            jsxc.gui.window.setText(bid, window.text);
+         });
+      },
+
+      /**
+       * This method submits the specified login form.
+       */
+      submitLoginForm: function() {
+         var form = jsxc.options.loginForm.form.off('submit');
+
+         // Attach original events
+         var submits = form.data('submits') || [];
+         $.each(submits, function(index, val) {
+            form.submit(val);
+         });
+
+         if (form.find('#submit').length > 0) {
+            form.find('#submit').click();
+         } else {
+            form.submit();
+         }
+      },
+
+      /**
+       * Escapes some characters to HTML character
+       */
+      escapeHTML: function(text) {
+         text = text.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');
+         return text.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');
+      },
+
+      /**
+       * Removes all html tags.
+       * 
+       * @memberOf jsxc
+       * @param text
+       * @returns stripped text
+       */
+      removeHTML: function(text) {
+         return $('<span>').html(text).text();
+      },
+
+      /**
+       * Executes only one of the given events
+       * 
+       * @param {string} obj.key event name
+       * @param {function} obj.value function to execute
+       * @returns {string} namespace of all events
+       */
+      switchEvents: function(obj) {
+         var ns = Math.random().toString(36).substr(2, 12);
+         var self = this;
+
+         $.each(obj, function(key, val) {
+            $(document).one(key + '.' + ns, function() {
+               $(document).off('.' + ns);
+
+               val.apply(self, arguments);
+            });
+         });
+
+         return ns;
+      },
+
+      /**
+       * Checks if tab is hidden.
+       * 
+       * @returns {boolean} True if tab is hidden
+       */
+      isHidden: function() {
+         var hidden = false;
+
+         if (typeof document.hidden !== 'undefined') {
+            hidden = document.hidden;
+         } else if (typeof document.webkitHidden !== 'undefined') {
+            hidden = document.webkitHidden;
+         } else if (typeof document.mozHidden !== 'undefined') {
+            hidden = document.mozHidden;
+         } else if (typeof document.msHidden !== 'undefined') {
+            hidden = document.msHidden;
+         }
+
+         // handle multiple tabs
+         if (hidden && jsxc.master) {
+            jsxc.storage.ink('hidden', 0);
+         } else if (!hidden && !jsxc.master) {
+            jsxc.storage.ink('hidden');
+         }
+
+         return hidden;
+      },
+
+      /**
+       * Replace %%tokens%% with correct translation.
+       * 
+       * @param {String} text Given text
+       * @returns {String} Translated string
+       */
+      translate: function(text) {
+         return text.replace(/%%([a-zA-Z0-9_-}{ .!,?/'@]+)%%/g, function(s, key) {
+            var k = key.replace(/ /gi, '_').replace(/[.!,?/'@]/g, '');
+
+            if (!jsxc.l[k]) {
+               jsxc.warn('No translation for: ' + k);
+            }
+
+            return jsxc.l[k] || key.replace(/_/g, ' ');
+         });
+      },
+   };
+
+   /**
+    * Set some options for the chat.
+    * 
+    * @namespace jsxc.options
+    */
+   jsxc.options = {
+
+      /** name of container application (e.g. owncloud or SOGo) */
+      app_name: 'web applications',
+
+      /** Timeout for the keepalive signal */
+      timeout: 3000,
+
+      /** Timeout for the keepalive signal if the master is busy */
+      busyTimeout: 15000,
+
+      /** OTR options */
+      otr: {
+         ERROR_START_AKE: true
+      },
+
+      /** xmpp options */
+      xmpp: {
+         url: null,
+         jid: null,
+         domain: null,
+         password: null,
+         overwrite: false,
+         onlogin: true
+      },
+
+      /** default xmpp priorities */
+      priority: {
+         online: 0,
+         chat: 0,
+         away: 0,
+         xa: 0,
+         dnd: 0
+      },
+
+      /** If all 3 properties are set, the login form is used */
+      loginForm: {
+         /** jquery object from form */
+         form: null,
+
+         /** jquery object from input element which contains the jid */
+         jid: null,
+
+         /** jquery object from input element which contains the password */
+         pass: null,
+
+         /** manipulate JID from input element */
+         preJid: function(jid) {
+            return jid;
+         }
+      },
+
+      /** jquery object from logout element */
+      logoutElement: null,
+
+      /** How many messages should be logged? */
+      numberOfMsg: 10,
+
+      /** Default language */
+      defaultLang: 'en',
+
+      /** auto language detection */
+      autoLang: true,
+
+      /** Place for roster */
+      rosterAppend: 'body',
+
+      /** Should we use the HTML5 notification API? */
+      notification: true,
+
+      /** duration for notification */
+      popupDuration: 6000,
+
+      /** Absolute path root of JSXC installation */
+      root: '',
+
+      /** Timeout for restore in ms */
+      loginTimeout: 1000 * 60 * 10,
+
+      /**
+       * This function decides wether the roster will be displayed or not if no
+       * connection is found.
+       */
+      displayRosterMinimized: function() {
+         return false;
+      },
+
+      /** Set to true if you want to hide offline buddies. */
+      hideOffline: false,
+
+      /** Mute notification sound? */
+      muteNotification: false,
+
+      /**
+       * If no avatar is found, this function is called.
+       * 
+       * @param jid Jid of that user.
+       * @this {jQuery} Elements to update with probable .jsxc_avatar elements
+       */
+      defaultAvatar: function() {
+
+      },
+
+      /**
+       * Returns permanent saved settings and overwrite default jsxc.options.
+       * 
+       * @memberOf jsxc.options
+       * @param username String username
+       * @param password String password
+       * @returns {object} at least xmpp.url
+       */
+      loadSettings: function() {
+
+      },
+
+      /**
+       * Call this function to save user settings permanent.
+       * 
+       * @memberOf jsxc.options
+       * @param data Holds all data as key/value
+       */
+      saveSettinsPermanent: function() {
+
+      }
+   };
+
+   /**
+    * Handle functions for chat window's and buddylist
+    * 
+    * @namespace jsxc.gui
+    */
+   jsxc.gui = {
+      /** Smilie token to file mapping */
+      emotions: [ [ 'O:-) O:)', 'angel' ], [ '>:-( >:( >:-( >:(', 'angry' ], [ ':-) :)', 'smile' ], [ ':-D :D', 'grin' ], [ ':-( :(', 'sad' ], [ ';-) ;)', 'wink' ], [ ':-P :P', 'tonguesmile' ], [ '=-O', 'surprised' ], [ ':kiss: :-*', 'kiss' ], [ '8-) :cool:', 'sunglassess' ], [ ':\'-( :\'( :&apos;-(', 'crysad' ], [ ':-/', 'doubt' ], [ ':-X :X', 'zip' ], [ ':yes:', 'thumbsup' ], [ ':no:', 'thumbsdown' ], [ ':beer:', 'beer' ], [ ':devil:', 'devil' ], [ ':kiss: :kissing:', 'kissin [...]
+
+      /**
+       * Different uri query actions as defined in XEP-0147.
+       * 
+       * @namespace jsxc.gui.queryActions
+       */
+      queryActions: {
+         /** xmpp:JID?message[;body=TEXT] */
+         message: function(jid, params) {
+            var win = jsxc.gui.window.open(jsxc.jidToBid(jid));
+
+            if (params && typeof params.body === 'string') {
+               win.find('.jsxc_textinput').val(params.body);
+            }
+         },
+
+         /** xmpp:JID?remove */
+         remove: function(jid) {
+            jsxc.gui.showRemoveDialog(jsxc.jidToBid(jid));
+         },
+
+         /** xmpp:JID?subscribe[;name=NAME] */
+         subscribe: function(jid, params) {
+            jsxc.gui.showContactDialog(jid);
+
+            if (params && typeof params.name) {
+               $('#jsxc_alias').val(params.name);
+            }
+         },
+
+         /** xmpp:JID?vcard */
+         vcard: function(jid) {
+            jsxc.gui.showVcard(jid);
+         }
+      },
+
+      /**
+       * Creates application skeleton.
+       * 
+       * @memberOf jsxc.gui
+       */
+      init: function() {
+         $('body').append($(jsxc.gui.template.get('windowList')));
+
+         $(window).resize(jsxc.gui.updateWindowListSB);
+         $('#jsxc_windowList').resize(jsxc.gui.updateWindowListSB);
+
+         $('#jsxc_windowListSB .jsxc_scrollLeft').click(function() {
+            jsxc.gui.scrollWindowListBy(-200);
+         });
+         $('#jsxc_windowListSB .jsxc_scrollRight').click(function() {
+            jsxc.gui.scrollWindowListBy(200);
+         });
+         $('#jsxc_windowList').on('wheel', function(ev) {
+            if ($('#jsxc_windowList').data('isOver')) {
+               jsxc.gui.scrollWindowListBy((ev.originalEvent.wheelDelta > 0) ? 200 : -200);
+            }
+         });
+
+         jsxc.gui.tooltip('#jsxc_windowList');
+
+         jsxc.gui.roster.init();
+
+         // prepare regexp for emotions
+         $.each(jsxc.gui.emotions, function(i, val) {
+            // escape characters
+            var reg = val[0].replace(/(\/|\||\*|\.|\+|\?|\^|\$|\(|\)|\[|\]|\{|\})/g, '\\$1');
+            reg = '(' + reg.split(' ').join('|') + ')';
+            jsxc.gui.emotions[i][2] = new RegExp(reg, 'g');
+         });
+
+         // We need this often, so we creates some template jquery objects
+         jsxc.gui.windowTemplate = $(jsxc.gui.template.get('chatWindow'));
+         jsxc.gui.buddyTemplate = $(jsxc.gui.template.get('rosterBuddy'));
+      },
+
+      /**
+       * Init tooltip plugin for given jQuery selector.
+       * 
+       * @param {String} selector jQuery selector
+       * @memberOf jsxc.gui
+       */
+      tooltip: function(selector) {
+         $(selector).tooltip({
+            show: {
+               delay: 600
+            },
+            content: function() {
+               return $(this).attr('title').replace(/\n/g, '<br />');
+            }
+         });
+      },
+
+      /**
+       * Updates Information in roster and chatbar
+       * 
+       * @param {String} bid bar jid
+       */
+      update: function(bid) {
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         if (!data) {
+            jsxc.debug('No data for ' + bid);
+            return;
+         }
+
+         var ri = jsxc.gui.roster.getItem(bid); // roster item from user
+         var we = jsxc.gui.window.get(bid); // window element from user
+         var ue = ri.add(we); // both
+         var spot = $('.jsxc_spot[data-bid="' + bid + '"]');
+
+         // Attach data to corresponding roster item
+         ri.data(data);
+
+         // Add online status
+         ue.add(spot).removeClass('jsxc_' + jsxc.CONST.STATUS.join(' jsxc_')).addClass('jsxc_' + jsxc.CONST.STATUS[data.status]);
+
+         // Change name and add title
+         ue.find('.jsxc_name').add(spot).text(data.name).attr('title', jsxc.l.is + ' ' + jsxc.CONST.STATUS[data.status]);
+
+         // Update gui according to encryption state
+         switch (data.msgstate) {
+            case 0:
+               we.find('.jsxc_transfer').removeClass('jsxc_enc jsxc_fin').attr('title', jsxc.l.your_connection_is_unencrypted);
+               we.find('.jsxc_settings .jsxc_verification').addClass('jsxc_disabled');
+               we.find('.jsxc_settings .jsxc_transfer').text(jsxc.l.start_private);
+               break;
+            case 1:
+               we.find('.jsxc_transfer').addClass('jsxc_enc').attr('title', jsxc.l.your_connection_is_encrypted);
+               we.find('.jsxc_settings .jsxc_verification').removeClass('jsxc_disabled');
+               we.find('.jsxc_settings .jsxc_transfer').text(jsxc.l.close_private);
+               break;
+            case 2:
+               we.find('.jsxc_settings .jsxc_verification').addClass('jsxc_disabled');
+               we.find('.jsxc_transfer').removeClass('jsxc_enc').addClass('jsxc_fin').attr('title', jsxc.l.your_buddy_closed_the_private_connection);
+               we.find('.jsxc_settings .jsxc_transfer').text(jsxc.l.close_private);
+               break;
+         }
+
+         // update gui according to verification state
+         if (data.trust) {
+            we.find('.jsxc_transfer').addClass('jsxc_trust').attr('title', jsxc.l.your_buddy_is_verificated);
+         } else {
+            we.find('.jsxc_transfer').removeClass('jsxc_trust');
+         }
+
+         // update gui according to subscription state
+         if (data.sub && data.sub !== 'both') {
+            ue.addClass('jsxc_oneway');
+         } else {
+            ue.removeClass('jsxc_oneway');
+         }
+
+         var info = '<b>' + Strophe.getBareJidFromJid(data.jid) + '</b>\n';
+         info += jsxc.translate('%%Subscription%%: %%' + data.sub + '%%\n');
+         info += jsxc.translate('%%Status%%: %%' + jsxc.CONST.STATUS[data.status] + '%%');
+
+         ri.find('.jsxc_name').attr('title', info);
+
+         if (data.avatar && data.avatar.length > 0) {
+            jsxc.gui.updateAvatar(ue, data.jid, data.avatar);
+         } else {
+            jsxc.options.defaultAvatar.call(ue, data.jid);
+         }
+      },
+
+      /**
+       * Update avatar on all given elements.
+       * 
+       * @memberOf jsxc.gui
+       * @param {jQuery} el Elements with subelement .jsxc_avatar
+       * @param {string} jid Jid
+       * @param {string} aid Avatar id (sha1 hash of image)
+       */
+      updateAvatar: function(el, jid, aid) {
+
+         if (typeof aid === 'undefined') {
+            if (typeof jsxc.options.defaultAvatar === 'function') {
+               jsxc.options.defaultAvatar.call(el, jid);
+            }
+            return;
+         }
+
+         var avatarSrc = jsxc.storage.getUserItem('avatar', aid);
+
+         var setAvatar = function(src) {
+            if (src === 0) {
+               jsxc.options.defaultAvatar.call(el, jid);
+               return;
+            }
+
+            el.find('.jsxc_avatar').removeAttr('style');
+
+            el.find('.jsxc_avatar').css({
+               'background-image': 'url(' + src + ')',
+               'text-indent': '999px'
+            });
+         };
+
+         if (avatarSrc !== null) {
+            setAvatar(avatarSrc);
+         } else {
+            jsxc.xmpp.conn.vcard.get(function(stanza) {
+               jsxc.debug('vCard', stanza);
+
+               var vCard = $(stanza).find("vCard > PHOTO");
+               var src;
+
+               if (vCard.length === 0) {
+                  jsxc.debug('No photo provided');
+                  src = 0;
+               } else {
+                  var img = vCard.find('BINVAL').text();
+                  var type = vCard.find('TYPE').text();
+                  src = 'data:' + type + ';base64,' + img;
+               }
+
+               jsxc.storage.setUserItem('avatar', aid, src);
+               setAvatar(src);
+            }, Strophe.getBareJidFromJid(jid), function(msg) {
+               jsxc.warn('Could not load vcard.', msg);
+
+               jsxc.storage.setUserItem('avatar', aid, 0);
+               setAvatar(0);
+            });
+         }
+      },
+
+      /**
+       * Updates scrollbar handlers.
+       * 
+       * @memberOf jsxc.gui
+       */
+      updateWindowListSB: function() {
+
+         if ($('#jsxc_windowList>ul').width() > $('#jsxc_windowList').width()) {
+            $('#jsxc_windowListSB > div').removeClass('jsxc_disabled');
+         } else {
+            $('#jsxc_windowListSB > div').addClass('jsxc_disabled');
+            $('#jsxc_windowList>ul').css('right', '0px');
+         }
+      },
+
+      /**
+       * Scroll window list by offset.
+       * 
+       * @memberOf jsxc.gui
+       * @param offset
+       */
+      scrollWindowListBy: function(offset) {
+
+         var scrollWidth = $('#jsxc_windowList>ul').width();
+         var width = $('#jsxc_windowList').width();
+         var el = $('#jsxc_windowList>ul');
+         var right = parseInt(el.css('right')) - offset;
+         var padding = $("#jsxc_windowListSB").width();
+
+         if (scrollWidth < width) {
+            return;
+         }
+
+         if (right > 0) {
+            right = 0;
+         }
+
+         if (right < width - scrollWidth - padding) {
+            right = width - scrollWidth - padding;
+         }
+
+         el.css('right', right + 'px');
+      },
+
+      /**
+       * Returns the window element
+       * 
+       * @param {String} bid
+       * @returns {jquery} jQuery object of the window element
+       */
+      getWindow: function(bid) {
+         jsxc.warn('jsxc.gui.getWindow is deprecated!');
+
+         return jsxc.gui.window.get(bid);
+      },
+
+      /**
+       * Toggle list with timeout, like menu or settings
+       * 
+       * @memberof jsxc.gui
+       */
+      toggleList: function() {
+         var self = $(this);
+
+         self.disableSelection();
+
+         var ul = self.find('ul');
+         var slideUp = null;
+
+         slideUp = function() {
+            ul.slideUp({
+               complete: function() {
+                  self.removeClass('jsxc_opened');
+               }
+            });
+
+            $('body').off('click', null, slideUp);
+         };
+
+         $(this).click(function() {
+
+            if (ul.is(":hidden")) {
+               // hide other lists
+               $('body').click();
+               $('body').one('click', slideUp);
+            } else {
+               $('body').off('click', null, slideUp);
+            }
+
+            ul.slideToggle();
+
+            window.clearTimeout(ul.data('timer'));
+
+            self.toggleClass('jsxc_opened');
+
+            return false;
+         }).mouseleave(function() {
+            ul.data('timer', window.setTimeout(slideUp, 2000));
+         }).mouseenter(function() {
+            window.clearTimeout(ul.data('timer'));
+         });
+      },
+
+      /**
+       * Creates and show loginbox
+       */
+      showLoginBox: function() {
+         // Set focus to password field
+         $(document).on("complete.dialog.jsxc", function() {
+            $('#jsxc_password').focus();
+         });
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('loginBox'));
+
+         $('#jsxc_dialog').find('form').submit(function() {
+
+            $(this).find('input[type=submit]').prop('disabled', true);
+
+            jsxc.options.loginForm.form = $(this);
+            jsxc.options.loginForm.jid = $(this).find('#jsxc_username');
+            jsxc.options.loginForm.pass = $(this).find('#jsxc_password');
+
+            var settings = jsxc.prepareLogin();
+
+            jsxc.triggeredFromBox = true;
+            jsxc.triggeredFromForm = false;
+
+            if (settings === false) {
+               jsxc.gui.showAuthFail();
+            } else {
+               jsxc.xmpp.login();
+            }
+
+            return false;
+         });
+      },
+
+      /**
+       * Creates and show the fingerprint dialog
+       * 
+       * @param {String} bid
+       */
+      showFingerprints: function(bid) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('fingerprintsDialog', bid));
+      },
+
+      /**
+       * Creates and show the verification dialog
+       * 
+       * @param {String} bid
+       */
+      showVerification: function(bid) {
+
+         // Check if there is a open dialog
+         if ($('#jsxc_dialog').length > 0) {
+            setTimeout(function() {
+               jsxc.gui.showVerification(bid);
+            }, 3000);
+            return;
+         }
+
+         // verification only possible if the connection is encrypted
+         if (jsxc.storage.getUserItem('buddy', bid).msgstate !== OTR.CONST.MSGSTATE_ENCRYPTED) {
+            jsxc.warn('Connection not encrypted');
+            return;
+         }
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('authenticationDialog', bid));
+
+         // Add handler
+
+         $('#jsxc_dialog > div:gt(0)').hide();
+         $('#jsxc_dialog select').change(function() {
+            $('#jsxc_dialog > div:gt(0)').hide();
+            $('#jsxc_dialog > div:eq(' + $(this).prop('selectedIndex') + ')').slideDown({
+               complete: function() {
+                  jsxc.gui.dialog.resize();
+               }
+            });
+         });
+
+         // Manual
+         $('#jsxc_dialog > div:eq(1) a.creation').click(function() {
+            if (jsxc.master) {
+               jsxc.otr.objects[bid].trust = true;
+            }
+
+            jsxc.storage.updateUserItem('buddy', bid, 'trust', true);
+
+            jsxc.gui.dialog.close();
+
+            jsxc.storage.updateUserItem('buddy', bid, 'trust', true);
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.conversation_is_now_verified);
+            jsxc.gui.update(bid);
+         });
+
+         // Question
+         $('#jsxc_dialog > div:eq(2) a.creation').click(function() {
+            var div = $('#jsxc_dialog > div:eq(2)');
+            var sec = div.find('#jsxc_secret2').val();
+            var quest = div.find('#jsxc_quest').val();
+
+            if (sec === '' || quest === '') {
+               // Add information for the user which form is missing
+               div.find('input[value=""]').addClass('jsxc_invalid').keyup(function() {
+                  if ($(this).val().match(/.*/)) {
+                     $(this).removeClass('jsxc_invalid');
+                  }
+               });
+               return;
+            }
+
+            if (jsxc.master) {
+               jsxc.otr.sendSmpReq(bid, sec, quest);
+            } else {
+               jsxc.storage.setUserItem('smp_' + bid, {
+                  sec: sec,
+                  quest: quest
+               });
+            }
+
+            jsxc.gui.dialog.close();
+
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.authentication_query_sent);
+         });
+
+         // Secret
+         $('#jsxc_dialog > div:eq(3) .creation').click(function() {
+            var div = $('#jsxc_dialog > div:eq(3)');
+            var sec = div.find('#jsxc_secret').val();
+
+            if (sec === '') {
+               // Add information for the user which form is missing
+               div.find('#jsxc_secret').addClass('jsxc_invalid').keyup(function() {
+                  if ($(this).val().match(/.*/)) {
+                     $(this).removeClass('jsxc_invalid');
+                  }
+               });
+               return;
+            }
+
+            if (jsxc.master) {
+               jsxc.otr.sendSmpReq(bid, sec);
+            } else {
+               jsxc.storage.setUserItem('smp_' + bid, {
+                  sec: sec,
+                  quest: null
+               });
+            }
+
+            jsxc.gui.dialog.close();
+
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.authentication_query_sent);
+         });
+      },
+
+      /**
+       * Create and show approve dialog
+       * 
+       * @param {type} from valid jid
+       */
+      showApproveDialog: function(from) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('approveDialog'), {
+            'noClose': true
+         });
+
+         $('#jsxc_dialog .jsxc_their_jid').text(Strophe.getBareJidFromJid(from));
+
+         $('#jsxc_dialog .jsxc_deny').click(function(ev) {
+            ev.stopPropagation();
+
+            jsxc.xmpp.resFriendReq(from, false);
+
+            jsxc.gui.dialog.close();
+         });
+
+         $('#jsxc_dialog .jsxc_approve').click(function(ev) {
+            ev.stopPropagation();
+
+            var data = jsxc.storage.getUserItem('buddy', jsxc.jidToBid(from));
+
+            jsxc.xmpp.resFriendReq(from, true);
+
+            // If friendship is not mutual show contact dialog
+            if (!data || data.sub === 'from') {
+               $(document).one('close.dialog.jsxc', function() {
+                  jsxc.gui.showContactDialog(from);
+               });
+            }
+
+            jsxc.gui.dialog.close();
+         });
+      },
+
+      /**
+       * Create and show dialog to add a buddy
+       * 
+       * @param {string} [username] jabber id
+       */
+      showContactDialog: function(username) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('contactDialog'));
+
+         // If we got a friendship request, we would display the username in our
+         // response
+         if (username) {
+            $('#jsxc_username').val(username);
+         }
+
+         $('#jsxc_dialog form').submit(function() {
+            var username = $('#jsxc_username').val();
+            var alias = $('#jsxc_alias').val();
+
+            if (!username.match(/@(.*)$/)) {
+               username += '@' + Strophe.getDomainFromJid(jsxc.storage.getItem('jid'));
+            }
+
+            // Check if the username is valid
+            if (!username || !username.match(jsxc.CONST.REGEX.JID)) {
+               // Add notification
+               $('#jsxc_username').addClass('jsxc_invalid').keyup(function() {
+                  if ($(this).val().match(jsxc.CONST.REGEX.JID)) {
+                     $(this).removeClass('jsxc_invalid');
+                  }
+               });
+               return false;
+            }
+            jsxc.xmpp.addBuddy(username, alias);
+
+            jsxc.gui.dialog.close();
+
+            return false;
+         });
+      },
+
+      /**
+       * Create and show dialog to remove a buddy
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      showRemoveDialog: function(bid) {
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('removeDialog', bid));
+
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         $('#jsxc_dialog .creation').click(function(ev) {
+            ev.stopPropagation();
+
+            if (jsxc.master) {
+               jsxc.xmpp.removeBuddy(data.jid);
+            } else {
+               // inform master
+               jsxc.storage.setUserItem('deletebuddy', bid, {
+                  jid: data.jid
+               });
+            }
+
+            jsxc.gui.dialog.close();
+         });
+      },
+
+      /**
+       * Create and show a wait dialog
+       * 
+       * @param {type} msg message to display to the user
+       * @returns {undefined}
+       */
+      showWaitAlert: function(msg) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('waitAlert', null, msg), {
+            'noClose': true
+         });
+      },
+
+      /**
+       * Create and show a wait dialog
+       * 
+       * @param {type} msg message to display to the user
+       * @returns {undefined}
+       */
+      showAlert: function(msg) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('alert', null, msg));
+      },
+
+      /**
+       * Create and show a auth fail dialog
+       * 
+       * @returns {undefined}
+       */
+      showAuthFail: function() {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('authFailDialog'));
+
+         if (jsxc.triggeredFromBox) {
+            $('#jsxc_dialog .jsxc_cancel').hide();
+         }
+
+         $('#jsxc_dialog .creation').click(function() {
+            jsxc.gui.dialog.close();
+         });
+
+         $('#jsxc_dialog .jsxc_cancel').click(function() {
+            jsxc.submitLoginForm();
+         });
+      },
+
+      /**
+       * Create and show a confirm dialog
+       * 
+       * @param {String} msg Message
+       * @param {function} confirm
+       * @param {function} dismiss
+       * @returns {undefined}
+       */
+      showConfirmDialog: function(msg, confirm, dismiss) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('confirmDialog', null, msg), {
+            noClose: true
+         });
+
+         if (confirm) {
+            $('#jsxc_dialog .creation').click(confirm);
+         }
+
+         if (dismiss) {
+            $('#jsxc_dialog .jsxc_cancel').click(dismiss);
+         }
+      },
+
+      /**
+       * Show about dialog.
+       * 
+       * @memberOf jsxc.gui
+       */
+      showAboutDialog: function() {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('aboutDialog'));
+
+         $('#jsxc_dialog .jsxc_debuglog').click(function() {
+            jsxc.gui.showDebugLog();
+         });
+      },
+
+      /**
+       * Show debug log.
+       * 
+       * @memberOf jsxc.gui
+       */
+      showDebugLog: function() {
+         var userInfo = '<h3>User information</h3>';
+
+         if (navigator) {
+            var key;
+            for (key in navigator) {
+               if (navigator.hasOwnProperty(key) && typeof navigator[key] === 'string') {
+                  userInfo += '<b>' + key + ':</b> ' + navigator[key] + '<br />';
+               }
+            }
+         }
+
+         if (window.screen) {
+            userInfo += '<b>Height:</b> ' + window.screen.height + '<br />';
+            userInfo += '<b>Width:</b> ' + window.screen.width + '<br />';
+         }
+
+         userInfo += '<b>jsxc version:</b> ' + jsxc.version + '<br />';
+
+         jsxc.gui.dialog.open('<div class="jsxc_log">' + userInfo + '<h3>Log</h3><pre>' + jsxc.escapeHTML(jsxc.log) + '</pre></div>');
+      },
+
+      /**
+       * Show vCard of user with the given bar jid.
+       * 
+       * @memberOf jsxc.gui
+       * @param {String} jid
+       */
+      showVcard: function(jid) {
+         var bid = jsxc.jidToBid(jid);
+         jsxc.gui.dialog.open(jsxc.gui.template.get('vCard', bid));
+
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         if (data) {
+            // Display resources and corresponding information
+            var i, j, res, identities, identity = null, cap, client;
+            for (i = 0; i < data.res.length; i++) {
+               res = data.res[i];
+
+               identities = [];
+               cap = jsxc.xmpp.getCapabilitiesByJid(bid + '/' + res);
+
+               if (cap !== null && cap.identities !== null) {
+                  identities = cap.identities;
+               }
+
+               client = '';
+               for (j = 0; j < identities.length; j++) {
+                  identity = identities[j];
+                  if (identity.category === 'client') {
+                     if (client !== '') {
+                        client += ',\n';
+                     }
+
+                     client += identity.name + ' (' + identity.type + ')';
+                  }
+               }
+
+               var status = jsxc.storage.getUserItem('res', bid)[res];
+
+               $('#jsxc_dialog ul.jsxc_vCard').append('<li class="jsxc_sep"><strong>' + jsxc.translate('%%Resource%%') + ':</strong> ' + res + '</li>');
+               $('#jsxc_dialog ul.jsxc_vCard').append('<li><strong>' + jsxc.translate('%%Client%%') + ':</strong> ' + client + '</li>');
+               $('#jsxc_dialog ul.jsxc_vCard').append('<li>' + jsxc.translate('<strong>%%Status%%:</strong> %%' + jsxc.CONST.STATUS[status] + '%%') + '</li>');
+            }
+         }
+
+         var printProp = function(el, depth) {
+            var content = '';
+
+            el.each(function() {
+               var item = $(this);
+               var children = $(this).children();
+
+               content += '<li>';
+
+               var prop = jsxc.translate('%%' + item[0].tagName + '%%');
+
+               if (prop !== ' ') {
+                  content += '<strong>' + prop + ':</strong> ';
+               }
+
+               if (item[0].tagName === 'PHOTO') {
+
+               } else if (children.length > 0) {
+                  content += '<ul>';
+                  content += printProp(children, depth + 1);
+                  content += '</ul>';
+               } else if (item.text() !== '') {
+                  content += jsxc.escapeHTML(item.text());
+               }
+
+               content += '</li>';
+
+               if (depth === 0 && $('#jsxc_dialog ul.jsxc_vCard').length > 0) {
+                  if ($('#jsxc_dialog ul.jsxc_vCard li.jsxc_sep:first').length > 0) {
+                     $('#jsxc_dialog ul.jsxc_vCard li.jsxc_sep:first').before(content);
+                  } else {
+                     $('#jsxc_dialog ul.jsxc_vCard').append(content);
+                  }
+                  content = '';
+               }
+            });
+
+            if (depth > 0) {
+               return content;
+            }
+         };
+
+         var failedToLoad = function() {
+            if ($('#jsxc_dialog ul.jsxc_vCard').length === 0) {
+               return;
+            }
+
+            $('#jsxc_dialog p').remove();
+
+            var content = '<p>';
+            content += jsxc.translate('%%Sorry, your buddy doesn\'t provide any information.%%');
+            content += '</p>';
+
+            $('#jsxc_dialog').append(content);
+         };
+
+         jsxc.xmpp.loadVcard(bid, function(stanza) {
+
+            if ($('#jsxc_dialog ul.jsxc_vCard').length === 0) {
+               return;
+            }
+
+            $('#jsxc_dialog p').remove();
+
+            var photo = $(stanza).find("vCard > PHOTO");
+
+            if (photo.length > 0) {
+               var img = photo.find('BINVAL').text();
+               var type = photo.find('TYPE').text();
+               var src = 'data:' + type + ';base64,' + img;
+
+               $('#jsxc_dialog h3').before('<img class="jsxc_vCard" src="' + src + '" alt="avatar" />');
+            }
+
+            if ($(stanza).find('vCard').length === 0 || ($(stanza).find('vcard > *').length === 1 && photo.length === 1)) {
+               failedToLoad();
+               return;
+            }
+
+            printProp($(stanza).find('vcard > *'), 0);
+
+         }, failedToLoad);
+      },
+
+      showSettings: function() {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('settings'));
+
+         if (jsxc.options.get('xmpp').overwrite === 'false' || jsxc.options.get('xmpp').overwrite === false) {
+            $('.jsxc_fieldsetXmpp').hide();
+         }
+
+         $('#jsxc_dialog form').each(function() {
+            var self = $(this);
+
+            self.find('input[type!="submit"]').each(function() {
+               var id = this.id.split("-");
+               var prop = id[0];
+               var key = id[1];
+               var type = this.type;
+
+               var data = jsxc.options.get(prop);
+
+               if (data && typeof data[key] !== 'undefined') {
+                  if (type === 'checkbox') {
+                     if (data[key] !== 'false' && data[key] !== false) {
+                        this.checked = 'checked';
+                     }
+                  } else {
+                     $(this).val(data[key]);
+                  }
+               }
+            });
+         });
+
+         $('#jsxc_dialog form').submit(function() {
+
+            var self = $(this);
+            var data = {};
+
+            self.find('input[type!="submit"]').each(function() {
+               var id = this.id.split("-");
+               var prop = id[0];
+               var key = id[1];
+               var val;
+               var type = this.type;
+
+               if (type === 'checkbox') {
+                  val = this.checked;
+               } else {
+                  val = $(this).val();
+               }
+
+               if (!data[prop]) {
+                  data[prop] = {};
+               }
+
+               data[prop][key] = val;
+            });
+
+            $.each(data, function(key, val) {
+               jsxc.options.set(key, val);
+            });
+
+            var err = jsxc.options.saveSettinsPermanent.call(this, data);
+
+            setTimeout(function() {
+               self.find('input[type="submit"]').effect('highlight', {
+                  color: (err) ? 'green' : 'red'
+               }, 4000);
+            }, 200);
+
+            return false;
+         });
+      },
+
+      /**
+       * Show prompt for notification permission.
+       * 
+       * @memberOf jsxc.gui
+       */
+      showRequestNotification: function() {
+         jsxc.gui.showConfirmDialog(jsxc.translate("%%Should we notify you_%%"), function() {
+            jsxc.gui.dialog.open(jsxc.gui.template.get('pleaseAccept'), {
+               noClose: true
+            });
+
+            jsxc.notification.requestPermission();
+         }, function() {
+            $(document).trigger('notificationfailure.jsxc');
+         });
+      },
+
+      showUnknownSender: function(bid) {
+         jsxc.gui.showConfirmDialog(jsxc.translate('%%You_received_a_message_from_an_unknown_sender%% (' + bid + '). %%Do_you_want_to_display_them%%'), function() {
+
+            jsxc.gui.dialog.close();
+
+            jsxc.storage.saveBuddy(bid, {
+               jid: bid,
+               name: bid,
+               status: 0,
+               sub: 'none',
+               res: []
+            });
+
+            jsxc.gui.window.open(bid);
+
+         }, function() {
+            // reset state
+            jsxc.storage.removeUserItem('chat', bid);
+         });
+      },
+
+      /**
+       * Change own presence to pres.
+       * 
+       * @memberOf jsxc.gui
+       * @param pres {CONST.STATUS} New presence state
+       * @param external {boolean} True if triggered from other tab.
+       */
+      changePresence: function(pres, external) {
+
+         if (external !== true) {
+            jsxc.storage.setUserItem('presence', pres);
+         }
+
+         if (jsxc.master) {
+            jsxc.xmpp.sendPres();
+         }
+
+         $('#jsxc_presence > span').text($('#jsxc_presence > ul .jsxc_' + pres).text());
+
+         jsxc.gui.updatePresence('own', pres);
+      },
+
+      /**
+       * Update all presence objects for given user.
+       * 
+       * @memberOf jsxc.gui
+       * @param bid bar jid of user.
+       * @param {CONST.STATUS} pres New presence state.
+       */
+      updatePresence: function(bid, pres) {
+
+         if (bid === 'own') {
+            if (pres === 'dnd') {
+               $('#jsxc_menu .jsxc_muteNotification').addClass('jsxc_disabled');
+               jsxc.notification.muteSound(true);
+            } else {
+               $('#jsxc_menu .jsxc_muteNotification').removeClass('jsxc_disabled');
+
+               if (!jsxc.options.get('muteNotification')) {
+                  jsxc.notification.unmuteSound(true);
+               }
+            }
+         }
+
+         $('.jsxc_presence[data-bid="' + bid + '"]').removeClass('jsxc_' + jsxc.CONST.STATUS.join(' jsxc_')).addClass('jsxc_' + pres);
+      },
+
+      /**
+       * Switch read state to UNread.
+       * 
+       * @memberOf jsxc.gui
+       * @param bid
+       */
+      unreadMsg: function(bid) {
+         var win = jsxc.gui.window.get(bid);
+
+         jsxc.gui.roster.getItem(bid).add(win).addClass('jsxc_unreadMsg');
+         jsxc.storage.updateUserItem('window', bid, 'unread', true);
+      },
+
+      /**
+       * Switch read state to read.
+       * 
+       * @memberOf jsxc.gui
+       * @param bid
+       */
+      readMsg: function(bid) {
+         var win = jsxc.gui.window.get(bid);
+
+         if (win.hasClass('jsxc_unreadMsg')) {
+            jsxc.gui.roster.getItem(bid).add(win).removeClass('jsxc_unreadMsg');
+            jsxc.storage.updateUserItem('window', bid, 'unread', false);
+         }
+      },
+
+      /**
+       * This function searches for URI scheme according to XEP-0147.
+       * 
+       * @memberOf jsxc.gui
+       * @param container In which element should we search?
+       */
+      detectUriScheme: function(container) {
+         container = (container) ? $(container) : $('body');
+
+         container.find("a[href^='xmpp:']").each(function() {
+
+            var element = $(this);
+            var href = element.attr('href').replace(/^xmpp:/, '');
+            var jid = href.split('?')[0];
+            var action, params = {};
+
+            if (href.indexOf('?') < 0) {
+               action = 'message';
+            } else {
+               var pairs = href.substring(href.indexOf('?') + 1).split(';');
+               action = pairs[0];
+
+               var i, key, value;
+               for (i = 1; i < pairs.length; i++) {
+                  key = pairs[i].split('=')[0];
+                  value = (pairs[i].indexOf('=') > 0) ? pairs[i].substring(pairs[i].indexOf('=') + 1) : null;
+
+                  params[decodeURIComponent(key)] = decodeURIComponent(value);
+               }
+            }
+
+            if (typeof jsxc.gui.queryActions[action] === 'function') {
+               element.addClass('jsxc_uriScheme jsxc_uriScheme_' + action);
+
+               element.off('click').click(function(ev) {
+                  ev.stopPropagation();
+
+                  jsxc.gui.queryActions[action].call(jsxc, jid, params);
+
+                  return false;
+               });
+            }
+         });
+      },
+
+      detectEmail: function(container) {
+         container = (container) ? $(container) : $('body');
+
+         container.find('a[href^="mailto:"]').each(function() {
+            var spot = $("<span>X</span>").addClass("jsxc_spot");
+            var href = $(this).attr("href").replace(/^ *mailto:/, "").trim();
+
+            if (href !== '' && href !== Strophe.getBareJidFromJid(jsxc.storage.getItem("jid"))) {
+               var bid = jsxc.jidToBid(href);
+               var self = $(this);
+               var s = self.prev();
+
+               if (!s.hasClass('jsxc_spot')) {
+                  s = spot.clone().attr('data-bid', bid);
+
+                  self.before(s);
+               }
+
+               s.off('click');
+
+               if (jsxc.storage.getUserItem('buddy', bid)) {
+                  jsxc.gui.update(bid);
+                  s.click(function() {
+                     jsxc.gui.window.open(bid);
+
+                     return false;
+                  });
+               } else {
+                  s.click(function() {
+                     jsxc.gui.showContactDialog(href);
+
+                     return false;
+                  });
+               }
+            }
+         });
+      }
+   };
+
+   /**
+    * Handle functions related to the gui of the roster
+    * 
+    * @namespace jsxc.gui.roster
+    */
+   jsxc.gui.roster = {
+
+      /**
+       * Init the roster skeleton
+       * 
+       * @memberOf jsxc.gui.roster
+       * @returns {undefined}
+       */
+      init: function() {
+         $(jsxc.options.rosterAppend + ':first').append($(jsxc.gui.template.get('roster')));
+
+         if (jsxc.options.get('hideOffline')) {
+            $('#jsxc_menu .jsxc_hideOffline').text(jsxc.translate('%%Show offline%%'));
+            $('#jsxc_buddylist').addClass('jsxc_hideOffline');
+         }
+
+         $('#jsxc_menu .jsxc_settings').click(function() {
+            jsxc.gui.showSettings();
+         });
+
+         $('#jsxc_menu .jsxc_hideOffline').click(function() {
+            var hideOffline = !jsxc.options.get('hideOffline');
+
+            if (hideOffline) {
+               $('#jsxc_buddylist').addClass('jsxc_hideOffline');
+            } else {
+               $('#jsxc_buddylist').removeClass('jsxc_hideOffline');
+            }
+
+            $(this).text(hideOffline ? jsxc.translate('%%Show offline%%') : jsxc.translate('%%Hide offline%%'));
+
+            jsxc.options.set('hideOffline', hideOffline);
+         });
+
+         if (jsxc.options.get('muteNotification')) {
+            jsxc.notification.muteSound();
+         }
+
+         $('#jsxc_menu .jsxc_muteNotification').click(function() {
+
+            if (jsxc.storage.getUserItem('presence') === 'dnd') {
+               return;
+            }
+
+            // invert current choice
+            var mute = !jsxc.options.get('muteNotification');
+
+            if (mute) {
+               jsxc.notification.muteSound();
+            } else {
+               jsxc.notification.unmuteSound();
+            }
+         });
+
+         $('#jsxc_roster .jsxc_addBuddy').click(function() {
+            jsxc.gui.showContactDialog();
+         });
+
+         $('#jsxc_roster .jsxc_onlineHelp').click(function() {
+            window.open("http://www.jsxc.org/manual.html", "onlineHelp");
+         });
+
+         $('#jsxc_roster .jsxc_about').click(function() {
+            jsxc.gui.showAboutDialog();
+         });
+
+         $('#jsxc_toggleRoster').click(function() {
+            jsxc.gui.roster.toggle();
+         });
+
+         $('#jsxc_presence > ul > li').click(function() {
+            var self = $(this);
+
+            jsxc.gui.changePresence(self.data('pres'));
+         });
+
+         $('#jsxc_buddylist').slimScroll({
+            distance: '3px',
+            height: ($('#jsxc_roster').height() - 31) + 'px',
+            width: $('#jsxc_buddylist').width() + 'px',
+            color: '#fff',
+            opacity: '0.5'
+         });
+
+         $('#jsxc_roster > .jsxc_bottom > div').each(function() {
+            jsxc.gui.toggleList.call($(this));
+         });
+
+         if (jsxc.storage.getUserItem('roster') === 'hidden') {
+            $('#jsxc_roster').css('right', '-200px');
+            $('#jsxc_windowList > ul').css('paddingRight', '10px');
+         }
+
+         var pres = jsxc.storage.getUserItem('presence') || 'online';
+         $('#jsxc_presence > span').text($('#jsxc_presence > ul .jsxc_' + pres).text());
+         jsxc.gui.updatePresence('own', pres);
+
+         $(document).on('cloaded.roster.jsxc', function() {
+            jsxc.gui.updateAvatar($('#jsxc_avatar'), jsxc.storage.getItem('jid'), 'own');
+         });
+
+         jsxc.gui.tooltip('#jsxc_roster');
+
+         jsxc.notice.load();
+
+         $(document).trigger('ready.roster.jsxc');
+      },
+
+      /**
+       * Create roster item and add it to the roster
+       * 
+       * @param {String} bid bar jid
+       */
+      add: function(bid) {
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var bud = jsxc.gui.buddyTemplate.clone().attr('data-bid', bid).attr('data-type', data.type || 'chat');
+
+         jsxc.gui.roster.insert(bid, bud);
+
+         bud.click(function() {
+            jsxc.gui.window.open(bid);
+         });
+
+         bud.find('.jsxc_chaticon').click(function() {
+            jsxc.gui.window.open(bid);
+         });
+
+         bud.find('.jsxc_rename').click(function() {
+            jsxc.gui.roster.rename(bid);
+            return false;
+         });
+
+         bud.find('.jsxc_delete').click(function() {
+            jsxc.gui.showRemoveDialog(bid);
+            return false;
+         });
+
+         var expandClick = function() {
+            bud.trigger('extra.jsxc');
+
+            bud.toggleClass('jsxc_expand');
+
+            jsxc.gui.updateAvatar(bud, data.jid, data.avatar);
+            return false;
+         };
+
+         bud.find('.jsxc_control').click(expandClick);
+         bud.dblclick(expandClick);
+
+         bud.find('.jsxc_vcardicon').click(function() {
+            jsxc.gui.showVcard(data.jid);
+            return false;
+         });
+
+         jsxc.gui.update(bid);
+
+         // update scrollbar
+         $('#jsxc_buddylist').slimScroll({
+            scrollTo: '0px'
+         });
+
+         $(document).trigger('add.roster.jsxc', [ bid, data, bud ]);
+      },
+
+      getItem: function(bid) {
+         return $("#jsxc_buddylist > li[data-bid='" + bid + "']");
+      },
+
+      /**
+       * Insert roster item. First order: online > away > offline. Second order:
+       * alphabetical of the name
+       * 
+       * @param {type} bid
+       * @param {jquery} li roster item which should be insert
+       * @returns {undefined}
+       */
+      insert: function(bid, li) {
+
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var listElements = $('#jsxc_buddylist > li');
+         var insert = false;
+
+         // Insert buddy with no mutual friendship to the end
+         var status = (data.sub === 'both') ? data.status : -1;
+
+         listElements.each(function() {
+
+            var thisStatus = ($(this).data('sub') === 'both') ? $(this).data('status') : -1;
+
+            if (($(this).data('name').toLowerCase() > data.name.toLowerCase() && thisStatus === status) || thisStatus < status) {
+
+               $(this).before(li);
+               insert = true;
+
+               return false;
+            }
+         });
+
+         if (!insert) {
+            li.appendTo('#jsxc_buddylist');
+         }
+      },
+
+      /**
+       * Initiate reorder of roster item
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      reorder: function(bid) {
+         jsxc.gui.roster.insert(bid, jsxc.gui.roster.remove(bid));
+      },
+
+      /**
+       * Removes buddy from roster
+       * 
+       * @param {String} bid bar jid
+       * @return {JQueryObject} Roster list element
+       */
+      remove: function(bid) {
+         return jsxc.gui.roster.getItem(bid).detach();
+      },
+
+      /**
+       * Removes buddy from roster and clean up
+       * 
+       * @param {String} bid bar compatible jid
+       */
+      purge: function(bid) {
+         if (jsxc.master) {
+            jsxc.storage.removeUserItem('buddy', bid);
+            jsxc.storage.removeUserItem('otr', bid);
+            jsxc.storage.removeUserItem('otr_version_' + bid);
+            jsxc.storage.removeUserItem('chat', bid);
+            jsxc.storage.removeUserItem('window', bid);
+            jsxc.storage.removeUserElement('buddylist', bid);
+            jsxc.storage.removeUserElement('windowlist', bid);
+         }
+
+         jsxc.gui.window._close(bid);
+         jsxc.gui.roster.remove(bid);
+      },
+
+      /**
+       * Create input element for rename action
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      rename: function(bid) {
+         var name = jsxc.gui.roster.getItem(bid).find('.jsxc_name');
+         var options = jsxc.gui.roster.getItem(bid).find('.jsxc_options, .jsxc_control');
+         var input = $('<input type="text" name="name"/>');
+
+         options.hide();
+         name = name.replaceWith(input);
+
+         input.val(name.text());
+         input.keypress(function(ev) {
+            if (ev.which !== 13) {
+               return;
+            }
+
+            options.show();
+            input.replaceWith(name);
+            jsxc.gui.roster._rename(bid, $(this).val());
+
+            $('html').off('click');
+         });
+
+         // Disable html click event, if click on input
+         input.click(function() {
+            return false;
+         });
+
+         $('html').one('click', function() {
+            options.show();
+            input.replaceWith(name);
+            jsxc.gui.roster._rename(bid, input.val());
+         });
+      },
+
+      /**
+       * Rename buddy
+       * 
+       * @param {type} bid
+       * @param {type} newname new name of buddy
+       * @returns {undefined}
+       */
+      _rename: function(bid, newname) {
+         if (jsxc.master) {
+            var d = jsxc.storage.getUserItem('buddy', bid);
+            var iq = $iq({
+               type: 'set'
+            }).c('query', {
+               xmlns: 'jabber:iq:roster'
+            }).c('item', {
+               jid: Strophe.getBareJidFromJid(d.jid),
+               name: newname
+            });
+            jsxc.xmpp.conn.sendIQ(iq);
+         }
+
+         jsxc.storage.updateUserItem('buddy', bid, 'name', newname);
+         jsxc.gui.update(bid);
+      },
+
+      /**
+       * Toogle complete roster
+       * 
+       * @param {Integer} d Duration in ms
+       */
+      toggle: function(d) {
+         var duration = d || 500;
+
+         var roster = $('#jsxc_roster');
+         var wl = $('#jsxc_windowList');
+
+         var roster_width = roster.innerWidth();
+         var roster_right = parseFloat($('#jsxc_roster').css('right'));
+         var state = (roster_right < 0) ? 'shown' : 'hidden';
+
+         jsxc.storage.setUserItem('roster', state);
+
+         roster.animate({
+            right: ((roster_width + roster_right) * -1) + 'px'
+         }, duration);
+         wl.animate({
+            right: (10 - roster_right) + 'px'
+         }, duration);
+
+         $(document).trigger('toggle.roster.jsxc', [ state, duration ]);
+      },
+
+      /**
+       * Shows a text with link to a login box that no connection exists.
+       */
+      noConnection: function() {
+         $('#jsxc_roster .slimScrollDiv').remove();
+         $('#jsxc_roster > .jsxc_bottom').remove();
+
+         $('#jsxc_roster').append($('<p>' + jsxc.l.no_connection + '</p>').append(' <a>' + jsxc.l.relogin + '</a>').click(function() {
+            jsxc.gui.showLoginBox();
+         }));
+      },
+
+      /**
+       * Shows a text with link to add a new buddy.
+       * 
+       * @memberOf jsxc.gui.roster
+       */
+      empty: function() {
+         var text = $('<p>' + jsxc.l.Your_roster_is_empty_add_a + '</p>');
+         var link = $('<a>' + jsxc.l.new_buddy + '</a>');
+
+         link.click(function() {
+            jsxc.gui.showContactDialog();
+         });
+         text.append(link);
+         text.append('.');
+
+         $('#jsxc_roster').prepend(text);
+      }
+   };
+
+   /**
+    * Wrapper for dialog
+    * 
+    * @namespace jsxc.gui.dialog
+    */
+   jsxc.gui.dialog = {
+      /**
+       * Open a Dialog.
+       * 
+       * @memberOf jsxc.gui.dialog
+       * @param {String} data Data of the dialog
+       * @param {Object} [o] Options for the dialog
+       * @param {Boolean} [o.noClose] If true, hide all default close options
+       * @returns {jQuery} Dialog object
+       */
+      open: function(data, o) {
+
+         var opt = o || {};
+
+         // default options
+         var options = {};
+         options = {
+            onComplete: function() {
+               $('#jsxc_dialog .jsxc_close').click(function(ev) {
+                  ev.preventDefault();
+
+                  jsxc.gui.dialog.close();
+               });
+
+               // workaround for old colorbox version (used by firstrunwizard)
+               if (options.closeButton === false) {
+                  $('#cboxClose').hide();
+               }
+
+               $.colorbox.resize();
+
+               $(document).trigger('complete.dialog.jsxc');
+            },
+            onClosed: function() {
+               $(document).trigger('close.dialog.jsxc');
+            },
+            onCleanup: function() {
+               $(document).trigger('cleanup.dialog.jsxc');
+            },
+            opacity: 0.5
+         };
+
+         if (opt.noClose) {
+            options.overlayClose = false;
+            options.escKey = false;
+            options.closeButton = false;
+            delete opt.noClose;
+         }
+
+         $.extend(options, opt);
+
+         options.html = '<div id="jsxc_dialog">' + data + '</div>';
+
+         $.colorbox(options);
+
+         return $('#jsxc_dialog');
+      },
+
+      /**
+       * Close current dialog.
+       */
+      close: function() {
+         jsxc.debug('close dialog');
+         $.colorbox.close();
+      },
+
+      /**
+       * Resizes current dialog.
+       * 
+       * @param {Object} options e.g. width and height
+       */
+      resize: function(options) {
+         $.colorbox.resize(options);
+      }
+   };
+
+   /**
+    * Handle functions related to the gui of the window
+    * 
+    * @namespace jsxc.gui.window
+    */
+   jsxc.gui.window = {
+      /**
+       * Init a window skeleton
+       * 
+       * @memberOf jsxc.gui.window
+       * @param {String} bid
+       * @returns {jQuery} Window object
+       */
+      init: function(bid) {
+         if (jsxc.gui.window.get(bid).length > 0) {
+            return jsxc.gui.window.get(bid);
+         }
+
+         var win = jsxc.gui.windowTemplate.clone().attr('data-bid', bid).hide().appendTo('#jsxc_windowList > ul').show('slow');
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         // Attach jid to window
+         win.data('jid', data.jid);
+
+         // Add handler
+
+         jsxc.gui.toggleList.call(win.find('.jsxc_settings'));
+
+         win.find('.jsxc_verification').click(function() {
+            jsxc.gui.showVerification(bid);
+         });
+
+         win.find('.jsxc_fingerprints').click(function() {
+            jsxc.gui.showFingerprints(bid);
+         });
+
+         win.find('.jsxc_transfer').click(function() {
+            jsxc.otr.toggleTransfer(bid);
+         });
+
+         win.find('.jsxc_bar').click(function() {
+            jsxc.gui.window.toggle(bid);
+         });
+
+         win.find('.jsxc_close').click(function() {
+            jsxc.gui.window.close(bid);
+         });
+
+         win.find('.jsxc_clear').click(function() {
+            jsxc.gui.window.clear(bid);
+         });
+
+         win.find('.jsxc_tools').click(function() {
+            return false;
+         });
+
+         win.find('.jsxc_textinput').keyup(function(ev) {
+            var body = $(this).val();
+
+            if (ev.which === 13) {
+               body = '';
+            }
+
+            jsxc.storage.updateUserItem('window', bid, 'text', body);
+
+            if (ev.which === 27) {
+               jsxc.gui.window.close(bid);
+            }
+         }).keypress(function(ev) {
+            if (ev.which !== 13 || !$(this).val()) {
+               return;
+            }
+
+            jsxc.gui.window.postMessage(bid, 'out', $(this).val());
+
+            $(this).val('');
+         }).focus(function() {
+            // remove unread flag
+            jsxc.gui.readMsg(bid);
+         }).mouseenter(function() {
+            $('#jsxc_windowList').data('isOver', true);
+         }).mouseleave(function() {
+            $('#jsxc_windowList').data('isOver', false);
+         });
+
+         win.find('.jsxc_textarea').click(function() {
+            win.find('.jsxc_textinput').focus();
+         });
+
+         win.find('.jsxc_textarea').slimScroll({
+            height: '234px',
+            distance: '3px'
+         });
+
+         win.find('.jsxc_fade').hide();
+
+         win.find('.jsxc_name').disableSelection();
+
+         win.find('.slimScrollDiv').resizable({
+            handles: 'w, nw, n',
+            minHeight: 234,
+            minWidth: 250,
+            resize: function(event, ui) {
+               win.width(ui.size.width);
+               win.find('.jsxc_textarea').slimScroll({
+                  height: ui.size.height
+               });
+               win.find('.jsxc_emoticons').css('top', (ui.size.height + 6) + 'px');
+            }
+         });
+
+         if ($.inArray(bid, jsxc.storage.getUserItem('windowlist')) < 0) {
+
+            // add window to windowlist
+            var wl = jsxc.storage.getUserItem('windowlist');
+            wl.push(bid);
+            jsxc.storage.setUserItem('windowlist', wl);
+
+            // init window element in storage
+            jsxc.storage.setUserItem('window', bid, {
+               minimize: true,
+               text: '',
+               unread: false
+            });
+         } else {
+
+            if (jsxc.storage.getUserItem('window', bid).unread) {
+               jsxc.gui.unreadMsg(bid);
+            }
+         }
+
+         $.each(jsxc.gui.emotions, function(i, val) {
+            var ins = val[0].split(' ')[0];
+            var li = $('<li><div title="' + ins + '" class="jsxc_' + val[1] + '"/></li>');
+            li.click(function() {
+               win.find('input').val(win.find('input').val() + ins);
+               win.find('input').focus();
+            });
+            win.find('.jsxc_emoticons ul').append(li);
+         });
+
+         jsxc.gui.toggleList.call(win.find('.jsxc_emoticons'));
+
+         jsxc.gui.window.restoreChat(bid);
+
+         jsxc.gui.update(bid);
+
+         jsxc.gui.updateWindowListSB();
+
+         // create related otr object
+         if (jsxc.master && !jsxc.otr.objects[bid]) {
+            jsxc.otr.create(bid);
+         } else {
+            jsxc.otr.enable(bid);
+         }
+
+         $(document).trigger('init.window.jsxc', [ win ]);
+
+         return win;
+      },
+
+      /**
+       * Returns the window element
+       * 
+       * @param {String} bid
+       * @returns {jquery} jQuery object of the window element
+       */
+      get: function(id) {
+         return $("li.jsxc_windowItem[data-bid='" + jsxc.jidToBid(id) + "']");
+      },
+
+      /**
+       * Open a window, related to the bid. If the window doesn't exist, it will
+       * be created.
+       * 
+       * @param {String} bid
+       * @returns {jQuery} Window object
+       */
+      open: function(bid) {
+         var win = jsxc.gui.window.init(bid);
+         jsxc.gui.window.show(bid);
+         jsxc.gui.window.highlight(bid);
+
+         var padding = $("#jsxc_windowListSB").width();
+         var innerWidth = $('#jsxc_windowList>ul').width();
+         var outerWidth = $('#jsxc_windowList').width() - padding;
+
+         if (innerWidth > outerWidth) {
+            var offset = parseInt($('#jsxc_windowList>ul').css('right'));
+            var width = win.outerWidth(true);
+
+            var right = innerWidth - win.position().left - width + offset;
+            var left = outerWidth - (innerWidth - win.position().left) - offset;
+
+            if (left < 0) {
+               jsxc.gui.scrollWindowListBy(left * -1);
+            }
+
+            if (right < 0) {
+               jsxc.gui.scrollWindowListBy(right);
+            }
+         }
+
+         return win;
+      },
+
+      /**
+       * Close chatwindow and clean up
+       * 
+       * @param {String} bid bar jid
+       */
+      close: function(bid) {
+
+         if (jsxc.gui.window.get(bid).length === 0) {
+            jsxc.warn('Want to close a window, that is not open.');
+            return;
+         }
+
+         jsxc.storage.removeUserElement('windowlist', bid);
+         jsxc.storage.removeUserItem('window', bid);
+
+         if (jsxc.storage.getUserItem('buddylist').indexOf(bid) < 0) {
+            // delete data from unknown sender
+
+            jsxc.storage.removeUserItem('buddy', bid);
+            jsxc.storage.removeUserItem('chat', bid);
+         }
+
+         jsxc.gui.window._close(bid);
+      },
+
+      /**
+       * Close chatwindow
+       * 
+       * @param {String} bid
+       */
+      _close: function(bid) {
+         jsxc.gui.window.get(bid).hide('slow', function() {
+            $(this).remove();
+
+            jsxc.gui.updateWindowListSB();
+         });
+      },
+
+      /**
+       * Toggle between minimize and maximize of the text area
+       * 
+       * @param {String} bid bar jid
+       */
+      toggle: function(bid) {
+
+         var win = jsxc.gui.window.get(bid);
+
+         if (win.parents("#jsxc_windowList").length === 0) {
+            return;
+         }
+
+         if (win.find('.jsxc_fade').is(':hidden')) {
+            jsxc.gui.window.show(bid);
+         } else {
+            jsxc.gui.window.hide(bid);
+         }
+
+         jsxc.gui.updateWindowListSB();
+      },
+
+      /**
+       * Maximize text area and save
+       * 
+       * @param {String} bid
+       */
+      show: function(bid) {
+
+         jsxc.storage.updateUserItem('window', bid, 'minimize', false);
+
+         jsxc.gui.window._show(bid);
+      },
+
+      /**
+       * Maximize text area
+       * 
+       * @param {String} bid
+       * @returns {undefined}
+       */
+      _show: function(bid) {
+         var win = jsxc.gui.window.get(bid);
+         jsxc.gui.window.get(bid).find('.jsxc_fade').slideDown();
+         win.removeClass('jsxc_min');
+
+         // If the area is hidden, the scrolldown function doesn't work. So we
+         // call it here.
+         jsxc.gui.window.scrollDown(bid);
+
+         if (jsxc.restoreCompleted) {
+            win.find('.jsxc_textinput').focus();
+         }
+
+         win.trigger('show.window.jsxc');
+      },
+
+      /**
+       * Minimize text area and save
+       * 
+       * @param {String} bid
+       */
+      hide: function(bid) {
+         jsxc.storage.updateUserItem('window', bid, 'minimize', true);
+
+         jsxc.gui.window._hide(bid);
+      },
+
+      /**
+       * Minimize text area
+       * 
+       * @param {String} bid
+       */
+      _hide: function(bid) {
+         jsxc.gui.window.get(bid).addClass('jsxc_min').find(' .jsxc_fade').slideUp();
+
+         jsxc.gui.window.get(bid).trigger('hidden.window.jsxc');
+      },
+
+      /**
+       * Highlight window
+       * 
+       * @param {type} bid
+       */
+      highlight: function(bid) {
+         var el = jsxc.gui.window.get(bid).find(' .jsxc_bar');
+
+         if (!el.is(':animated')) {
+            el.effect('highlight', {
+               color: 'orange'
+            }, 2000);
+         }
+      },
+
+      /**
+       * Scroll chat area to the bottom
+       * 
+       * @param {String} bid bar jid
+       */
+      scrollDown: function(bid) {
+         var chat = jsxc.gui.window.get(bid).find('.jsxc_textarea');
+
+         // check if chat exist
+         if (chat.length === 0) {
+            return;
+         }
+
+         chat.slimScroll({
+            scrollTo: (chat.get(0).scrollHeight + 'px')
+         });
+      },
+
+      /**
+       * Write Message to chat area and save
+       * 
+       * @param {String} bid bar jid
+       * @param {String} direction 'in' message is received or 'out' message is
+       *        send
+       * @param {String} msg Message to display
+       */
+      postMessage: function(bid, direction, msg) {
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var html_msg = msg;
+
+         // remove html tags and reencode html tags
+         msg = jsxc.removeHTML(msg);
+         msg = jsxc.escapeHTML(msg);
+
+         // exceptions:
+
+         if (direction === 'out' && data.msgstate === 2) {
+            direction = 'sys';
+            msg = jsxc.l.your_message_wasnt_send_please_end_your_private_conversation;
+         }
+
+         if (direction === 'in' && data.msgstate === 2) {
+            direction = 'sys';
+            msg = jsxc.l.unencrypted_message_received + ' ' + msg;
+         }
+
+         if (direction === 'out' && data.sub === 'from') {
+            direction = 'sys';
+            msg = jsxc.l.your_message_wasnt_send_because_you_have_no_valid_subscription;
+         }
+
+         var post = jsxc.storage.saveMessage(bid, direction, msg);
+
+         if (direction === 'in') {
+            $(document).trigger('postmessagein.jsxc', [ bid, html_msg ]);
+         }
+
+         if (direction === 'out' && jsxc.master) {
+            jsxc.xmpp.sendMessage(bid, html_msg, post.uid);
+         }
+
+         jsxc.gui.window._postMessage(bid, post);
+
+         if (direction === 'out' && msg === '?') {
+            jsxc.gui.window.postMessage(bid, 'sys', '42');
+         }
+      },
+
+      /**
+       * Write Message to chat area
+       * 
+       * @param {String} bid bar jid
+       * @param {Object} post Post object with direction, msg, uid, received
+       * @param {Bool} restore If true no highlights are used and so unread flag
+       *        set
+       */
+      _postMessage: function(bid, post, restore) {
+         var win = jsxc.gui.window.get(bid);
+         var msg = post.msg;
+         var direction = post.direction;
+         var uid = post.uid;
+         var received = post.received || false;
+
+         if (win.find('.jsxc_textinput').is(':not(:focus)') && jsxc.restoreCompleted && direction === 'in' && !restore) {
+            jsxc.gui.window.highlight(bid);
+         }
+
+         msg = msg.replace(jsxc.CONST.REGEX.URL, function(url) {
+
+            var href = (url.match(/^https?:\/\//i)) ? url : 'http://' + url;
+
+            return '<a href="' + href + '" target="_blank">' + url + '</a>';
+         });
+
+         msg = msg.replace(new RegExp('(xmpp:)?(' + jsxc.CONST.REGEX.JID.source + ')(\\?[^\\s]+\\b)?', 'i'), function(match, protocol, jid, action) {
+            if (protocol === 'xmpp:') {
+               if (typeof action === 'string') {
+                  jid += action;
+               }
+
+               return '<a href="xmpp:' + jid + '">' + jid + '</a>';
+            }
+
+            return '<a href="mailto:' + jid + '" target="_blank">' + jid + '</a>';
+         });
+
+         $.each(jsxc.gui.emotions, function(i, val) {
+            msg = msg.replace(val[2], function(match, p1) {
+
+               // escape value for alt and title, this prevents double
+               // replacement
+               var esc = '', i;
+               for (i = 0; i < p1.length; i++) {
+                  esc += '&#' + p1.charCodeAt(i) + ';';
+               }
+
+               return '<div title="' + esc + '" class="jsxc_emoticon jsxc_' + val[1] + '"/>';
+            });
+         });
+
+         var msgDiv = $("<div>");
+         msgDiv.addClass('jsxc_chatmessage jsxc_' + direction);
+         msgDiv.attr('id', uid);
+         msgDiv.html(msg);
+
+         if (received) {
+            msgDiv.addClass('jsxc_received');
+         }
+
+         if (direction === 'sys') {
+            jsxc.gui.window.get(bid).find('.jsxc_textarea').append('<div style="clear:both"/>');
+         }
+
+         win.find('.jsxc_textarea').append(msgDiv);
+
+         jsxc.gui.detectUriScheme(win);
+         jsxc.gui.detectEmail(win);
+
+         jsxc.gui.window.scrollDown(bid);
+
+         // if window has no focus set unread flag
+         if (!win.find('.jsxc_textinput').is(':focus') && jsxc.restoreCompleted && !restore) {
+            jsxc.gui.unreadMsg(bid);
+         }
+      },
+
+      /**
+       * Set text into input area
+       * 
+       * @param {type} bid
+       * @param {type} text
+       * @returns {undefined}
+       */
+      setText: function(bid, text) {
+         jsxc.gui.window.get(bid).find('.jsxc_textinput').val(text);
+      },
+
+      /**
+       * Load old log into chat area
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      restoreChat: function(bid) {
+         var chat = jsxc.storage.getUserItem('chat', bid);
+
+         while (chat !== null && chat.length > 0) {
+            var c = chat.pop();
+            jsxc.gui.window._postMessage(bid, c, true);
+         }
+      },
+
+      /**
+       * Clear chat history
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      clear: function(bid) {
+         jsxc.storage.setUserItem('chat', bid, []);
+         jsxc.gui.window.get(bid).find('.jsxc_textarea').empty();
+      }
+   };
+
+   /**
+    * Hold all HTML templates.
+    * 
+    * @namespace jsxc.gui.template
+    */
+   jsxc.gui.template = {
+      /**
+       * Return requested template and replace all placeholder
+       * 
+       * @memberOf jsxc.gui.template;
+       * @param {type} name template name
+       * @param {type} bid
+       * @param {type} msg
+       * @returns {String} HTML Template
+       */
+      get: function(name, bid, msg) {
+
+         // common placeholder
+         var ph = {
+            my_priv_fingerprint: jsxc.storage.getUserItem('priv_fingerprint') ? jsxc.storage.getUserItem('priv_fingerprint').replace(/(.{8})/g, '$1 ') : jsxc.l.not_available,
+            my_jid: jsxc.storage.getItem('jid') || '',
+            my_node: Strophe.getNodeFromJid(jsxc.storage.getItem('jid') || '') || '',
+            root: jsxc.options.root,
+            app_name: jsxc.options.app_name
+         };
+
+         // placeholder depending on bid
+         if (bid) {
+            var data = jsxc.storage.getUserItem('buddy', bid);
+
+            $.extend(ph, {
+               bid_priv_fingerprint: (data && data.fingerprint) ? data.fingerprint.replace(/(.{8})/g, '$1 ') : jsxc.l.not_available,
+               bid_jid: bid,
+               bid_name: (data && data.name) ? data.name : bid
+            });
+         }
+
+         // placeholder depending on msg
+         if (msg) {
+            $.extend(ph, {
+               msg: msg
+            });
+         }
+
+         var ret = jsxc.gui.template[name];
+
+         if (typeof (ret) === 'string') {
+            ret = jsxc.translate(ret);
+
+            ret = ret.replace(/\{\{([a-zA-Z0-9_\-]+)\}\}/g, function(s, key) {
+               return (typeof ph[key] === 'string') ? ph[key] : s;
+            });
+
+            return ret;
+         }
+
+         jsxc.debug('Template not available: ' + name);
+         return name;
+      },
+      authenticationDialog: '<h3>Verification</h3>\
+            <p>%%Authenticating_a_buddy_helps_%%</p>\
+            <div>\
+              <p style="margin:0px;">%%How_do_you_want_to_authenticate_your_buddy%%</p>\
+              <select size="1">\
+                <option>%%Select_method%%</option>\
+                <option>%%Manual%%</option>\
+                <option>%%Question%%</option>\
+                <option>%%Secret%%</option>\
+              </select>\
+            </div>\
+            <div style="display:none">\
+              <p class=".jsxc_explanation">%%To_verify_the_fingerprint_%%</p>\
+              <p><strong>%%Your_fingerprint%%</strong><br />\
+              <span style="text-transform:uppercase">{{my_priv_fingerprint}}</span></p>\
+              <p><strong>%%Buddy_fingerprint%%</strong><br />\
+              <span style="text-transform:uppercase">{{bid_priv_fingerprint}}</span></p><br />\
+              <p class="jsxc_right"><a href="#" class="jsxc_close button">%%Close%%</a> <a href="#" class="button creation">%%Compared%%</a></p>\
+            </div>\
+            <div style="display:none">\
+              <p class=".jsxc_explanation">%%To_authenticate_using_a_question_%%</p>\
+              <p><label for="jsxc_quest">%%Question%%:</label><input type="text" name="quest" id="jsxc_quest" /></p>\
+              <p><label for="jsxc_secret2">%%Secret%%:</label><input type="text" name="secret2" id="jsxc_secret2" /></p>\
+              <p class="jsxc_right"><a href="#" class="button jsxc_close">%%Close%%</a> <a href="#" class="button creation">%%Ask%%</a></p>\
+            </div>\
+            <div style="display:none">\
+              <p class=".jsxc_explanation">%%To_authenticate_pick_a_secret_%%</p>\
+              <p><label for="jsxc_secret">%%Secret%%:</label><input type="text" name="secret" id="jsxc_secret" /></p>\
+              <p class="jsxc_right"><a href="#" class="button jsxc_close">%%Close%%</a> <a href="#" class="button creation">%%Compare%%</a></p>\
+            </div>',
+      fingerprintsDialog: '<div>\
+          <p class="jsxc_maxWidth">%%A_fingerprint_%%</p>\
+          <p><strong>%%Your_fingerprint%%</strong><br />\
+          <span style="text-transform:uppercase">{{my_priv_fingerprint}}</span></p>\
+          <p><strong>%%Buddy_fingerprint%%</strong><br />\
+          <span style="text-transform:uppercase">{{bid_priv_fingerprint}}</span></p><br />\
+          <p class="jsxc_right"><a href="#" class="button jsxc_close">%%Close%%</a></p>\
+        </div>',
+      chatWindow: '<li class="jsxc_min jsxc_windowItem">\
+            <div class="jsxc_window">\
+                <div class="jsxc_bar">\
+                     <div class="jsxc_avatar">☺</div>\
+                     <div class="jsxc_tools">\
+                           <div class="jsxc_settings">\
+                               <ul>\
+                                   <li class="jsxc_fingerprints jsxc_otr jsxc_disabled">%%Fingerprints%%</li>\
+                                   <li class="jsxc_verification">%%Authentication%%</li>\
+                                   <li class="jsxc_transfer jsxc_otr jsxc_disabled">%%start_private%%</li>\
+                                   <li class="jsxc_clear">%%clear_history%%</li>\
+                               </ul>\
+                           </div>\
+                           <div class="jsxc_transfer jsxc_otr jsxc_disabled"/>\
+                           <div class="jsxc_close">×</div>\
+                     </div>\
+                     <div class="jsxc_name"/>\
+                     <div class="jsxc_cycle"/>\
+                </div>\
+                <div class="jsxc_fade">\
+                   <div class="jsxc_gradient"/>\
+                   <div class="jsxc_textarea"/>\
+                   <div class="jsxc_emoticons"><ul/></div>\
+                   <input type="text" class="jsxc_textinput" placeholder="...%%Message%%" />\
+                </div>\
+            </div>\
+        </li>',
+      roster: '<div id="jsxc_roster">\
+           <ul id="jsxc_buddylist"></ul>\
+           <div class="jsxc_bottom jsxc_presence" data-bid="own">\
+              <div id="jsxc_avatar">\
+                 <div class="jsxc_avatar">☺</div>\
+              </div>\
+              <div id="jsxc_menu">\
+                 <span></span>\
+                 <ul>\
+                     <li class="jsxc_settings">%%Settings%%</li>\
+                     <li class="jsxc_muteNotification">%%Mute%%</li>\
+                     <li class="jsxc_addBuddy">%%Add_buddy%%</li>\
+                     <li class="jsxc_hideOffline">%%Hide offline%%</li>\
+                     <li class="jsxc_onlineHelp">%%Online help%%</li>\
+                     <li class="jsxc_about">%%About%%</li>\
+                 </ul>\
+              </div>\
+              <div id="jsxc_notice">\
+                 <span></span>\
+                 <ul></ul>\
+              </div>\
+              <div id="jsxc_presence">\
+                 <span>%%Online%%</span>\
+                 <ul>\
+                     <li data-pres="online" class="jsxc_online">%%Online%%</li>\
+                     <li data-pres="chat" class="jsxc_chat">%%Chatty%%</li>\
+                     <li data-pres="away" class="jsxc_away">%%Away%%</li>\
+                     <li data-pres="xa" class="jsxc_xa">%%Extended away%%</li>\
+                     <li data-pres="dnd" class="jsxc_dnd">%%dnd%%</li>\
+                     <!-- <li data-pres="offline" class="jsxc_offline">%%Offline%%</li> -->\
+                 </ul>\
+              </div>\
+           </div>\
+           <div id="jsxc_toggleRoster"></div>\
+       </div>',
+      windowList: '<div id="jsxc_windowList">\
+               <ul></ul>\
+            </div>\
+            <div id="jsxc_windowListSB">\
+               <div class="jsxc_scrollLeft jsxc_disabled"><</div>\
+               <div class="jsxc_scrollRight jsxc_disabled">></div>\
+            </div>',
+      rosterBuddy: '<li>\
+            <div class="jsxc_avatar">☺</div>\
+            <div class="jsxc_control"></div>\
+            <div class="jsxc_name"/>\
+            <div class="jsxc_options jsxc_right">\
+                <div class="jsxc_rename" title="%%rename_buddy%%">✎</div>\
+                <div class="jsxc_delete" title="%%delete_buddy%%">✘</div>\
+            </div>\
+            <div class="jsxc_options jsxc_left">\
+                <div class="jsxc_chaticon" title="%%send_message%%"/>\
+                <div class="jsxc_vcardicon" title="%%get_info%%">i</div>\
+            </div>\
+        </li>',
+      loginBox: '<h3>%%Login%%</h3>\
+        <form>\
+            <p><label for="jsxc_username">%%Username%%:</label>\
+               <input type="text" name="username" id="jsxc_username" required="required" value="{{my_node}}"/></p>\
+            <p><label for="jsxc_password">%%Password%%:</label>\
+               <input type="password" name="password" required="required" id="jsxc_password" /></p>\
+            <div class="bottom_submit_section">\
+                <input type="reset" class="button jsxc_close" name="clear" value="%%Cancel%%"/>\
+                <input type="submit" class="button creation" name="commit" value="%%Connect%%"/>\
+            </div>\
+        </form>',
+      contactDialog: '<h3>%%Add_buddy%%</h3>\
+         <p class=".jsxc_explanation">%%Type_in_the_full_username_%%</p>\
+         <form>\
+         <p><label for="jsxc_username">* %%Username%%:</label>\
+            <input type="text" name="username" id="jsxc_username" pattern="^[^\\x22&\'\\/:<>@\\s]+(@[.\\-_\\w]+)?" required="required" /></p>\
+         <p><label for="jsxc_alias">%%Alias%%:</label>\
+            <input type="text" name="alias" id="jsxc_alias" /></p>\
+         <p class="jsxc_right">\
+            <input class="button" type="submit" value="%%Add%%" />\
+         </p>\
+         <form>',
+      approveDialog: '<h3>%%Subscription_request%%</h3>\
+        <p>%%You_have_a_request_from%% <b class="jsxc_their_jid"></b>.</p>\
+        <p class="jsxc_right"><a href="#" class="button jsxc_deny">%%Deny%%</a> <a href="#" class="button creation jsxc_approve">%%Approve%%</a></p>',
+      removeDialog: '<h3>%%Remove buddy%%</h3>\
+        <p class="jsxc_maxWidth">%%You_are_about_to_remove_%%</p>\
+        <p class="jsxc_right"><a href="#" class="button jsxc_cancel jsxc_close">%%Cancel%%</a> <a href="#" class="button creation">%%Remove%%</a></p>',
+      waitAlert: '<h3>{{msg}}</h3>\
+        <p>%%Please_wait%%</p>\
+        <p class="jsxc_center"><img src="{{root}}/img/loading.gif" alt="wait" width="32px" height="32px" /></p>',
+      alert: '<h3>%%Alert%%</h3>\
+        <p>{{msg}}</p>\
+        <p class="jsxc_right"><a href="#" class="button jsxc_close jsxc_cancel">%%Ok%%</a></p>',
+      authFailDialog: '<h3>%%Login_failed%%</h3>\
+        <p>%%Sorry_we_cant_authentikate_%%</p>\
+        <p class="jsxc_right">\
+            <a class="button jsxc_cancel">%%Continue_without_chat%%</a>\
+            <a class="button creation">%%Retry%%</a>\
+        </p>',
+      confirmDialog: '<p>{{msg}}</p>\
+        <p class="jsxc_right">\
+            <a class="button jsxc_cancel jsxc_close">%%Dismiss%%</a>\
+            <a class="button creation">%%Confirm%%</a>\
+        </p>',
+      pleaseAccept: '<p>%%Please_accept_%%</p>',
+      aboutDialog: '<h3>JavaScript XMPP Chat</h3>\
+         <p><b>Version: </b>' + jsxc.version + '<br />\
+         <a href="http://jsxc.org/" target="_blank">www.jsxc.org</a><br />\
+         <br />\
+         <i>Released under the MIT license</i><br />\
+         <br />\
+         Real-time chat app for {{app_name}} and more.<br />\
+         Requires an external <a href="https://xmpp.org/xmpp-software/servers/" target="_blank">XMPP server</a>.<br />\
+         <br />\
+         <b>Credits: </b> <a href="http://www.beepzoid.com/old-phones/" target="_blank">David English (Ringtone)</a>,\
+         <a href="https://soundcloud.com/freefilmandgamemusic/ping-1?in=freefilmandgamemusic/sets/free-notification-sounds-and" target="_blank">CameronMusic (Ping)</a></p>\
+         <p class="jsxc_right"><a class="button jsxc_debuglog" href="#">Show debug log</a></p>',
+      vCard: '<h3>%%Info_about%% {{bid_name}}</h3>\
+         <ul class="jsxc_vCard"></ul>\
+         <p><img src="{{root}}/img/loading.gif" alt="wait" width="32px" height="32px" /> %%Please_wait%%...</p>',
+      settings: '<h3>%%User_settings%%</h3>\
+         <p></p>\
+         <form>\
+            <fieldset class="jsxc_fieldsetXmpp jsxc_fieldset">\
+               <legend>%%Login options%%</legend>\
+               <label for="xmpp-url">%%BOSH url%%</label><input type="text" id="xmpp-url" readonly="readonly"/><br />\
+               <label for="xmpp-username">%%Username%%</label><input type="text" id="xmpp-username"/><br />\
+               <label for="xmpp-domain">%%Domain%%</label><input type="text" id="xmpp-domain"/><br />\
+               <label for="xmpp-resource">%%Resource%%</label><input type="text" id="xmpp-resource"/><br />\
+               <label for="xmpp-onlogin">%%On login%%</label><input type="checkbox" id="xmpp-onlogin" /><br />\
+               <input type="submit" value="%%Save%%"/>\
+            </fieldset>\
+         </form>\
+         <p></p>\
+         <form>\
+            <fieldset class="jsxc_fieldsetPriority jsxc_fieldset">\
+               <legend>%%Priority%%</legend>\
+               <label for="priority-online">%%Online%%</label><input type="number" value="0" id="priority-online" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-chat">%%Chatty%%</label><input type="number" value="0" id="priority-chat" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-away">%%Away%%</label><input type="number" value="0" id="priority-away" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-xa">%%Extended_away%%</label><input type="number" value="0" id="priority-xa" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-dnd">%%dnd%%</label><input type="number" value="0" id="priority-dnd" min="-128" max="127" step="1" required="required"/><br />\
+               <input type="submit" value="%%Save%%"/>\
+            </fieldset>\
+         </form>'
+   };
+
+   /**
+    * Handle XMPP stuff.
+    * 
+    * @namespace jsxc.xmpp
+    */
+   jsxc.xmpp = {
+      conn: null, // connection
+
+      /**
+       * Create new connection or attach to old
+       * 
+       * @memberOf jsxc.xmpp
+       */
+      login: function() {
+
+         var sid = jsxc.storage.getItem('sid');
+         var rid = jsxc.storage.getItem('rid');
+         var jid = jsxc.storage.getItem('jid');
+         var url = jsxc.options.get('xmpp').url;
+
+         // Register eventlistener
+         $(document).on('connected.jsxc', jsxc.xmpp.connected);
+         $(document).on('attached.jsxc', jsxc.xmpp.attached);
+         $(document).on('disconnected.jsxc', jsxc.xmpp.disconnected);
+         $(document).on('ridChange', jsxc.xmpp.onRidChange);
+
+         Strophe.addNamespace('RECEIPTS', 'urn:xmpp:receipts');
+
+         // Create new connection (no login)
+         jsxc.xmpp.conn = new Strophe.Connection(url);
+
+         // Override default function to preserve unique id
+         var stropheGetUniqueId = jsxc.xmpp.conn.getUniqueId;
+         jsxc.xmpp.conn.getUniqueId = function(suffix) {
+            var uid = stropheGetUniqueId.call(jsxc.xmpp.conn, suffix);
+            jsxc.storage.setItem('_uniqueId', jsxc.xmpp.conn._uniqueId);
+
+            return uid;
+         };
+
+         if (jsxc.storage.getItem('debug') === true) {
+            jsxc.xmpp.conn.xmlInput = function(data) {
+               console.log('<', data);
+            };
+            jsxc.xmpp.conn.xmlOutput = function(data) {
+               console.log('>', data);
+            };
+         }
+
+         var callback = function(status, condition) {
+
+            jsxc.debug(Object.getOwnPropertyNames(Strophe.Status)[status] + ': ' + condition);
+
+            switch (status) {
+               case Strophe.Status.CONNECTED:
+                  jsxc.bid = jsxc.jidToBid(jsxc.xmpp.conn.jid.toLowerCase());
+                  $(document).trigger('connected.jsxc');
+                  break;
+               case Strophe.Status.ATTACHED:
+                  $(document).trigger('attached.jsxc');
+                  break;
+               case Strophe.Status.DISCONNECTED:
+                  $(document).trigger('disconnected.jsxc');
+                  break;
+               case Strophe.Status.CONNFAIL:
+                  jsxc.xmpp.onConnfail(condition);
+                  break;
+               case Strophe.Status.AUTHFAIL:
+                  jsxc.gui.showAuthFail();
+                  break;
+            }
+         };
+
+         if (jsxc.xmpp.conn.caps) {
+            jsxc.xmpp.conn.caps.node = 'http://jsxc.org/';
+         }
+
+         if (jsxc.restore && sid && rid) {
+            jsxc.debug('Try to attach');
+            jsxc.debug('SID: ' + sid);
+            jsxc.xmpp.conn.attach(jid, sid, rid, callback);
+         } else {
+            jsxc.debug('New connection');
+
+            jsxc.xmpp.conn.connect(jsxc.options.xmpp.jid, jsxc.options.xmpp.password, callback);
+         }
+      },
+
+      /**
+       * Logs user out of his xmpp session and does some clean up.
+       * 
+       * @returns {Boolean}
+       */
+      logout: function() {
+
+         // instruct all tabs
+         jsxc.storage.removeItem('sid');
+
+         // clean up
+         jsxc.storage.removeUserItem('buddylist');
+         jsxc.storage.removeUserItem('windowlist');
+         jsxc.storage.removeItem('_uniqueId');
+
+         if (!jsxc.master) {
+            $('#jsxc_roster').remove();
+            $('#jsxc_windowlist').remove();
+            return true;
+         }
+
+         if (jsxc.xmpp.conn === null) {
+            return true;
+         }
+
+         // Hide dropdown menu
+         $('body').click();
+
+         jsxc.triggeredFromElement = true;
+
+         // restore all otr objects
+         $.each(jsxc.storage.getUserItem('otrlist'), function(i, val) {
+            jsxc.otr.create(val);
+         });
+
+         var numOtr = Object.keys(jsxc.otr.objects).length + 1;
+         var disReady = function() {
+            if (--numOtr <= 0) {
+               jsxc.xmpp.conn.flush();
+
+               setTimeout(function() {
+                  jsxc.xmpp.conn.disconnect();
+               }, 600);
+            }
+         };
+
+         // end all private conversations
+         $.each(jsxc.otr.objects, function(key, obj) {
+            if (obj.msgstate === OTR.CONST.MSGSTATE_ENCRYPTED) {
+               obj.endOtr.call(obj, function() {
+                  obj.init.call(obj);
+                  jsxc.otr.backup(key);
+
+                  disReady();
+               });
+            } else {
+               disReady();
+            }
+         });
+
+         disReady();
+
+         // Trigger real logout in jsxc.xmpp.disconnected()
+         return false;
+      },
+
+      /**
+       * Triggered if connection is established
+       * 
+       * @private
+       */
+      connected: function() {
+
+         jsxc.xmpp.conn.pause();
+
+         var nomJid = Strophe.getBareJidFromJid(jsxc.xmpp.conn.jid).toLowerCase() + '/' + Strophe.getResourceFromJid(jsxc.xmpp.conn.jid);
+
+         // Save sid and jid
+         jsxc.storage.setItem('sid', jsxc.xmpp.conn._proto.sid);
+         jsxc.storage.setItem('jid', nomJid);
+
+         jsxc.storage.setItem('lastActivity', (new Date()).getTime());
+
+         // make shure roster will be reloaded
+         jsxc.storage.removeUserItem('buddylist');
+
+         jsxc.storage.removeUserItem('windowlist');
+         jsxc.storage.removeUserItem('own');
+         jsxc.storage.removeUserItem('avatar', 'own');
+         jsxc.storage.removeUserItem('otrlist');
+
+         // submit login form
+         if (jsxc.triggeredFromForm) {
+            // Trigger normal submit
+            jsxc.submitLoginForm();
+            return;
+         }
+
+         // reload page after login from login box
+         if (jsxc.triggeredFromBox) {
+            window.location.reload();
+            return;
+         }
+
+         jsxc.xmpp.connectionReady();
+      },
+
+      /**
+       * Triggered if connection is attached
+       * 
+       * @private
+       */
+      attached: function() {
+
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onRosterChanged, 'jabber:iq:roster', 'iq', 'set');
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onMessage, null, 'message', 'chat');
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onReceived, null, 'message');
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onPresence, null, 'presence');
+
+         // Only load roaster if necessary
+         if (!jsxc.restore || !jsxc.storage.getUserItem('buddylist')) {
+            // in order to not overide existing presence information, we send
+            // pres first after roster is ready
+            $(document).one('cloaded.roster.jsxc', jsxc.xmpp.sendPres);
+
+            $('#jsxc_roster > p:first').remove();
+
+            var iq = $iq({
+               type: 'get'
+            }).c('query', {
+               xmlns: 'jabber:iq:roster'
+            });
+
+            jsxc.xmpp.conn.sendIQ(iq, jsxc.xmpp.onRoster);
+         } else {
+            jsxc.xmpp.sendPres();
+         }
+
+         jsxc.xmpp.connectionReady();
+      },
+
+      /**
+       * Triggered if the connection is ready
+       */
+      connectionReady: function() {
+
+         // Load saved unique id
+         jsxc.xmpp.conn._uniqueId = jsxc.storage.getItem('_uniqueId') || new Date().getTime();
+
+         $(document).trigger('connectionReady.jsxc');
+      },
+
+      /**
+       * Sends presence stanza to server.
+       */
+      sendPres: function() {
+         // disco stuff
+         if (jsxc.xmpp.conn.disco) {
+            jsxc.xmpp.conn.disco.addIdentity('client', 'web', 'JSXC');
+            jsxc.xmpp.conn.disco.addFeature(Strophe.NS.DISCO_INFO);
+            jsxc.xmpp.conn.disco.addFeature(Strophe.NS.RECEIPTS);
+         }
+
+         // create presence stanza
+         var pres = $pres();
+
+         if (jsxc.xmpp.conn.caps) {
+            // attach caps
+            pres.c('c', jsxc.xmpp.conn.caps.generateCapsAttrs()).up();
+         }
+
+         var presState = jsxc.storage.getUserItem('presence') || 'online';
+         if (presState !== 'online') {
+            pres.c('show').t(presState).up();
+         }
+
+         var priority = jsxc.options.get('priority');
+         if (priority && typeof priority[presState] !== 'undefined' && parseInt(priority[presState]) !== 0) {
+            pres.c('priority').t(priority[presState]).up();
+         }
+
+         jsxc.debug('Send presence', pres.toString());
+         jsxc.xmpp.conn.send(pres);
+      },
+
+      /**
+       * Triggered if lost connection
+       * 
+       * @private
+       */
+      disconnected: function() {
+         jsxc.debug('disconnected');
+
+         jsxc.storage.removeItem('sid');
+         jsxc.storage.removeItem('rid');
+         jsxc.storage.removeItem('lastActivity');
+         jsxc.storage.removeItem('hidden');
+         jsxc.storage.removeUserItem('avatar', 'own');
+         jsxc.storage.removeUserItem('otrlist');
+
+         jsxc.xmpp.conn = null;
+
+         $('#jsxc_windowList').remove();
+
+         if (jsxc.triggeredFromElement) {
+            $('#jsxc_roster').remove();
+
+            if (jsxc.triggeredFromLogout) {
+               window.location = jsxc.options.logoutElement.attr('href');
+            }
+         } else {
+            jsxc.gui.roster.noConnection();
+         }
+
+         window.clearInterval(jsxc.keepalive);
+      },
+
+      /**
+       * Triggered on connection fault
+       * 
+       * @param {String} condition information why we lost the connection
+       * @private
+       */
+      onConnfail: function(condition) {
+         jsxc.debug('XMPP connection failed: ' + condition);
+
+         if (jsxc.triggeredFromForm) {
+            jsxc.submitLoginForm();
+         }
+      },
+
+      /**
+       * Triggered on initial roster load
+       * 
+       * @param {dom} iq
+       * @private
+       */
+      onRoster: function(iq) {
+         /*
+          * <iq from='' type='get' id=''> <query xmlns='jabber:iq:roster'> <item
+          * jid='' name='' subscription='' /> ... </query> </iq>
+          */
+
+         jsxc.debug('Load roster', iq);
+
+         var buddies = [];
+
+         $(iq).find('item').each(function() {
+            var jid = $(this).attr('jid');
+            var name = $(this).attr('name') || jid;
+            var bid = jsxc.jidToBid(jid);
+            var sub = $(this).attr('subscription');
+
+            buddies.push(bid);
+
+            jsxc.storage.removeUserItem('res', bid);
+
+            jsxc.storage.saveBuddy(bid, {
+               jid: jid,
+               name: name,
+               status: 0,
+               sub: sub,
+               res: []
+            });
+
+            jsxc.gui.roster.add(bid);
+         });
+
+         if (buddies.length === 0) {
+            jsxc.gui.roster.empty();
+         }
+
+         jsxc.storage.setUserItem('buddylist', buddies);
+
+         jsxc.debug('Roster loaded');
+         $(document).trigger('cloaded.roster.jsxc');
+      },
+
+      /**
+       * Triggerd on roster changes
+       * 
+       * @param {dom} iq
+       * @returns {Boolean} True to preserve handler
+       * @private
+       */
+      onRosterChanged: function(iq) {
+         /*
+          * <iq from='' type='set' id=''> <query xmlns='jabber:iq:roster'> <item
+          * jid='' name='' subscription='' /> </query> </iq>
+          */
+
+         jsxc.debug('onRosterChanged', iq);
+
+         $(iq).find('item').each(function() {
+            var jid = $(this).attr('jid');
+            var name = $(this).attr('name') || jid;
+            var bid = jsxc.jidToBid(jid);
+            var sub = $(this).attr('subscription');
+            // var ask = $(this).attr('ask');
+
+            if (sub === 'remove') {
+               jsxc.gui.roster.purge(bid);
+            } else {
+               var bl = jsxc.storage.getUserItem('buddylist');
+
+               if (bl.indexOf(bid) < 0) {
+                  bl.push(bid); // (INFO) push returns the new length
+                  jsxc.storage.setUserItem('buddylist', bl);
+               }
+
+               var temp = jsxc.storage.saveBuddy(bid, {
+                  jid: jid,
+                  name: name,
+                  sub: sub
+               });
+
+               if (temp === 'updated') {
+
+                  jsxc.gui.update(bid);
+                  jsxc.gui.roster.reorder(bid);
+               } else {
+                  jsxc.gui.roster.add(bid);
+               }
+            }
+
+            // Remove pending friendship request from notice list
+            if (sub === 'from' || sub === 'both') {
+               var notices = jsxc.storage.getUserItem('notices');
+               var noticeKey = null, notice;
+
+               for (noticeKey in notices) {
+                  notice = notices[noticeKey];
+
+                  if (notice.fnName === 'gui.showApproveDialog' && notice.fnParams[0] === jid) {
+                     jsxc.debug('Remove notice with key ' + noticeKey);
+
+                     jsxc.notice.remove(noticeKey);
+                  }
+               }
+            }
+         });
+
+         if (!jsxc.storage.getUserItem('buddylist') || jsxc.storage.getUserItem('buddylist').length === 0) {
+            jsxc.gui.roster.empty();
+         } else {
+            $('#jsxc_roster > p:first').remove();
+         }
+
+         // preserve handler
+         return true;
+      },
+
+      /**
+       * Triggered on incoming presence stanzas
+       * 
+       * @param {dom} presence
+       * @private
+       */
+      onPresence: function(presence) {
+         /*
+          * <presence xmlns='jabber:client' type='unavailable' from='' to=''/>
+          * 
+          * <presence xmlns='jabber:client' from='' to=''> <priority>5</priority>
+          * <c xmlns='http://jabber.org/protocol/caps'
+          * node='http://psi-im.org/caps' ver='caps-b75d8d2b25' ext='ca cs
+          * ep-notify-2 html'/> </presence>
+          * 
+          * <presence xmlns='jabber:client' from='' to=''> <show>chat</show>
+          * <status></status> <priority>5</priority> <c
+          * xmlns='http://jabber.org/protocol/caps'
+          * node='http://psi-im.org/caps' ver='caps-b75d8d2b25' ext='ca cs
+          * ep-notify-2 html'/> </presence>
+          */
+         jsxc.debug('onPresence', presence);
+
+         var ptype = $(presence).attr('type');
+         var from = $(presence).attr('from');
+         var jid = Strophe.getBareJidFromJid(from).toLowerCase();
+         var to = $(presence).attr('to');
+         to = (to) ? Strophe.getBareJidFromJid(to).toLowerCase() : jid;
+         var r = Strophe.getResourceFromJid(from);
+         var bid = jsxc.jidToBid(jid);
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var res = jsxc.storage.getUserItem('res', bid) || {};
+         var status = null;
+         var xVCard = $(presence).find('x[xmlns="vcard-temp:x:update"]');
+
+         if (jid === to) {
+            return true;
+         }
+
+         if (ptype === 'error') {
+            jsxc.error('[XMPP] ' + $(presence).attr('code'));
+            return true;
+         }
+
+         // incoming friendship request
+         if (ptype === 'subscribe') {
+            jsxc.storage.setUserItem('friendReq', {
+               jid: jid,
+               approve: -1
+            });
+            jsxc.notice.add('%%Friendship request%%', '%%from%% ' + jid, 'gui.showApproveDialog', [ jid ]);
+
+            return true;
+         } else if (ptype === 'unavailable' || ptype === 'unsubscribed') {
+            status = jsxc.CONST.STATUS.indexOf('offline');
+         } else {
+            var show = $(presence).find('show').text();
+            if (show === '') {
+               status = jsxc.CONST.STATUS.indexOf('online');
+            } else {
+               status = jsxc.CONST.STATUS.indexOf(show);
+            }
+         }
+
+         if (status === 0) {
+            delete res[r];
+         } else {
+            res[r] = status;
+         }
+
+         var maxVal = [];
+         var max = 0, prop = null;
+         for (prop in res) {
+            if (res.hasOwnProperty(prop)) {
+               if (max <= res[prop]) {
+                  if (max !== res[prop]) {
+                     maxVal = [];
+                     max = res[prop];
+                  }
+                  maxVal.push(prop);
+               }
+            }
+         }
+
+         if (data.status === 0 && max > 0) {
+            // buddy has come online
+            jsxc.notification.notify(data.name, jsxc.translate('%%has come online%%.'));
+         }
+
+         data.status = max;
+         data.res = maxVal;
+         data.jid = jid;
+
+         // Looking for avatar
+         if (xVCard.length > 0) {
+            var photo = xVCard.find('photo');
+
+            if (photo.length > 0 && photo.text() !== data.avatar) {
+               jsxc.storage.removeUserItem('avatar', data.avatar);
+               data.avatar = photo.text();
+            }
+         }
+
+         // Reset jid
+         if (jsxc.gui.window.get(bid).length > 0) {
+            jsxc.gui.window.get(bid).data('jid', jid);
+         }
+
+         jsxc.storage.setUserItem('buddy', bid, data);
+         jsxc.storage.setUserItem('res', bid, res);
+
+         jsxc.debug('Presence (' + from + '): ' + status);
+
+         jsxc.gui.update(bid);
+         jsxc.gui.roster.reorder(bid);
+
+         $(document).trigger('presence.jsxc', [ from, status, presence ]);
+
+         // preserve handler
+         return true;
+      },
+
+      /**
+       * Triggered on incoming message stanzas
+       * 
+       * @param {dom} presence
+       * @returns {Boolean}
+       * @private
+       */
+      onMessage: function(message) {
+         /*
+          * <message xmlns='jabber:client' type='chat' to='' id='' from=''>
+          * <body>...</body> <active
+          * xmlns='http://jabber.org/protocol/chatstates'/> </message>
+          */
+
+         jsxc.debug('Incoming message', message);
+
+         var type = $(message).attr('type');
+         var from = $(message).attr('from');
+         var mid = $(message).attr('id');
+         var jid = Strophe.getBareJidFromJid(from);
+         var bid = jsxc.jidToBid(jid);
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var body = $(message).find('body:first').text();
+         var request = $(message).find("request[xmlns='urn:xmpp:receipts']");
+
+         if (!body) {
+            return true;
+         }
+
+         if (data === null) {
+            // jid not in roster
+
+            var chat = jsxc.storage.getUserItem('chat', bid) || [];
+
+            if (chat.length === 0) {
+               jsxc.notice.add('%%Unknown sender%%', '%%You received a message from an unknown sender%% (' + bid + ').', 'gui.showUnknownSender', [ bid ]);
+            }
+
+            var msg = jsxc.removeHTML(body);
+            msg = jsxc.escapeHTML(msg);
+
+            jsxc.storage.saveMessage(bid, 'in', msg);
+
+            return true;
+         }
+
+         var win = jsxc.gui.window.init(bid);
+
+         // If we now the full jid, we use it
+         if (type === 'chat') {
+            win.data('jid', from);
+            jsxc.storage.updateUserItem('buddy', bid, {
+               jid: from
+            });
+         }
+
+         $(document).trigger('message.jsxc', [ from, body ]);
+
+         // create related otr object
+         if (jsxc.master && !jsxc.otr.objects[bid]) {
+            jsxc.otr.create(bid);
+         }
+
+         if (mid !== null && request.length && data !== null && (data.sub === 'both' || data.sub === 'from') && type === 'chat') {
+            // Send received according to XEP-0184
+            jsxc.xmpp.conn.send($msg({
+               to: from
+            }).c('received', {
+               xmlns: 'urn:xmpp:receipts',
+               id: mid
+            }));
+         }
+
+         if (jsxc.otr.objects.hasOwnProperty(bid)) {
+            jsxc.otr.objects[bid].receiveMsg(body);
+         } else {
+            jsxc.gui.window.postMessage(bid, 'in', body);
+         }
+
+         // preserve handler
+         return true;
+      },
+
+      /**
+       * Triggerd if the rid changed
+       * 
+       * @param {event} ev
+       * @param {obejct} data
+       * @private
+       */
+      onRidChange: function(ev, data) {
+         jsxc.storage.setItem('rid', data.rid);
+      },
+
+      /**
+       * response to friendship request
+       * 
+       * @param {string} from jid from original friendship req
+       * @param {boolean} approve
+       */
+      resFriendReq: function(from, approve) {
+         if (jsxc.master) {
+            jsxc.xmpp.conn.send($pres({
+               to: from,
+               type: (approve) ? 'subscribed' : 'unsubscribed'
+            }));
+
+            jsxc.storage.removeUserItem('friendReq');
+            jsxc.gui.dialog.close();
+
+         } else {
+            jsxc.storage.updateUserItem('friendReq', 'approve', approve);
+         }
+      },
+
+      /**
+       * Add buddy to my friends
+       * 
+       * @param {string} username jid
+       * @param {string} alias
+       */
+      addBuddy: function(username, alias) {
+         var bid = jsxc.jidToBid(username);
+
+         if (jsxc.master) {
+            // add buddy to roster (trigger onRosterChanged)
+            var iq = $iq({
+               type: 'set'
+            }).c('query', {
+               xmlns: 'jabber:iq:roster'
+            }).c('item', {
+               jid: username,
+               name: alias || ''
+            });
+            jsxc.xmpp.conn.sendIQ(iq);
+
+            // send subscription request to buddy (trigger onRosterChanged)
+            jsxc.xmpp.conn.send($pres({
+               to: username,
+               type: 'subscribe'
+            }));
+
+            jsxc.storage.removeUserItem('add_' + bid);
+         } else {
+            jsxc.storage.setUserItem('add_' + bid, {
+               username: username,
+               alias: alias || null
+            });
+         }
+      },
+
+      /**
+       * Remove buddy from my friends
+       * 
+       * @param {type} jid
+       */
+      removeBuddy: function(jid) {
+         var bid = jsxc.jidToBid(jid);
+
+         // Shortcut to remove buddy from roster and cancle all subscriptions
+         var iq = $iq({
+            type: 'set'
+         }).c('query', {
+            xmlns: 'jabber:iq:roster'
+         }).c('item', {
+            jid: Strophe.getBareJidFromJid(jid),
+            subscription: 'remove'
+         });
+         jsxc.xmpp.conn.sendIQ(iq);
+
+         jsxc.gui.roster.purge(bid);
+      },
+
+      onReceived: function(message) {
+         var from = $(message).attr('from');
+         var jid = Strophe.getBareJidFromJid(from);
+         var bid = jsxc.jidToBid(jid);
+         var received = $(message).find("received[xmlns='urn:xmpp:receipts']");
+
+         if (received.length) {
+            var receivedId = received.attr('id').replace(/:/, '-');
+            var chat = jsxc.storage.getUserItem('chat', bid);
+            var i;
+
+            for (i = chat.length - 1; i >= 0; i--) {
+               if (chat[i].uid === receivedId) {
+                  chat[i].received = true;
+
+                  $('#' + receivedId).addClass('jsxc_received');
+
+                  jsxc.storage.setUserItem('chat', bid, chat);
+                  break;
+               }
+            }
+         }
+
+         return true;
+      },
+
+      /**
+       * Public function to send message.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param bid css jid of user
+       * @param msg message
+       * @param uid unique id
+       */
+      sendMessage: function(bid, msg, uid) {
+         if (jsxc.otr.objects.hasOwnProperty(bid)) {
+            jsxc.otr.objects[bid].sendMsg(msg, uid);
+         } else {
+            jsxc.xmpp._sendMessage(jsxc.gui.window.get(bid).data('jid'), msg, uid);
+         }
+      },
+
+      /**
+       * Create message stanza and send it.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param jid Jabber id
+       * @param msg Message
+       * @param uid unique id
+       * @private
+       */
+      _sendMessage: function(jid, msg, uid) {
+         var data = jsxc.storage.getUserItem('buddy', jsxc.jidToBid(jid)) || {};
+         var isBar = (Strophe.getBareJidFromJid(jid) === jid);
+         var type = data.type || 'chat';
+
+         var xmlMsg = $msg({
+            to: jid,
+            type: type,
+            id: uid
+         }).c('body').t(msg);
+
+         if (type === 'chat' && (isBar || jsxc.xmpp.conn.caps.hasFeatureByJid(jid, Strophe.NS.RECEIPTS))) {
+            // Add request according to XEP-0184
+            xmlMsg.up().c('request', {
+               xmlns: 'urn:xmpp:receipts'
+            });
+         }
+
+         jsxc.xmpp.conn.send(xmlMsg);
+      },
+
+      /**
+       * This function loads a vcard.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param bid
+       * @param cb
+       * @param error_cb
+       */
+      loadVcard: function(bid, cb, error_cb) {
+         if (jsxc.master) {
+            jsxc.xmpp.conn.vcard.get(cb, bid, error_cb);
+         } else {
+            jsxc.storage.setUserItem('vcard', bid, 'request:' + (new Date()).getTime());
+
+            $(document).one('loaded.vcard.jsxc', function(ev, result) {
+               if (result && result.state === 'success') {
+                  cb($(result.data).get(0));
+               } else {
+                  error_cb();
+               }
+            });
+         }
+      },
+
+      /**
+       * Retrieves capabilities.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param jid
+       * @returns List of known capabilities
+       */
+      getCapabilitiesByJid: function(jid) {
+         if (jsxc.xmpp.conn) {
+            return jsxc.xmpp.conn.caps.getCapabilitiesByJid(jid);
+         }
+
+         var jidVerIndex = JSON.parse(localStorage.getItem('strophe.caps._jidVerIndex')) || {};
+         var knownCapabilities = JSON.parse(localStorage.getItem('strophe.caps._knownCapabilities')) || {};
+
+         if (jidVerIndex[jid]) {
+            return knownCapabilities[jidVerIndex[jid]];
+         }
+
+         return null;
+      }
+   };
+
+   /**
+    * Handle long-live data
+    * 
+    * @namespace jsxc.storage
+    */
+   jsxc.storage = {
+      /**
+       * Prefix for localstorage
+       * 
+       * @privat
+       */
+      PREFIX: 'jsxc',
+
+      SEP: ':',
+
+      /**
+       * @param {type} uk Should we generate a user prefix?
+       * @returns {String} prefix
+       * @memberOf jsxc.storage
+       */
+      getPrefix: function(uk) {
+         var self = jsxc.storage;
+
+         return self.PREFIX + self.SEP + ((uk && jsxc.bid) ? jsxc.bid + self.SEP : '');
+      },
+
+      /**
+       * Save item to storage
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {Object} value value
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      setItem: function(key, value, uk) {
+
+         // Workaround for non-conform browser
+         if (jsxc.storageNotConform > 0 && key !== 'rid' && key !== 'lastActivity') {
+            if (jsxc.storageNotConform > 1 && jsxc.toSNC === null) {
+               jsxc.toSNC = window.setTimeout(function() {
+                  jsxc.storageNotConform = 0;
+                  jsxc.storage.setItem('storageNotConform', 0);
+               }, 1000);
+            }
+
+            jsxc.ls.push(JSON.stringify({
+               key: key,
+               value: value
+            }));
+         }
+
+         if (typeof (value) === 'object') {
+            value = JSON.stringify(value);
+         }
+
+         localStorage.setItem(jsxc.storage.getPrefix(uk) + key, value);
+      },
+
+      setUserItem: function(type, key, value) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 2) {
+            value = key;
+            key = type;
+            type = '';
+         } else if (arguments.length === 3) {
+            key = type + self.SEP + key;
+         }
+
+         return jsxc.storage.setItem(key, value, true);
+      },
+
+      /**
+       * Load item from storage
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      getItem: function(key, uk) {
+         key = jsxc.storage.getPrefix(uk) + key;
+
+         var value = localStorage.getItem(key);
+         try {
+            return JSON.parse(value);
+         } catch (e) {
+            return value;
+         }
+      },
+
+      /**
+       * Get a user item from storage.
+       * 
+       * @param key
+       * @returns user item
+       */
+      getUserItem: function(type, key) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 1) {
+            key = type;
+         } else if (arguments.length === 2) {
+            key = type + self.SEP + key;
+         }
+
+         return jsxc.storage.getItem(key, true);
+      },
+
+      /**
+       * Remove item from storage
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      removeItem: function(key, uk) {
+
+         // Workaround for non-conform browser
+         if (jsxc.storageNotConform && key !== 'rid' && key !== 'lastActivity') {
+            jsxc.ls.push(JSON.stringify({
+               key: jsxc.storage.prefix + key,
+               value: ''
+            }));
+         }
+
+         localStorage.removeItem(jsxc.storage.getPrefix(uk) + key);
+      },
+
+      /**
+       * Remove user item from storage.
+       * 
+       * @param key
+       */
+      removeUserItem: function(type, key) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 1) {
+            key = type;
+         } else if (arguments.length === 2) {
+            key = type + self.SEP + key;
+         }
+
+         jsxc.storage.removeItem(key, true);
+      },
+
+      /**
+       * Updates value of a variable in a saved object.
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String|object} variable variablename in object or object with
+       *        variable/key pairs
+       * @param {Object} [value] value
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      updateItem: function(key, variable, value, uk) {
+
+         var data = jsxc.storage.getItem(key, uk) || {};
+
+         if (typeof (variable) === 'object') {
+
+            $.each(variable, function(key, val) {
+               if (typeof (data[key]) === 'undefined') {
+                  jsxc.debug('Variable ' + key + ' doesn\'t exist in ' + variable + '. It was created.');
+               }
+
+               data[key] = val;
+            });
+         } else {
+            if (typeof (data[variable]) === 'undefined') {
+               jsxc.debug('Variable ' + variable + ' doesn\'t exist. It was created.');
+            }
+
+            data[variable] = value;
+         }
+
+         jsxc.storage.setItem(key, data, uk);
+      },
+
+      /**
+       * Updates value of a variable in a saved user object.
+       * 
+       * @param {String} key variablename
+       * @param {String|object} variable variablename in object or object with
+       *        variable/key pairs
+       * @param {Object} [value] value
+       */
+      updateUserItem: function(type, key, variable, value) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 4 || (arguments.length === 3 && typeof variable === 'object')) {
+            key = type + self.SEP + key;
+         } else {
+            value = variable;
+            variable = key;
+            key = type;
+         }
+
+         return jsxc.storage.updateItem(key, variable, value, true);
+      },
+
+      /**
+       * Inkrements value
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      ink: function(key, uk) {
+
+         jsxc.storage.setItem(key, Number(jsxc.storage.getItem(key, uk)) + 1, uk);
+      },
+
+      /**
+       * Remove element from array or object
+       * 
+       * @param {string} key name of array or object
+       * @param {string} name name of element in array or object
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       * @returns {undefined}
+       */
+      removeElement: function(key, name, uk) {
+         var item = jsxc.storage.getItem(key, uk);
+
+         if ($.isArray(item)) {
+            item = $.grep(item, function(e) {
+               return e !== name;
+            });
+         } else if (typeof (item) === 'object') {
+            delete item[name];
+         }
+
+         jsxc.storage.setItem(key, item, uk);
+      },
+
+      removeUserElement: function(type, key, name) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 2) {
+            name = key;
+            key = type;
+         } else if (arguments.length === 3) {
+            key = type + self.SEP + key;
+         }
+
+         return jsxc.storage.removeElement(key, name, true);
+      },
+
+      /**
+       * Triggered if changes are recognized
+       * 
+       * @function
+       * @param {event} e Storageevent
+       * @param {String} e.key Keyname which triggered event
+       * @param {Object} e.oldValue Old Value for key
+       * @param {Object} e.newValue New Value for key
+       * @param {String} e.url
+       */
+      onStorage: function(e) {
+
+         // skip
+         if (e.key === jsxc.storage.PREFIX + jsxc.storage.SEP + 'rid' || e.key === jsxc.storage.PREFIX + jsxc.storage.SEP + 'lastActivity') {
+            return;
+         }
+
+         var re = new RegExp('^' + jsxc.storage.PREFIX + jsxc.storage.SEP + '(?:[^' + jsxc.storage.SEP + ']+@[^' + jsxc.storage.SEP + ']+' + jsxc.storage.SEP + ')?(.*)', 'i');
+         var key = e.key.replace(re, '$1');
+
+         // Workaround for non-conform browser: Triggered event on every page
+         // (own)
+         if (jsxc.storageNotConform > 0 && jsxc.ls.length > 0) {
+
+            var val = e.newValue;
+            try {
+               val = JSON.parse(val);
+            } catch (err) {
+            }
+
+            var index = $.inArray(JSON.stringify({
+               key: key,
+               value: val
+            }), jsxc.ls);
+
+            if (index >= 0) {
+
+               // confirm that the storage event is not fired regularly
+               if (jsxc.storageNotConform > 1) {
+                  window.clearTimeout(jsxc.toSNC);
+                  jsxc.storageNotConform = 1;
+                  jsxc.storage.setItem('storageNotConform', 1);
+               }
+
+               jsxc.ls.splice(index, 1);
+               return;
+            }
+         }
+
+         // Workaround for non-conform browser
+         if (e.oldValue === e.newValue) {
+            return;
+         }
+
+         var n, o;
+         var bid = key.replace(new RegExp('[^' + jsxc.storage.SEP + ']+' + jsxc.storage.SEP + '(.*)', 'i'), '$1');
+
+         // react if someone ask, if there is a master
+         if (jsxc.master && key === 'alive') {
+            jsxc.debug('Master request.');
+
+            jsxc.storage.ink('alive');
+            return;
+         }
+
+         // master alive
+         if (!jsxc.master && (key === 'alive' || key === 'alive_busy') && !jsxc.triggeredFromElement) {
+
+            // reset timeout
+            window.clearTimeout(jsxc.to);
+            jsxc.to = window.setTimeout(jsxc.checkMaster, ((key === 'alive') ? jsxc.options.timeout : jsxc.options.busyTimeout) + jsxc.random(60));
+
+            // only call the first time
+            if (!jsxc.role_allocation) {
+               jsxc.onSlave();
+            }
+
+            return;
+         }
+
+         if (key.match(/^notices/)) {
+            jsxc.notice.load();
+         }
+
+         if (key.match(/^presence/)) {
+            jsxc.gui.changePresence(e.newValue, true);
+         }
+
+         if (key.match(/^options/) && e.newValue) {
+            n = JSON.parse(e.newValue);
+
+            if (typeof n.muteNotification !== 'undefined' && n.muteNotification) {
+               jsxc.notification.muteSound(true);
+            } else {
+               jsxc.notification.unmuteSound(true);
+            }
+         }
+
+         if (key.match(/^hidden/)) {
+            if (jsxc.master) {
+               clearTimeout(jsxc.toNotification);
+            } else {
+               jsxc.isHidden();
+            }
+         }
+
+         if (key.match(new RegExp('^chat' + jsxc.storage.SEP))) {
+
+            var posts = JSON.parse(e.newValue);
+            var data, el;
+
+            while (posts.length > 0) {
+               data = posts.pop();
+               el = $('#' + data.uid);
+
+               if (el.length === 0) {
+                  if (jsxc.master && data.direction === 'out') {
+                     jsxc.xmpp.sendMessage(bid, data.msg, data.uid);
+                  }
+
+                  jsxc.gui.window._postMessage(bid, data);
+               } else if (data.received) {
+                  el.addClass('jsxc_received');
+               }
+            }
+            return;
+         }
+
+         if (key.match(new RegExp('^window' + jsxc.storage.SEP))) {
+
+            if (!e.newValue) {
+               jsxc.gui.window._close(bid);
+               return;
+            }
+
+            if (!e.oldValue) {
+               jsxc.gui.window.open(bid);
+               return;
+            }
+
+            n = JSON.parse(e.newValue);
+
+            if (n.minimize) {
+               jsxc.gui.window._hide(bid);
+            } else {
+               jsxc.gui.window._show(bid);
+            }
+
+            jsxc.gui.window.setText(bid, n.text);
+
+            return;
+         }
+
+         if (key.match(new RegExp('^smp' + jsxc.storage.SEP))) {
+
+            if (!e.newValue) {
+
+               jsxc.gui.dialog.close();
+
+               if (jsxc.master) {
+                  jsxc.otr.objects[bid].sm.abort();
+               }
+
+               return;
+            }
+
+            n = JSON.parse(e.newValue);
+
+            if (typeof (n.data) !== 'undefined') {
+
+               jsxc.otr.onSmpQuestion(bid, n.data);
+
+            } else if (jsxc.master && n.sec) {
+               jsxc.gui.dialog.close();
+
+               jsxc.otr.sendSmpReq(bid, n.sec, n.quest);
+            }
+         }
+
+         if (!jsxc.master && key.match(new RegExp('^buddy' + jsxc.storage.SEP))) {
+
+            if (!e.newValue) {
+               jsxc.gui.roster.purge(bid);
+               return;
+            }
+            if (!e.oldValue) {
+               jsxc.gui.roster.add(bid);
+               return;
+            }
+
+            n = JSON.parse(e.newValue);
+            o = JSON.parse(e.oldValue);
+
+            jsxc.gui.update(bid);
+
+            if (o.status !== n.status || o.sub !== n.sub) {
+               jsxc.gui.roster.reorder(bid);
+            }
+         }
+
+         if (jsxc.master && key.match(new RegExp('^deletebuddy' + jsxc.storage.SEP)) && e.newValue) {
+            n = JSON.parse(e.newValue);
+
+            jsxc.xmpp.removeBuddy(n.jid);
+            jsxc.storage.removeUserItem(key);
+         }
+
+         if (jsxc.master && key.match(new RegExp('^buddy' + jsxc.storage.SEP))) {
+
+            n = JSON.parse(e.newValue);
+            o = JSON.parse(e.oldValue);
+
+            if (o.transferReq !== n.transferReq) {
+               jsxc.storage.updateUserItem('buddy', bid, 'transferReq', -1);
+
+               if (n.transferReq === 0) {
+                  jsxc.otr.goPlain(bid);
+               }
+               if (n.transferReq === 1) {
+                  jsxc.otr.goEncrypt(bid);
+               }
+            }
+
+            if (o.name !== n.name) {
+               jsxc.gui.roster._rename(bid, n.name);
+            }
+         }
+
+         // logout
+         if (key === 'sid') {
+            if (!e.newValue) {
+               // if (jsxc.master && jsxc.xmpp.conn) {
+               // jsxc.xmpp.conn.disconnect();
+               // jsxc.triggeredFromElement = true;
+               // }
+               jsxc.xmpp.logout();
+
+            }
+            return;
+         }
+
+         if (key === 'friendReq') {
+            n = JSON.parse(e.newValue);
+
+            if (jsxc.master && n.approve >= 0) {
+               jsxc.xmpp.resFriendReq(n.jid, n.approve);
+            }
+         }
+
+         if (jsxc.master && key.match(new RegExp('^add' + jsxc.storage.SEP))) {
+            n = JSON.parse(e.newValue);
+
+            jsxc.xmpp.addBuddy(n.username, n.alias);
+         }
+
+         if (key === 'roster') {
+            jsxc.gui.roster.toggle();
+         }
+
+         if (jsxc.master && key.match(new RegExp('^vcard' + jsxc.storage.SEP)) && e.newValue !== null && e.newValue.match(/^request:/)) {
+
+            jsxc.xmpp.loadVcard(bid, function(stanza) {
+               jsxc.storage.setUserItem('vcard', bid, {
+                  state: 'success',
+                  data: $('<div>').append(stanza).html()
+               });
+            }, function() {
+               jsxc.storage.setUserItem('vcard', bid, {
+                  state: 'error'
+               });
+            });
+         }
+
+         if (!jsxc.master && key.match(new RegExp('^vcard' + jsxc.storage.SEP)) && e.newValue !== null && !e.newValue.match(/^request:/)) {
+            n = JSON.parse(e.newValue);
+
+            if (typeof n.state !== 'undefined') {
+               $(document).trigger('loaded.vcard.jsxc', n);
+            }
+
+            jsxc.storage.removeUserItem('vcard', bid);
+         }
+      },
+
+      /**
+       * Save message to storage.
+       * 
+       * @memberOf jsxc.storage
+       * @param bid
+       * @param direction
+       * @param msg
+       * @return post
+       */
+      saveMessage: function(bid, direction, msg) {
+         var chat = jsxc.storage.getUserItem('chat', bid) || [];
+
+         var uid = new Date().getTime() + ':msg';
+
+         if (chat.length > jsxc.options.get('numberOfMsg')) {
+            chat.pop();
+         }
+
+         var post = {
+            direction: direction,
+            msg: msg,
+            uid: uid.replace(/:/, '-'),
+            received: false
+         };
+
+         chat.unshift(post);
+         jsxc.storage.setUserItem('chat', bid, chat);
+
+         return post;
+      },
+
+      /**
+       * Save or update buddy data.
+       * 
+       * @memberOf jsxc.storage
+       * @param bid
+       * @param data
+       * @returns {String} Updated or created
+       */
+      saveBuddy: function(bid, data) {
+
+         if (jsxc.storage.getUserItem('buddy', bid)) {
+            jsxc.storage.updateUserItem('buddy', bid, data);
+
+            return 'updated';
+         }
+
+         jsxc.storage.setUserItem('buddy', bid, $.extend({
+            jid: '',
+            name: '',
+            status: 0,
+            sub: 'none',
+            msgstate: 0,
+            transferReq: -1,
+            trust: false,
+            fingerprint: null,
+            res: [],
+            type: 'chat'
+         }, data));
+
+         return 'created';
+      }
+   };
+
+   /**
+    * @namespace jsxc.otr
+    */
+   jsxc.otr = {
+      /** list of otr objects */
+      objects: {},
+
+      dsaFallback: null,
+      /**
+       * Handler for otr receive event
+       * 
+       * @memberOf jsxc.otr
+       * @param {string} bid
+       * @param {string} msg received message
+       * @param {string} encrypted True, if msg was encrypted.
+       */
+      receiveMessage: function(bid, msg, encrypted) {
+
+         if (jsxc.otr.objects[bid].msgstate !== OTR.CONST.MSGSTATE_PLAINTEXT) {
+            jsxc.otr.backup(bid);
+         }
+
+         if (jsxc.otr.objects[bid].msgstate !== OTR.CONST.MSGSTATE_PLAINTEXT && !encrypted) {
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.translate('%%Received an unencrypted message.%% [') + msg + ']');
+         } else {
+            jsxc.gui.window.postMessage(bid, 'in', msg);
+         }
+      },
+
+      /**
+       * Handler for otr send event
+       * 
+       * @param {string} jid
+       * @param {string} msg message to be send
+       */
+      sendMessage: function(jid, msg, uid) {
+         if (jsxc.otr.objects[jsxc.jidToBid(jid)].msgstate !== 0) {
+            jsxc.otr.backup(jsxc.jidToBid(jid));
+         }
+
+         jsxc.xmpp._sendMessage(jid, msg, uid);
+      },
+
+      /**
+       * Create new otr instance
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      create: function(bid) {
+
+         if (jsxc.otr.objects.hasOwnProperty(bid)) {
+            return;
+         }
+
+         if (!jsxc.options.otr.priv) {
+            return;
+         }
+
+         // save list of otr objects
+         var ol = jsxc.storage.getUserItem('otrlist') || [];
+         if (ol.indexOf(bid) < 0) {
+            ol.push(bid);
+            jsxc.storage.setUserItem('otrlist', ol);
+         }
+
+         jsxc.otr.objects[bid] = new OTR(jsxc.options.otr);
+
+         if (jsxc.options.otr.SEND_WHITESPACE_TAG) {
+            jsxc.otr.objects[bid].SEND_WHITESPACE_TAG = true;
+         }
+
+         if (jsxc.options.otr.WHITESPACE_START_AKE) {
+            jsxc.otr.objects[bid].WHITESPACE_START_AKE = true;
+         }
+
+         jsxc.otr.objects[bid].on('status', function(status) {
+            var data = jsxc.storage.getUserItem('buddy', bid);
+
+            if (data === null) {
+               return;
+            }
+
+            switch (status) {
+               case OTR.CONST.STATUS_SEND_QUERY:
+                  jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.trying_to_start_private_conversation);
+                  break;
+               case OTR.CONST.STATUS_AKE_SUCCESS:
+                  data.fingerprint = jsxc.otr.objects[bid].their_priv_pk.fingerprint();
+                  data.msgstate = OTR.CONST.MSGSTATE_ENCRYPTED;
+
+                  var msg = (jsxc.otr.objects[bid].trust ? jsxc.l.Verified : jsxc.l.Unverified) + ' ' + jsxc.l.private_conversation_started;
+                  jsxc.gui.window.postMessage(bid, 'sys', msg);
+                  break;
+               case OTR.CONST.STATUS_END_OTR:
+                  data.fingerprint = null;
+
+                  if (jsxc.otr.objects[bid].msgstate === OTR.CONST.MSGSTATE_PLAINTEXT) {
+                     // we abort the private conversation
+
+                     data.msgstate = OTR.CONST.MSGSTATE_PLAINTEXT;
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.private_conversation_aborted);
+
+                  } else {
+                     // the buddy abort the private conversation
+
+                     data.msgstate = OTR.CONST.MSGSTATE_FINISHED;
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.your_buddy_closed_the_private_conversation_you_should_do_the_same);
+                  }
+                  break;
+               case OTR.CONST.STATUS_SMP_HANDLE:
+                  jsxc.keepBusyAlive();
+                  break;
+            }
+
+            jsxc.storage.setUserItem('buddy', bid, data);
+
+            // for encryption and verification state
+            jsxc.gui.update(bid);
+         });
+
+         jsxc.otr.objects[bid].on('smp', function(type, data) {
+            switch (type) {
+               case 'question': // verification request received
+                  jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.Authentication_request_received);
+
+                  if ($('#jsxc_dialog').length > 0) {
+                     jsxc.otr.objects[bid].sm.abort();
+                     break;
+                  }
+
+                  jsxc.otr.onSmpQuestion(bid, data);
+                  jsxc.storage.setUserItem('smp_' + bid, {
+                     data: data || null
+                  });
+
+                  break;
+               case 'trust': // verification completed
+                  jsxc.otr.objects[bid].trust = data;
+                  jsxc.storage.updateUserItem('buddy', bid, 'trust', data);
+                  jsxc.otr.backup(bid);
+                  jsxc.gui.update(bid);
+
+                  if (data) {
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.conversation_is_now_verified);
+                  } else {
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.authentication_failed);
+                  }
+                  jsxc.storage.removeUserItem('smp_' + bid);
+                  jsxc.gui.dialog.close();
+                  break;
+               case 'abort':
+                  jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.Authentication_aborted);
+                  break;
+               default:
+                  jsxc.debug('[OTR] sm callback: Unknown type: ' + type);
+            }
+         });
+
+         // Receive message
+         jsxc.otr.objects[bid].on('ui', function(msg, encrypted) {
+            jsxc.otr.receiveMessage(bid, msg, encrypted === true);
+         });
+
+         // Send message
+         jsxc.otr.objects[bid].on('io', function(msg, uid) {
+            var jid = jsxc.gui.window.get(bid).data('jid') || jsxc.otr.objects[bid].jid;
+
+            jsxc.otr.objects[bid].jid = jid;
+
+            jsxc.otr.sendMessage(jid, msg, uid);
+         });
+
+         jsxc.otr.objects[bid].on('error', function(err) {
+            // Handle this case in jsxc.otr.receiveMessage
+            if (err !== 'Received an unencrypted message.') {
+               jsxc.gui.window.postMessage(bid, 'sys', '[OTR] ' + jsxc.translate('%%' + err + '%%'));
+            }
+
+            jsxc.error('[OTR] ' + err);
+         });
+
+         jsxc.otr.restore(bid);
+      },
+
+      /**
+       * show verification dialog with related part (secret or question)
+       * 
+       * @param {type} bid
+       * @param {string} [data]
+       * @returns {undefined}
+       */
+      onSmpQuestion: function(bid, data) {
+         jsxc.gui.showVerification(bid);
+
+         $('#jsxc_dialog select').prop('selectedIndex', (data ? 2 : 3)).change();
+         $('#jsxc_dialog > div:eq(0)').hide();
+
+         if (data) {
+            $('#jsxc_dialog > div:eq(2)').find('#jsxc_quest').val(data).prop('disabled', true);
+            $('#jsxc_dialog > div:eq(2)').find('.creation').text('Answer');
+            $('#jsxc_dialog > div:eq(2)').find('.jsxc_explanation').text(jsxc.l.your_buddy_is_attempting_to_determine_ + ' ' + jsxc.l.to_authenticate_to_your_buddy + jsxc.l.enter_the_answer_and_click_answer);
+         } else {
+            $('#jsxc_dialog > div:eq(3)').find('.jsxc_explanation').text(jsxc.l.your_buddy_is_attempting_to_determine_ + ' ' + jsxc.l.to_authenticate_to_your_buddy + jsxc.l.enter_the_secret);
+         }
+
+         $('#jsxc_dialog .jsxc_close').click(function() {
+            jsxc.storage.removeUserItem('smp_' + bid);
+
+            if (jsxc.master) {
+               jsxc.otr.objects[bid].sm.abort();
+            }
+         });
+      },
+
+      /**
+       * Send verification request to buddy
+       * 
+       * @param {string} bid
+       * @param {string} sec secret
+       * @param {string} [quest] question
+       * @returns {undefined}
+       */
+      sendSmpReq: function(bid, sec, quest) {
+         jsxc.keepBusyAlive();
+
+         jsxc.otr.objects[bid].smpSecret(sec, quest || '');
+      },
+
+      /**
+       * Toggle encryption state
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      toggleTransfer: function(bid) {
+         if (jsxc.storage.getUserItem('buddy', bid).msgstate === 0) {
+            jsxc.otr.goEncrypt(bid);
+         } else {
+            jsxc.otr.goPlain(bid);
+         }
+      },
+
+      /**
+       * Send request to encrypt the session
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      goEncrypt: function(bid) {
+         if (jsxc.master) {
+            jsxc.otr.objects[bid].sendQueryMsg();
+         } else {
+            jsxc.storage.updateUserItem('buddy', bid, 'transferReq', 1);
+         }
+      },
+
+      /**
+       * Abort encryptet session
+       * 
+       * @param {type} bid
+       * @param cb callback
+       * @returns {undefined}
+       */
+      goPlain: function(bid, cb) {
+         if (jsxc.master) {
+            jsxc.otr.objects[bid].endOtr.call(jsxc.otr.objects[bid], cb);
+            jsxc.otr.objects[bid].init.call(jsxc.otr.objects[bid]);
+
+            jsxc.otr.backup(bid);
+         } else {
+            jsxc.storage.updateUserItem('buddy', bid, 'transferReq', 0);
+         }
+      },
+
+      /**
+       * Backups otr session
+       * 
+       * @param {string} bid
+       */
+      backup: function(bid) {
+         var o = jsxc.otr.objects[bid]; // otr object
+         var r = {}; // return value
+
+         if (o === null) {
+            return;
+         }
+
+         // all variables which should be saved
+         var savekey = [ 'jid', 'our_instance_tag', 'msgstate', 'authstate', 'fragment', 'their_y', 'their_old_y', 'their_keyid', 'their_instance_tag', 'our_dh', 'our_old_dh', 'our_keyid', 'sessKeys', 'storedMgs', 'oldMacKeys', 'trust', 'transmittedRS', 'ssid', 'receivedPlaintext', 'authstate', 'send_interval' ];
+
+         var i;
+         for (i = 0; i < savekey.length; i++) {
+            r[savekey[i]] = JSON.stringify(o[savekey[i]]);
+         }
+
+         if (o.their_priv_pk !== null) {
+            r.their_priv_pk = JSON.stringify(o.their_priv_pk.packPublic());
+         }
+
+         if (o.ake.otr_version && o.ake.otr_version !== '') {
+            r.otr_version = JSON.stringify(o.ake.otr_version);
+         }
+
+         jsxc.storage.setUserItem('otr', bid, r);
+      },
+
+      /**
+       * Restore old otr session
+       * 
+       * @param {string} bid
+       */
+      restore: function(bid) {
+         var o = jsxc.otr.objects[bid];
+         var d = jsxc.storage.getUserItem('otr', bid);
+
+         if (o !== null || d !== null) {
+            var key;
+            for (key in d) {
+               if (d.hasOwnProperty(key)) {
+                  var val = JSON.parse(d[key]);
+                  if (key === 'their_priv_pk' && val !== null) {
+                     val = DSA.parsePublic(val);
+                  }
+                  if (key === 'otr_version' && val !== null) {
+                     o.ake.otr_version = val;
+                  } else {
+                     o[key] = val;
+                  }
+               }
+            }
+
+            jsxc.otr.objects[bid] = o;
+
+            if (o.msgstate === 1 && o.their_priv_pk !== null) {
+               o._smInit.call(jsxc.otr.objects[bid]);
+            }
+         }
+
+         jsxc.otr.enable(bid);
+      },
+
+      /**
+       * Create or load DSA key
+       * 
+       * @returns {unresolved}
+       */
+      createDSA: function() {
+         if (jsxc.options.otr.priv) {
+            return;
+         }
+
+         if (jsxc.storage.getUserItem('key') === null) {
+            var msg = jsxc.l.Creating_your_private_key_;
+            var worker = null;
+
+            if (Worker) {
+               // try to create web-worker
+
+               try {
+                  worker = new Worker(jsxc.options.root + '/lib/otr/build/dsa-webworker.js');
+               } catch (err) {
+                  jsxc.warn('Couldn\'t create web-worker.', err);
+               }
+            }
+
+            jsxc.otr.dsaFallback = (worker === null);
+
+            if (!jsxc.otr.dsaFallback) {
+               // create DSA key in background
+
+               jsxc._onMaster();
+
+               worker.onmessage = function(e) {
+                  var type = e.data.type;
+                  var val = e.data.val;
+
+                  if (type === 'debug') {
+                     jsxc.debug(val);
+                  } else if (type === 'data') {
+                     jsxc.otr.DSAready(DSA.parsePrivate(val));
+                  }
+               };
+
+               // start worker
+               worker.postMessage({
+                  imports: [ jsxc.options.root + '/lib/otr/vendor/salsa20.js', jsxc.options.root + '/lib/otr/vendor/bigint.js', jsxc.options.root + '/lib/otr/vendor/crypto.js', jsxc.options.root + '/lib/otr/vendor/eventemitter.js', jsxc.options.root + '/lib/otr/lib/const.js', jsxc.options.root + '/lib/otr/lib/helpers.js', jsxc.options.root + '/lib/otr/lib/dsa.js' ],
+                  seed: BigInt.getSeed(),
+                  debug: true
+               });
+
+            } else {
+               // fallback
+
+               jsxc.gui.dialog.open(jsxc.gui.template.get('waitAlert', null, msg), {
+                  noClose: true
+               });
+
+               jsxc.debug('DSA key creation started.');
+
+               // wait until the wait alert is opened
+               setTimeout(function() {
+                  var dsa = new DSA();
+                  jsxc.otr.DSAready(dsa);
+               }, 500);
+            }
+         } else {
+            jsxc.debug('DSA key loaded');
+            jsxc.options.otr.priv = DSA.parsePrivate(jsxc.storage.getUserItem('key'));
+
+            jsxc.otr._createDSA();
+         }
+      },
+
+      /**
+       * Ending of createDSA().
+       */
+      _createDSA: function() {
+
+         jsxc.storage.setUserItem('priv_fingerprint', jsxc.options.otr.priv.fingerprint());
+
+         if (jsxc.otr.dsaFallback !== false) {
+            jsxc._onMaster();
+         }
+      },
+
+      /**
+       * Ending of DSA key generation.
+       * 
+       * @param {DSA} dsa DSA object
+       */
+      DSAready: function(dsa) {
+         jsxc.storage.setUserItem('key', dsa.packPrivate());
+         jsxc.options.otr.priv = dsa;
+
+         // close wait alert
+         if (jsxc.otr.dsaFallback) {
+            jsxc.gui.dialog.close();
+         } else {
+            $.each(jsxc.storage.getUserItem('windowlist'), function(index, val) {
+               jsxc.otr.create(val);
+            });
+         }
+
+         jsxc.otr._createDSA();
+      },
+
+      enable: function(bid) {
+         jsxc.gui.window.get(bid).find('.jsxc_otr').removeClass('jsxc_disabled');
+      }
+   };
+
+   /**
+    * This namespace handles the Notification API.
+    * 
+    * @namespace jsxc.notification
+    */
+   jsxc.notification = {
+
+      /** Current audio file. */
+      audio: null,
+
+      /**
+       * Register notification on incoming messages.
+       * 
+       * @memberOf jsxc.notification
+       */
+      init: function() {
+         $(document).on('postmessagein.jsxc', function(event, bid, msg) {
+            msg = (msg.match(/^\?OTR/)) ? jsxc.translate('%%Encrypted message%%') : msg;
+            var data = jsxc.storage.getUserItem('buddy', bid);
+
+            jsxc.notification.notify(jsxc.translate('%%New message from%% ') + data.name, msg, undefined, undefined, jsxc.CONST.SOUNDS.MSG);
+         });
+
+         $(document).on('callincoming.jingle', function() {
+            jsxc.notification.playSound(jsxc.CONST.SOUNDS.CALL, true, true);
+         });
+
+         $(document).on('accept.call.jsxc reject.call.jsxc', function() {
+            jsxc.notification.stopSound();
+         });
+      },
+
+      /**
+       * Shows a pop up notification and optional play sound.
+       * 
+       * @param title Title
+       * @param msg Message
+       * @param d Duration
+       * @param force Should message also shown, if tab is visible?
+       * @param soundFile Playing given sound file
+       * @param loop Loop sound file?
+       */
+      notify: function(title, msg, d, force, soundFile, loop) {
+         if (!jsxc.options.notification || !jsxc.notification.hasPermission()) {
+            return; // notifications disabled
+         }
+
+         if (!jsxc.isHidden() && !force) {
+            return; // Tab is visible
+         }
+
+         jsxc.toNotification = setTimeout(function() {
+
+            if (typeof soundFile === 'string') {
+               jsxc.notification.playSound(soundFile, loop, force);
+            }
+
+            var popup = new Notification(jsxc.translate(title), {
+               body: jsxc.translate(msg),
+               icon: jsxc.options.root + '/img/XMPP_logo.png'
+            });
+
+            var duration = d || jsxc.options.popupDuration;
+
+            if (duration > 0) {
+               setTimeout(function() {
+                  popup.close();
+               }, duration);
+            }
+         }, jsxc.toNotificationDelay);
+      },
+
+      /**
+       * Checks if browser has support for notifications and add on chrome to
+       * the default api.
+       * 
+       * @returns {Boolean} True if the browser has support.
+       */
+      hasSupport: function() {
+         if (window.webkitNotifications) {
+            // prepare chrome
+
+            window.Notification = function(title, opt) {
+               var popup = window.webkitNotifications.createNotification(null, title, opt.body);
+               popup.show();
+
+               popup.close = function() {
+                  popup.cancel();
+               };
+
+               return popup;
+            };
+
+            var permission;
+            switch (window.webkitNotifications.checkPermission()) {
+               case 0:
+                  permission = jsxc.CONST.NOTIFICATION_GRANTED;
+                  break;
+               case 2:
+                  permission = jsxc.CONST.NOTIFICATION_DENIED;
+                  break;
+               default: // 1
+                  permission = jsxc.CONST.NOTIFICATION_DEFAULT;
+            }
+            window.Notification.permission = permission;
+
+            window.Notification.requestPermission = function(func) {
+               window.webkitNotifications.requestPermission(func);
+            };
+
+            return true;
+         } else if (window.Notification) {
+            return true;
+         } else {
+            return false;
+         }
+      },
+
+      /**
+       * Ask user on first incoming message if we should inform him about new
+       * messages.
+       */
+      prepareRequest: function() {
+
+         $(document).one('postmessagein.jsxc', function() {
+            jsxc.switchEvents({
+               'notificationready.jsxc': function() {
+                  jsxc.gui.dialog.close();
+                  jsxc.notification.init();
+                  jsxc.storage.setUserItem('notification', true);
+               },
+               'notificationfailure.jsxc': function() {
+                  jsxc.gui.dialog.close();
+                  jsxc.options.notification = false;
+                  jsxc.storage.setUserItem('notification', false);
+               }
+            });
+
+            setTimeout(function() {
+               jsxc.notice.add('%%Notifications%%?', '%%Should_we_notify_you_%%', 'gui.showRequestNotification');
+            }, 1000);
+         });
+      },
+
+      /**
+       * Request notification permission.
+       */
+      requestPermission: function() {
+         window.Notification.requestPermission(function(status) {
+            if (window.Notification.permission !== status) {
+               window.Notification.permission = status;
+            }
+
+            if (jsxc.notification.hasPermission()) {
+               $(document).trigger('notificationready.jsxc');
+            } else {
+               $(document).trigger('notificationfailure.jsxc');
+            }
+         });
+      },
+
+      /**
+       * Check permission.
+       * 
+       * @returns {Boolean} True if we have the permission
+       */
+      hasPermission: function() {
+         return window.Notification.permission === jsxc.CONST.NOTIFICATION_GRANTED;
+      },
+
+      /**
+       * Plays the given file.
+       * 
+       * @memberOf jsxc.notification
+       * @param {string} soundFile File relative to the sound directory
+       * @param {boolean} loop True for loop
+       * @param {boolean} force Play even if a tab is visible. Default: false.
+       */
+      playSound: function(soundFile, loop, force) {
+         if (!jsxc.master) {
+            // only master plays sound
+            return;
+         }
+
+         if (jsxc.options.get('muteNotification') || jsxc.storage.getUserItem('presence') === 'dnd') {
+            // sound mute or own presence is dnd
+            return;
+         }
+
+         if (!jsxc.isHidden() && !force) {
+            // tab is visible
+            return;
+         }
+
+         // stop current audio file
+         jsxc.notification.stopSound();
+
+         var audio = new Audio(jsxc.options.root + '/sound/' + soundFile);
+         audio.loop = loop || false;
+         audio.play();
+
+         jsxc.notification.audio = audio;
+      },
+
+      /**
+       * Stop/remove current sound.
+       * 
+       * @memberOf jsxc.notification
+       */
+      stopSound: function() {
+         var audio = jsxc.notification.audio;
+
+         if (typeof audio !== 'undefined' && audio !== null) {
+            audio.pause();
+            jsxc.notification.audio = null;
+         }
+      },
+
+      /**
+       * Mute sound.
+       * 
+       * @memberOf jsxc.notification
+       * @param {boolean} external True if triggered from external tab. Default:
+       *        false.
+       */
+      muteSound: function(external) {
+         $('#jsxc_menu .jsxc_muteNotification').text(jsxc.translate('%%Unmute%%'));
+
+         if (external !== true) {
+            jsxc.options.set('muteNotification', true);
+         }
+      },
+
+      /**
+       * Unmute sound.
+       * 
+       * @memberOf jsxc.notification
+       * @param {boolean} external True if triggered from external tab. Default:
+       *        false.
+       */
+      unmuteSound: function(external) {
+         $('#jsxc_menu .jsxc_muteNotification').text(jsxc.translate('%%Mute%%'));
+
+         if (external !== true) {
+            jsxc.options.set('muteNotification', false);
+         }
+      }
+   };
+
+   /**
+    * This namespace handle the notice system.
+    * 
+    * @namspace jsxc.notice
+    * @memberOf jsxc
+    */
+   jsxc.notice = {
+      /** Number of notices. */
+      _num: 0,
+
+      /**
+       * Loads the saved notices.
+       * 
+       * @memberOf jsxc.notice
+       */
+      load: function() {
+         // reset list
+         $('#jsxc_notice ul li').remove();
+         $('#jsxc_notice > span').text('');
+         jsxc.notice._num = 0;
+
+         var saved = jsxc.storage.getUserItem('notices') || [];
+         var key = null;
+
+         for (key in saved) {
+            if (saved.hasOwnProperty(key)) {
+               var val = saved[key];
+
+               jsxc.notice.add(val.msg, val.description, val.fnName, val.fnParams, key);
+            }
+         }
+      },
+
+      /**
+       * Add a new notice to the stack;
+       * 
+       * @memberOf jsxc.notice
+       * @param msg Header message
+       * @param description Notice description
+       * @param fnName Function name to be called if you open the notice
+       * @param fnParams Array of params for function
+       * @param id Notice id
+       */
+      add: function(msg, description, fnName, fnParams, id) {
+         var nid = id || Date.now();
+         var list = $('#jsxc_notice ul');
+         var notice = $('<li/>');
+
+         notice.click(function() {
+            jsxc.notice.remove(nid);
+
+            var fnList = fnName.split('.');
+            var fn = jsxc[fnList[0]];
+            var i;
+            for (i = 1; i < fnList.length; i++) {
+               fn = fn[fnList[i]];
+            }
+
+            if (typeof fn === 'function') {
+               fn.apply(null, fnParams);
+            }
+
+            return false;
+         });
+
+         notice.text(jsxc.translate(msg));
+         notice.attr('title', jsxc.translate(description) || '');
+         notice.attr('data-nid', nid);
+         list.append(notice);
+
+         $('#jsxc_notice > span').text(++jsxc.notice._num);
+
+         if (!id) {
+            var saved = jsxc.storage.getUserItem('notices') || {};
+            saved[nid] = {
+               msg: msg,
+               description: description,
+               fnName: fnName,
+               fnParams: fnParams
+            };
+            jsxc.storage.setUserItem('notices', saved);
+
+            jsxc.notification.notify(msg, description || '', null, true, jsxc.CONST.SOUNDS.NOTICE);
+         }
+      },
+
+      /**
+       * Removes notice from stack
+       * 
+       * @memberOf jsxc.notice
+       * @param nid The notice id
+       */
+      remove: function(nid) {
+         var el = $('#jsxc_notice li[data-nid=' + nid + ']');
+
+         el.remove();
+         $('#jsxc_notice > span').text(--jsxc.notice._num || '');
+
+         var s = jsxc.storage.getUserItem('notices');
+         delete s[nid];
+         jsxc.storage.setUserItem('notices', s);
+      }
+   };
+
+   /**
+    * Contains all available translations
+    * 
+    * @namespace jsxc.l10n
+    * @memberOf jsxc
+    */
+   jsxc.l10n = {
+      en: {
+         Logging_in: 'Logging in…',
+         your_connection_is_unencrypted: 'Your connection is unencrypted.',
+         your_connection_is_encrypted: 'Your connection is encrypted.',
+         your_buddy_closed_the_private_connection: 'Your buddy closed the private connection.',
+         start_private: 'Start private',
+         close_private: 'Close private',
+         your_buddy_is_verificated: 'Your buddy is verified.',
+         you_have_only_a_subscription_in_one_way: 'You only have a one-way subscription.',
+         authentication_query_sent: 'Authentication query sent.',
+         your_message_wasnt_send_please_end_your_private_conversation: 'Your message was not sent. Please end your private conversation.',
+         unencrypted_message_received: 'Unencrypted message received:',
+         your_message_wasnt_send_because_you_have_no_valid_subscription: 'Your message was not sent because you have no valid subscription.',
+         not_available: 'Not available',
+         no_connection: 'No connection!',
+         relogin: 'relogin',
+         trying_to_start_private_conversation: 'Trying to start private conversation!',
+         Verified: 'Verified',
+         Unverified: 'Unverified',
+         private_conversation_started: 'Private conversation started.',
+         private_conversation_aborted: 'Private conversation aborted!',
+         your_buddy_closed_the_private_conversation_you_should_do_the_same: 'Your buddy closed the private conversation! You should do the same.',
+         conversation_is_now_verified: 'Conversation is now verified.',
+         authentication_failed: 'Authentication failed.',
+         your_buddy_is_attempting_to_determine_: 'You buddy is attempting to determine if he or she is really talking to you.',
+         to_authenticate_to_your_buddy: 'To authenticate to your buddy, ',
+         enter_the_answer_and_click_answer: 'enter the answer and click Answer.',
+         enter_the_secret: 'enter the secret.',
+         Creating_your_private_key_: 'Creating your private key; this may take a while.',
+         Authenticating_a_buddy_helps_: 'Authenticating a buddy helps ensure that the person you are talking to is really the one he or she claims to be.',
+         How_do_you_want_to_authenticate_your_buddy: 'How do you want to authenticate {{bid_name}} (<b>{{bid_jid}}</b>)?',
+         Select_method: 'Select method...',
+         Manual: 'Manual',
+         Question: 'Question',
+         Secret: 'Secret',
+         To_verify_the_fingerprint_: 'To verify the fingerprint, contact your buddy via some other trustworthy channel, such as the telephone.',
+         Your_fingerprint: 'Your fingerprint',
+         Buddy_fingerprint: 'Buddy fingerprint',
+         Close: 'Close',
+         Compared: 'Compared',
+         To_authenticate_using_a_question_: 'To authenticate using a question, pick a question whose answer is known only you and your buddy.',
+         Ask: 'Ask',
+         To_authenticate_pick_a_secret_: 'To authenticate, pick a secret known only to you and your buddy.',
+         Compare: 'Compare',
+         Fingerprints: 'Fingerprints',
+         Authentication: 'Authentication',
+         Message: 'Message',
+         Add_buddy: 'Add buddy',
+         rename_buddy: 'rename buddy',
+         delete_buddy: 'delete buddy',
+         Login: 'Login',
+         Username: 'Username',
+         Password: 'Password',
+         Cancel: 'Cancel',
+         Connect: 'Connect',
+         Type_in_the_full_username_: 'Type in the full username and an optional alias.',
+         Alias: 'Alias',
+         Add: 'Add',
+         Subscription_request: 'Subscription request',
+         You_have_a_request_from: 'You have a request from',
+         Deny: 'Deny',
+         Approve: 'Approve',
+         Remove_buddy: 'Remove buddy',
+         You_are_about_to_remove_: 'You are about to remove {{bid_name}} (<b>{{bid_jid}}</b>) from your buddy list. All related chats will be closed.',
+         Continue_without_chat: 'Continue without chat',
+         Please_wait: 'Please wait',
+         Login_failed: 'Chat login failed',
+         Sorry_we_cant_authentikate_: 'Authentication failed with the chat server. Maybe the password is wrong?',
+         Retry: 'Back',
+         clear_history: 'Clear history',
+         New_message_from: 'New message from',
+         Should_we_notify_you_: 'Should we notify you about new messages in the future?',
+         Please_accept_: 'Please click the "Allow" button at the top.',
+         Hide_offline: 'Hide offline contacts',
+         Show_offline: 'Show offline contacts',
+         About: 'About',
+         dnd: 'Do Not Disturb',
+         Mute: 'Mute',
+         Unmute: 'Unmute',
+         Subscription: 'Subscription',
+         both: 'both',
+         Status: 'Status',
+         online: 'online',
+         chat: 'chat',
+         away: 'away',
+         xa: 'extended away',
+         offline: 'offline',
+         none: 'none',
+         Unknown_instance_tag: 'Unknown instance tag.',
+         Not_one_of_our_latest_keys: 'Not one of our latest keys.',
+         Received_an_unreadable_encrypted_message: 'Received an unreadable encrypted message.',
+         Online: 'Online',
+         Chatty: 'Chatty',
+         Away: 'Away',
+         Extended_away: 'Extended away',
+         Offline: 'Offline',
+         Friendship_request: 'Friendship request',
+         Confirm: 'Confirm',
+         Dismiss: 'Dismiss',
+         Remove: 'Remove',
+         Online_help: 'Online help',
+         FN: 'Full name',
+         N: ' ',
+         FAMILY: 'Family name',
+         GIVEN: 'Given name',
+         NICKNAME: 'Nickname',
+         URL: 'URL',
+         ADR: 'Address',
+         STREET: 'Street Address',
+         EXTADD: 'Extended Address',
+         LOCALITY: 'Locality',
+         REGION: 'Region',
+         PCODE: 'Postal Code',
+         CTRY: 'Country',
+         TEL: 'Telephone',
+         NUMBER: 'Number',
+         EMAIL: 'Email',
+         USERID: ' ',
+         ORG: 'Organization',
+         ORGNAME: 'Name',
+         ORGUNIT: 'Unit',
+         TITLE: 'Job title',
+         ROLE: 'Role',
+         BDAY: 'Birthday',
+         DESC: 'Description',
+         PHOTO: ' ',
+         send_message: 'Send message',
+         get_info: 'Show information',
+         Settings: 'Settings',
+         Priority: 'Priority',
+         Save: 'Save',
+         User_settings: 'User settings',
+         A_fingerprint_: 'A fingerprint is used to make sure that the person you are talking to is who he or she is saying.',
+         Your_roster_is_empty_add_a: 'Your roster is empty, add a ',
+         new_buddy: 'new buddy',
+         is: 'is',
+         Login_options: 'Login options',
+         BOSH_url: 'BOSH URL',
+         Domain: 'Domain',
+         Resource: 'Resource',
+         On_login: 'On login',
+         Received_an_unencrypted_message: 'Received an unencrypted message',
+         Sorry_your_buddy_doesnt_provide_any_information: 'Sorry, your buddy does not provide any information.',
+         Info_about: 'Info about',
+         Authentication_aborted: 'Authentication aborted.',
+         Authentication_request_received: 'Authentication request received.',
+         Do_you_want_to_display_them: 'Do you want to display them?',
+         Log_in_without_chat: 'Log in without chat',
+         has_come_online: 'has come online',
+         Unknown_sender: 'Unknown sender',
+         You_received_a_message_from_an_unknown_sender: 'You received a message from an unknown sender'
+      },
+      de: {
+         Logging_in: 'Login läuft…',
+         your_connection_is_unencrypted: 'Deine Verbindung ist UNverschlüsselt.',
+         your_connection_is_encrypted: 'Deine Verbindung ist verschlüsselt.',
+         your_buddy_closed_the_private_connection: 'Dein Freund hat die private Verbindung getrennt.',
+         start_private: 'Privat starten',
+         close_private: 'Privat abbrechen',
+         your_buddy_is_verificated: 'Dein Freund ist verifiziert.',
+         you_have_only_a_subscription_in_one_way: 'Die Freundschaft ist nur einseitig.',
+         authentication_query_sent: 'Authentifizierungsanfrage gesendet.',
+         your_message_wasnt_send_please_end_your_private_conversation: 'Deine Nachricht wurde nicht gesendet. Bitte beende die private Konversation.',
+         unencrypted_message_received: 'Unverschlüsselte Nachricht erhalten.',
+         your_message_wasnt_send_because_you_have_no_valid_subscription: 'Deine Nachricht wurde nicht gesandt, da die Freundschaft einseitig ist.',
+         not_available: 'Nicht verfügbar.',
+         no_connection: 'Keine Verbindung.',
+         relogin: 'Neu anmelden.',
+         trying_to_start_private_conversation: 'Versuche private Konversation zu starten.',
+         Verified: 'Verifiziert',
+         Unverified: 'Unverifiziert',
+         private_conversation_started: 'Private Konversation gestartet.',
+         private_conversation_aborted: 'Private Konversation abgebrochen.',
+         your_buddy_closed_the_private_conversation_you_should_do_the_same: 'Dein Freund hat die private Konversation beendet. Das solltest du auch tun!',
+         conversation_is_now_verified: 'Konversation ist jetzt verifiziert',
+         authentication_failed: 'Authentifizierung fehlgeschlagen.',
+         your_buddy_is_attempting_to_determine_: 'Dein Freund versucht herauszufinden ob er wirklich mit dir redet.',
+         to_authenticate_to_your_buddy: 'Um dich gegenüber deinem Freund zu verifizieren ',
+         enter_the_answer_and_click_answer: 'gib die Antwort ein und klick auf Antworten.',
+         enter_the_secret: 'gib das Geheimnis ein.',
+         Creating_your_private_key_: 'Wir werden jetzt deinen privaten Schlüssel generieren. Das kann einige Zeit in Anspruch nehmen.',
+         Authenticating_a_buddy_helps_: 'Einen Freund zu authentifizieren hilft sicher zustellen, dass die Person mit der du sprichst auch die ist die sie sagt.',
+         How_do_you_want_to_authenticate_your_buddy: 'Wie willst du {{bid_name}} (<b>{{bid_jid}}</b>) authentifizieren?',
+         Select_method: 'Wähle...',
+         Manual: 'Manual',
+         Question: 'Frage',
+         Secret: 'Geheimnis',
+         To_verify_the_fingerprint_: 'Um den Fingerprint zu verifizieren kontaktiere dein Freund über einen anderen Kommunikationsweg. Zum Beispiel per Telefonanruf.',
+         Your_fingerprint: 'Dein Fingerprint',
+         Buddy_fingerprint: 'Sein/Ihr Fingerprint',
+         Close: 'Schließen',
+         Compared: 'Verglichen',
+         To_authenticate_using_a_question_: 'Um die Authentifizierung per Frage durchzuführen, wähle eine Frage bei welcher nur dein Freund die Antwort weiß.',
+         Ask: 'Frage',
+         To_authenticate_pick_a_secret_: 'Um deinen Freund zu authentifizieren, wähle ein Geheimnis welches nur deinem Freund und dir bekannt ist.',
+         Compare: 'Vergleiche',
+         Fingerprints: 'Fingerprints',
+         Authentication: 'Authentifizierung',
+         Message: 'Nachricht',
+         Add_buddy: 'Freund hinzufügen',
+         rename_buddy: 'Freund umbenennen',
+         delete_buddy: 'Freund löschen',
+         Login: 'Anmeldung',
+         Username: 'Benutzername',
+         Password: 'Passwort',
+         Cancel: 'Abbrechen',
+         Connect: 'Verbinden',
+         Type_in_the_full_username_: 'Gib bitte den vollen Benutzernamen und optional ein Alias an.',
+         Alias: 'Alias',
+         Add: 'Hinzufügen',
+         Subscription_request: 'Freundschaftsanfrage',
+         You_have_a_request_from: 'Du hast eine Anfrage von',
+         Deny: 'Ablehnen',
+         Approve: 'Bestätigen',
+         Remove_buddy: 'Freund entfernen',
+         You_are_about_to_remove_: 'Du bist gerade dabei {{bid_name}} (<b>{{bid_jid}}</b>) von deiner Kontaktliste zu entfernen. Alle Chats werden geschlossen.',
+         Continue_without_chat: 'Weiter ohne Chat',
+         Please_wait: 'Bitte warten',
+         Login_failed: 'Chat-Anmeldung fehlgeschlagen',
+         Sorry_we_cant_authentikate_: 'Der Chatserver hat die Anmeldung abgelehnt. Falsches Passwort?',
+         Retry: 'Zurück',
+         clear_history: 'Lösche Verlauf',
+         New_message_from: 'Neue Nachricht von',
+         Should_we_notify_you_: 'Sollen wir dich in Zukunft über eingehende Nachrichten informieren, auch wenn dieser Tab nicht im Vordergrund ist?',
+         Please_accept_: 'Bitte klick auf den "Zulassen" Button oben.',
+         Menu: 'Menü',
+         Hide_offline: 'Offline ausblenden',
+         Show_offline: 'Offline einblenden',
+         About: 'Über',
+         dnd: 'Beschäftigt',
+         Mute: 'Ton aus',
+         Unmute: 'Ton an',
+         Subscription: 'Bezug',
+         both: 'beidseitig',
+         Status: 'Status',
+         online: 'online',
+         chat: 'chat',
+         away: 'abwesend',
+         xa: 'länger abwesend',
+         offline: 'offline',
+         none: 'keine',
+         Unknown_instance_tag: 'Unbekannter instance tag.',
+         Not_one_of_our_latest_keys: 'Nicht einer unserer letzten Schlüssel.',
+         Received_an_unreadable_encrypted_message: 'Eine unlesbare verschlüsselte Nachricht erhalten.',
+         Online: 'Online',
+         Chatty: 'Gesprächig',
+         Away: 'Abwesend',
+         Extended_away: 'Länger abwesend',
+         Offline: 'Offline',
+         Friendship_request: 'Freundschaftsanfrage',
+         Confirm: 'Bestätigen',
+         Dismiss: 'Ablehnen',
+         Remove: 'Löschen',
+         Online_help: 'Online Hilfe',
+         FN: 'Name',
+         N: ' ',
+         FAMILY: 'Familienname',
+         GIVEN: 'Vorname',
+         NICKNAME: 'Spitzname',
+         URL: 'URL',
+         ADR: 'Adresse',
+         STREET: 'Straße',
+         EXTADD: 'Zusätzliche Adresse',
+         LOCALITY: 'Ortschaft',
+         REGION: 'Region',
+         PCODE: 'Postleitzahl',
+         CTRY: 'Land',
+         TEL: 'Telefon',
+         NUMBER: 'Nummer',
+         EMAIL: 'E-Mail',
+         USERID: ' ',
+         ORG: 'Organisation',
+         ORGNAME: 'Name',
+         ORGUNIT: 'Abteilung',
+         TITLE: 'Titel',
+         ROLE: 'Rolle',
+         BDAY: 'Geburtstag',
+         DESC: 'Beschreibung',
+         PHOTO: ' ',
+         send_message: 'Sende Nachricht',
+         get_info: 'Benutzerinformationen',
+         Settings: 'Einstellungen',
+         Priority: 'Priorität',
+         Save: 'Speichern',
+         User_settings: 'Benutzereinstellungen',
+         A_fingerprint_: 'Ein Fingerabdruck wird dazu benutzt deinen Gesprächspartner zu identifizieren.',
+         Your_roster_is_empty_add_a: 'Deine Freundesliste ist leer, füge einen neuen Freund ',
+         new_buddy: 'hinzu',
+         is: 'ist',
+         Login_options: 'Anmeldeoptionen',
+         BOSH_url: 'BOSH url',
+         Domain: 'Domain',
+         Resource: 'Ressource',
+         On_login: 'Beim Anmelden',
+         Received_an_unencrypted_message: 'Unverschlüsselte Nachricht empfangen',
+         Sorry_your_buddy_doesnt_provide_any_information: 'Dein Freund stellt leider keine Informationen bereit.',
+         Info_about: 'Info über',
+         Authentication_aborted: 'Authentifizierung abgebrochen.',
+         Authentication_request_received: 'Authentifizierunganfrage empfangen.',
+         Log_in_without_chat: 'Anmelden ohne Chat',
+         Do_you_want_to_display_them: 'Möchtest du sie sehen?',
+         has_come_online: 'ist online gekommen',
+         Unknown_sender: 'Unbekannter Sender',
+         You_received_a_message_from_an_unknown_sender: 'Du hast eine Nachricht von einem unbekannten Sender erhalten'
+      },
+      es: {
+         Logging_in: 'Por favor, espere...',
+         your_connection_is_unencrypted: 'Su conexión no está cifrada.',
+         your_connection_is_encrypted: 'Su conexión está cifrada.',
+         your_buddy_closed_the_private_connection: 'Su amigo ha cerrado la conexión privada.',
+         start_private: 'Iniciar privado',
+         close_private: 'Cerrar privado',
+         your_buddy_is_verificated: 'Tu amigo está verificado.',
+         you_have_only_a_subscription_in_one_way: 'Sólo tienes una suscripción de un modo.',
+         authentication_query_sent: 'Consulta de verificación enviada.',
+         your_message_wasnt_send_please_end_your_private_conversation: 'Su mensaje no fue enviado. Por favor, termine su conversación privada.',
+         unencrypted_message_received: 'Mensaje no cifrado recibido:',
+         your_message_wasnt_send_because_you_have_no_valid_subscription: 'Su mensaje no se ha enviado, porque usted no tiene suscripción válida.',
+         not_available: 'No disponible',
+         no_connection: 'Sin conexión!',
+         relogin: 'iniciar sesión nuevamente',
+         trying_to_start_private_conversation: 'Intentando iniciar una conversación privada!',
+         Verified: 'Verificado',
+         Unverified: 'No verificado',
+         private_conversation_started: 'se inició una conversación privada.',
+         private_conversation_aborted: 'Conversación privada abortada!',
+         your_buddy_closed_the_private_conversation_you_should_do_the_same: 'Su amigo cerró la conversación privada! Usted debería hacer lo mismo.',
+         conversation_is_now_verified: 'La conversación es ahora verificada.',
+         authentication_failed: 'Fallo la verificación.',
+         your_buddy_is_attempting_to_determine_: 'Tu amigo está tratando de determinar si él o ella está realmente hablando con usted.',
+         to_authenticate_to_your_buddy: 'Para autenticar a su amigo, ',
+         enter_the_answer_and_click_answer: 'introduce la respuesta y haga clic en Contestar.',
+         enter_the_secret: 'especifique el secreto.',
+         Creating_your_private_key_: 'Ahora vamos a crear su clave privada. Esto puede tomar algún tiempo.',
+         Authenticating_a_buddy_helps_: 'Autenticación de un amigo ayuda a garantizar que la persona que está hablando es quien él o ella está diciendo.',
+         How_do_you_want_to_authenticate_your_buddy: '¿Cómo desea autenticar {{bid_name}} (<b>{{bid_jid}}</b>)?',
+         Select_method: 'Escoja un método...',
+         Manual: 'Manual',
+         Question: 'Pregunta',
+         Secret: 'Secreto',
+         To_verify_the_fingerprint_: 'Para verificar la firma digital, póngase en contacto con su amigo a través de algún otro canal autenticado, como el teléfono.',
+         Your_fingerprint: 'Tu firma digital',
+         Buddy_fingerprint: 'firma digital de tu amigo',
+         Close: 'Cerrar',
+         Compared: 'Comparado',
+         To_authenticate_using_a_question_: 'Para autenticar mediante una pregunta, elegir una pregunta cuya respuesta se conoce sólo usted y su amigo.',
+         Ask: 'Preguntar',
+         To_authenticate_pick_a_secret_: 'Para autenticar, elija un secreto conocido sólo por usted y su amigo.',
+         Compare: 'Comparar',
+         Fingerprints: 'Firmas digitales',
+         Authentication: 'Autenticación',
+         Message: 'Mensaje',
+         Add_buddy: 'Añadir amigo',
+         rename_buddy: 'renombrar amigo',
+         delete_buddy: 'eliminar amigo',
+         Login: 'Iniciar Sesión',
+         Username: 'Usuario',
+         Password: 'Contraseña',
+         Cancel: 'Cancelar',
+         Connect: 'Conectar',
+         Type_in_the_full_username_: 'Escriba el usuario completo y un alias opcional.',
+         Alias: 'Alias',
+         Add: 'Añadir',
+         Subscription_request: 'Solicitud de suscripción',
+         You_have_a_request_from: 'Tienes una petición de',
+         Deny: 'Rechazar',
+         Approve: 'Aprobar',
+         Remove_buddy: 'Eliminar amigo',
+         You_are_about_to_remove_: 'Vas a eliminar a {{bid_name}} (<b>{{bid_jid}}</b>) de tu lista de amigos. Todas las conversaciones relacionadas serán cerradas.',
+         Continue_without_chat: 'Continuar',
+         Please_wait: 'Espere por favor',
+         Login_failed: 'Fallo el inicio de sesión',
+         Sorry_we_cant_authentikate_: 'Lo sentimos, no podemos autentificarlo en nuestro servidor de chat. ¿Tal vez la contraseña es incorrecta?',
+         Retry: 'Reintentar',
+         clear_history: 'Borrar el historial',
+         New_message_from: 'Nuevo mensaje de',
+         Should_we_notify_you_: '¿Debemos notificarle sobre nuevos mensajes en el futuro?',
+         Please_accept_: 'Por favor, haga clic en el botón "Permitir" en la parte superior.',
+         dnd: 'No Molestar',
+         Mute: 'Desactivar sonido',
+         Unmute: 'Activar sonido',
+         Subscription: 'Suscripción',
+         both: 'ambos',
+         Status: 'Estado',
+         online: 'en línea',
+         chat: 'chat',
+         away: 'ausente',
+         xa: 'mas ausente',
+         offline: 'desconectado',
+         none: 'nadie',
+         Unknown_instance_tag: 'Etiqueta de instancia desconocida.',
+         Not_one_of_our_latest_keys: 'No de nuestra ultima tecla.',
+         Received_an_unreadable_encrypted_message: 'Se recibió un mensaje cifrado ilegible.',
+         Online: 'En linea',
+         Chatty: 'Hablador',
+         Away: 'Ausente',
+         Extended_away: 'Mas ausente',
+         Offline: 'Desconectado',
+         Friendship_request: 'Solicitud de amistad',
+         Confirm: 'Confirmar',
+         Dismiss: 'Rechazar',
+         Remove: 'Eliminar',
+         Online_help: 'Ayuda en línea',
+         FN: 'Nombre completo ',
+         N: ' ',
+         FAMILY: 'Apellido',
+         GIVEN: 'Nombre',
+         NICKNAME: 'Apodar',
+         URL: 'URL',
+         ADR: 'Dirección',
+         STREET: 'Calle',
+         EXTADD: 'Extendido dirección',
+         LOCALITY: 'Población',
+         REGION: 'Región',
+         PCODE: 'Código postal',
+         CTRY: 'País',
+         TEL: 'Teléfono',
+         NUMBER: 'Número',
+         EMAIL: 'Emilio',
+         USERID: ' ',
+         ORG: 'Organización',
+         ORGNAME: 'Nombre',
+         ORGUNIT: 'Departamento',
+         TITLE: 'Título',
+         ROLE: 'Rol',
+         BDAY: 'Cumpleaños',
+         DESC: 'Descripción',
+         PHOTO: ' ',
+         send_message: 'mandar un texto',
+         get_info: 'obtener información',
+         Settings: 'Ajustes',
+         Priority: 'Prioridad',
+         Save: 'Guardar',
+         User_settings: 'Configuración de usuario',
+         A_fingerprint_: 'La huella digital se utiliza para que puedas estar seguro que la persona con la que estas hablando es quien realmente dice ser',
+         Your_roster_is_empty_add_a: 'Tu lista de amigos esta vacia',
+         new_buddy: 'Nuevo amigo',
+         is: 'es',
+         Login_options: 'Opciones de login',
+         BOSH_url: 'BOSH url',
+         Domain: 'Dominio',
+         Resource: 'Recurso',
+         On_login: 'Iniciar sesión',
+         Received_an_unencrypted_message: 'Recibe un mensaje no cifrado'
+      }
+   };
+}(jQuery));
+
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org> <br>
+ * Released under the MIT license
+ * 
+ * Please see http://www.jsxc.org/
+ * 
+ * @author Klaus Herberth <klaus at jsxc.org>
+ * @version 1.0.0
+ * @license MIT
+ */
+
+/* global jsxc, Strophe, SDPUtil, getUserMediaWithConstraints, setupRTC, jQuery, MediaStreamTrack */
+
+var RTC = null, RTCPeerconnection = null;
+
+jsxc.gui.template.incomingCall = '<h3>%%Incoming_call%%</h3>\
+        <p>%%Do_you_want_to_accept_the_call_from%% {{bid_name}}?</p>\
+        <p class="jsxc_right">\
+            <a href="#" class="button jsxc_reject">%%Reject%%</a> <a href="#" class="button creation jsxc_accept">%%Accept%%</a>\
+         </p>';
+
+jsxc.gui.template.allowMediaAccess = '<p>%%Please_allow_access_to_microphone_and_camera%%</p>';
+
+jsxc.gui.template.videoWindow = '<div class="jsxc_webrtc">\
+            <div class="jsxc_chatarea">\
+                <ul></ul>\
+            </div>\
+            <div class="jsxc_videoContainer">\
+                <video class="jsxc_localvideo" autoplay></video>\
+                <video class="jsxc_remotevideo" autoplay></video>\
+                <div class="jsxc_status"></div>\
+               <div class="bubblingG">\
+                  <span id="bubblingG_1">\
+                  </span>\
+                  <span id="bubblingG_2">\
+                  </span>\
+                  <span id="bubblingG_3">\
+                  </span>\
+               </div>\
+                <div class="jsxc_noRemoteVideo">\
+                   <div>\
+                     <div></div>\
+                     <p>%%No_video_signal%%</p>\
+                     <div></div>\
+                   </div>\
+                </div>\
+            </div>\
+            <div class="jsxc_controlbar">\
+                <button type="button" class="jsxc_hangUp">%%hang_up%%</button>\
+                <input type="range" class="jsxc_volume" min="0.0" max="1.0" step="0.05" value="0.5" />\
+                <div class="jsxc_buttongroup">\
+                    <button type="button" class="jsxc_snapshot">%%snapshot%%</button><button type="button" class="jsxc_snapshots">▼</button>\
+                </div>\
+                <!-- <button type="button" class="jsxc_mute_local">%%mute_my_audio%%</button>\
+                <button type="button" class="jsxc_pause_local">%%pause_my_video%%</button> --> \
+                <button type="button" class="jsxc_showchat">%%chat%%</button>\
+                <button type="button" class="jsxc_fullscreen">%%fullscreen%%</button>\
+                <button type="button" class="jsxc_info">%%Info%%</button>\
+            </div>\
+            <div class="jsxc_multi">\
+               <div class="jsxc_snapshotbar">\
+                   <p>No pictures yet!</p>\
+               </div>\n\
+               <!--<div class="jsxc_chatarea">\
+                   <ul></ul>\
+               </div>-->\
+               <div class="jsxc_infobar"></div>\
+            </div>\
+        </div>';
+
+(function($) {
+   "use strict";
+
+   /**
+    * WebRTC namespace for jsxc.
+    * 
+    * @namespace jsxc.webrtc
+    */
+   jsxc.webrtc = {
+      /** strophe connection */
+      conn: null,
+
+      /** local video stream */
+      localStream: null,
+
+      /** remote video stream */
+      remoteStream: null,
+
+      /** jid of the last caller */
+      last_caller: null,
+
+      /** should we auto accept incoming calls? */
+      AUTO_ACCEPT: false,
+
+      /** required disco features */
+      reqVideoFeatures: [ 'urn:xmpp:jingle:apps:rtp:video', 'urn:xmpp:jingle:apps:rtp:audio', 'urn:xmpp:jingle:transports:ice-udp:1', 'urn:xmpp:jingle:apps:dtls:0' ],
+
+      /** bare jid to current jid mapping */
+      chatJids: {},
+
+      /**
+       * Initialize webrtc plugin.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      init: function() {
+         var self = jsxc.webrtc;
+
+         // shortcut
+         self.conn = jsxc.xmpp.conn;
+
+         if (RTC.browser === 'firefox') {
+            self.conn.jingle.media_constraints.mandatory.MozDontOfferDataChannel = true;
+         }
+
+         if (!self.conn.jingle) {
+            jsxc.error('No jingle plugin found!');
+            return;
+         }
+
+         // jingle configuration
+         self.conn.jingle.PRANSWER = false;
+         self.conn.jingle.AUTOACCEPT = false;
+         self.conn.jingle.ice_config = jsxc.storage.getUserItem('iceConfig');
+         self.conn.jingle.MULTIPARTY = false;
+         self.conn.jingle.pc_constraints = RTC.pc_constraints;
+
+         $(document).on('message.jsxc', $.proxy(self.onMessage, self));
+         $(document).on('presence.jsxc', $.proxy(self.onPresence, self));
+
+         $(document).on('mediaready.jingle', $.proxy(self.onMediaReady, self));
+         $(document).on('mediafailure.jingle', $.proxy(self.onMediaFailure, self));
+         $(document).on('callincoming.jingle', $.proxy(self.onCallIncoming, self));
+         $(document).on('callterminated.jingle', $.proxy(self.onCallTerminated, self));
+         $(document).on('ringing.jingle', $.proxy(self.onCallRinging, self));
+
+         $(document).on('remotestreamadded.jingle', $.proxy(self.onRemoteStreamAdded, self));
+         $(document).on('remotestreamremoved.jingle', $.proxy(self.onRemoteStreamRemoved, self));
+         $(document).on('iceconnectionstatechange.jingle', $.proxy(self.onIceConnectionStateChanged, self));
+         $(document).on('nostuncandidates.jingle', $.proxy(self.noStunCandidates, self));
+
+         $(document).on('error.jingle', function(ev, sid, error) {
+            jsxc.error('[JINGLE]', error);
+         });
+
+         if (self.conn.disco) {
+            self.conn.disco.addFeature('urn:xmpp:jingle:apps:dtls:0');
+         }
+
+         if (self.conn.caps) {
+            $(document).on('caps.strophe', $.proxy(self.onCaps, self));
+         }
+
+         self.getTurnCrendentials();
+      },
+
+      /**
+       * Checks if cached configuration is valid and if necessary update it.
+       * 
+       * @memberOf jsxc.webrtc
+       */
+      getTurnCrendentials: function() {
+
+         if (!jsxc.options.turnCredentialsPath) {
+            jsxc.debug('No path for TURN credentials defined!');
+            return;
+         }
+
+         var ttl = (jsxc.storage.getUserItem('iceValidity') || 0) - (new Date()).getTime();
+         if (ttl > 0) {
+            // credentials valid
+
+            window.setTimeout(jsxc.webrtc.getTurnCrendentials, ttl + 500);
+            return;
+         }
+
+         $.ajax(jsxc.options.turnCredentialsPath, {
+            async: true,
+            success: function(data) {
+               var iceConfig = {
+                  iceServers: [ {
+                     url: 'turn:' + data.url,
+                     credential: data.credential,
+                     username: data.username
+                  } ]
+               };
+
+               jsxc.webrtc.conn.jingle.ice_config = iceConfig;
+               jsxc.storage.setUserItem('iceConfig', iceConfig);
+               jsxc.storage.setUserItem('iceValidity', (new Date()).getTime() + 1000 * data.ttl);
+            },
+            dataType: 'json'
+         });
+      },
+
+      /**
+       * Return list of video capable resources.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param jid
+       * @returns {Array}
+       */
+      getCapableRes: function(jid) {
+         var self = jsxc.webrtc;
+         var bid = jsxc.jidToBid(jid);
+         var res = jsxc.storage.getUserItem('res', bid) || [];
+
+         var available = [];
+         $.each(res, function(r) {
+            if (self.conn.caps.hasFeatureByJid(bid + '/' + r, self.reqVideoFeatures)) {
+               available.push(r);
+            }
+         });
+
+         return available;
+      },
+
+      /**
+       * Add "video" button to roster
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param bid bid of roster item
+       * @param data data wich belongs to bid
+       * @param el the roster item
+       */
+      onAddRosterItem: function(event, bid, data, el) {
+         var self = jsxc.webrtc;
+
+         if (!self.conn) {
+            $(document).one('connectionReady.jsxc', function() {
+               self.onAddRosterItem(null, bid, data, el);
+            });
+            return;
+         }
+
+         var videoIcon = $('<div class="jsxc_video jsxc_disabled" title="' + jsxc.l.Start_video_call + '"></div>');
+
+         videoIcon.click(function() {
+            self.startCall(data.jid);
+            return false;
+         });
+
+         el.find('.jsxc_options.jsxc_left').append(videoIcon);
+
+         el.on('extra.jsxc', function() {
+            self.updateIcon(bid);
+         });
+      },
+
+      /**
+       * Add "video" button to window menu.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param win jQuery window object
+       */
+      initWindow: function(event, win) {
+         var self = jsxc.webrtc;
+
+         jsxc.debug('webrtc.initWindow');
+
+         if (!self.conn) {
+            $(document).one('connectionReady.jsxc', function() {
+               self.initWindow(null, win);
+            });
+            return;
+         }
+
+         var div = $('<div>').addClass('jsxc_video');
+         win.find('.jsxc_transfer:eq(1)').after(div);
+
+         self.updateIcon(jsxc.jidToBid(win.data('jid')));
+      },
+
+      /**
+       * Enable or disable "video" icon and assign full jid.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param bid CSS conform jid
+       */
+      updateIcon: function(bid) {
+         jsxc.debug('Update icon', bid);
+
+         var self = jsxc.webrtc;
+         var win = jsxc.gui.window.get(bid);
+         var jid = win.data('jid') || jsxc.storage.getUserItem('buddy', bid).jid;
+
+         var el = win.find('.jsxc_video').add(jsxc.gui.roster.getItem(bid).find('.jsxc_video'));
+
+         var capableRes = self.getCapableRes(jid);
+         var targetRes = Strophe.getResourceFromJid(jid);
+
+         if (targetRes === null) {
+            $.each(jsxc.storage.getUserItem('buddy', bid).res, function(index, val) {
+               if (capableRes.indexOf(val) > -1) {
+                  targetRes = val;
+                  return false;
+               }
+            });
+         }
+
+         el.off('click');
+
+         if (capableRes.indexOf(targetRes) > -1) {
+            el.click(function() {
+               self.startCall(jid + '/' + targetRes);
+            });
+
+            el.removeClass('jsxc_disabled');
+
+            el.attr('title', jsxc.translate('%%Start video call%%'));
+         } else {
+            el.addClass('jsxc_disabled');
+
+            el.attr('title', jsxc.translate('%%Video call not possible.%%'));
+         }
+      },
+
+      /**
+       * Check if full jid changed.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param e
+       * @param from full jid
+       */
+      onMessage: function(e, from) {
+         var self = jsxc.webrtc;
+         var bid = jsxc.jidToBid(from);
+
+         jsxc.debug('webrtc.onmessage', from);
+
+         if (self.chatJids[bid] !== from) {
+            self.updateIcon(bid);
+            self.chatJids[bid] = from;
+         }
+      },
+
+      /**
+       * Update icon on presence.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param ev
+       * @param status
+       * @private
+       */
+      onPresence: function(ev, jid) {
+         var self = jsxc.webrtc;
+
+         jsxc.debug('webrtc.onpresence', jid);
+
+         self.updateIcon(jsxc.jidToBid(jid));
+      },
+
+      /**
+       * Display status message to user.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param txt message
+       * @param d duration in ms
+       */
+      setStatus: function(txt, d) {
+         var status = $('.jsxc_webrtc .jsxc_status');
+         var duration = (typeof d === 'undefined' || d === null) ? 4000 : d;
+
+         jsxc.debug('[Webrtc]', txt);
+
+         if (status.html()) {
+            // attach old messages
+            txt = status.html() + '<br />' + txt;
+         }
+
+         status.html(txt);
+
+         status.css({
+            'margin-left': '-' + (status.width() / 2) + 'px',
+            opacity: 0,
+            display: 'block'
+         });
+
+         status.stop().animate({
+            opacity: 1
+         });
+
+         clearTimeout(status.data('timeout'));
+
+         if (duration === 0) {
+            return;
+         }
+
+         var to = setTimeout(function() {
+            status.stop().animate({
+               opacity: 0
+            }, function() {
+               status.html('');
+            });
+         }, duration);
+
+         status.data('timeout', to);
+      },
+
+      /**
+       * Update "video" button if we receive cap information.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param jid
+       */
+      onCaps: function(event, jid) {
+         var self = jsxc.webrtc;
+
+         self.updateIcon(jsxc.jidToBid(jid));
+      },
+
+      /**
+       * Called if video/audio is ready. Open window and display some messages.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param stream
+       */
+      onMediaReady: function(event, stream) {
+         jsxc.debug('media ready');
+
+         var self = jsxc.webrtc;
+
+         self.localStream = stream;
+         self.conn.jingle.localStream = stream;
+
+         jsxc.gui.showVideoWindow(self.last_caller);
+
+         var i;
+         for (i = 0; i < stream.getAudioTracks().length; i++) {
+            self.setStatus((stream.getAudioTracks().length > 0) ? 'Use local audio device.' : 'No local audio device.');
+
+            jsxc.debug('using audio device "' + stream.getAudioTracks()[i].label + '"');
+         }
+         for (i = 0; i < stream.getVideoTracks().length; i++) {
+            self.setStatus((stream.getVideoTracks().length > 0) ? 'Use local video device.' : 'No local video device.');
+
+            jsxc.debug('using video device "' + stream.getVideoTracks()[i].label + '"');
+            $('#jsxc_dialog .jsxc_localvideo').show();
+         }
+
+         $(document).one('cleanup.dialog.jsxc', $.proxy(self.hangUp, self));
+         $(document).trigger('finish.mediaready.jsxc');
+      },
+
+      /**
+       * Called if media failes.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      onMediaFailure: function(ev, err) {
+         this.setStatus('media failure');
+
+         jsxc.gui.window.postMessage(jsxc.jidToBid(jsxc.webrtc.last_caller), 'sys', jsxc.translate('%%Media failure%%: ') + err.name);
+         jsxc.debug('media failure: ' + err.name);
+      },
+
+      /**
+       * Called on incoming call.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param sid Session id
+       */
+      onCallIncoming: function(event, sid) {
+         jsxc.debug('incoming call' + sid);
+
+         var self = this;
+         var sess = this.conn.jingle.sessions[sid];
+         var bid = jsxc.jidToBid(sess.peerjid);
+
+         jsxc.gui.window.postMessage(bid, 'sys', jsxc.translate('%%Incoming call.%%'));
+
+         // display notification
+         jsxc.notification.notify(jsxc.translate('%%Incoming call%%'), jsxc.translate('%%from%% ' + bid));
+
+         // send signal to partner
+         sess.sendRinging();
+
+         jsxc.webrtc.last_caller = sess.peerjid;
+
+         jsxc.switchEvents({
+            'mediaready.jingle': function(event, stream) {
+               self.setStatus('Accept call');
+
+               sess.localStream = stream;
+               sess.peerconnection.addStream(stream);
+
+               sess.sendAnswer();
+               sess.accept();
+            },
+            'mediafailure.jingle': function() {
+               sess.sendTerminate('decline');
+               sess.terminate();
+            }
+         });
+
+         if (jsxc.webrtc.AUTO_ACCEPT) {
+            self.reqUserMedia();
+            return;
+         }
+
+         var dialog = jsxc.gui.dialog.open(jsxc.gui.template.get('incomingCall', bid), {
+            noClose: true
+         });
+
+         dialog.find('.jsxc_accept').click(function() {
+            $(document).trigger('accept.call.jsxc');
+
+            self.reqUserMedia();
+         });
+
+         dialog.find('.jsxc_reject').click(function() {
+            jsxc.gui.dialog.close();
+            $(document).trigger('reject.call.jsxc');
+
+            sess.sendTerminate('decline');
+            sess.terminate();
+         });
+      },
+
+      /**
+       * Called if call is terminated.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param sid Session id
+       * @param reason Reason for termination
+       * @param [text] Optional explanation
+       */
+      onCallTerminated: function(event, sid, reason, text) {
+         this.setStatus('call terminated ' + sid + (reason ? (': ' + reason + ' ' + text) : ''));
+
+         var bid = jsxc.jidToBid(jsxc.webrtc.last_caller);
+
+         if (this.localStream) {
+            this.localStream.stop();
+         }
+
+         if ($('.jsxc_videoContainer').length) {
+            $('.jsxc_remotevideo')[0].src = "";
+            $('.jsxc_localvideo')[0].src = "";
+         }
+
+         this.conn.jingle.localStream = null;
+         this.localStream = null;
+         this.remoteStream = null;
+
+         var win = $('#jsxc_dialog .jsxc_chatarea > ul > li');
+         $('#jsxc_windowList > ul').prepend(win.detach());
+         win.find('.slimScrollDiv').resizable('enable');
+
+         $(document).off('cleanup.dialog.jsxc');
+         $(document).off('error.jingle');
+         jsxc.gui.dialog.close();
+
+         jsxc.gui.window.postMessage(bid, 'sys', jsxc.translate('%%Call terminated%%' + (reason ? (': %%' + reason + '%%') : '') + '.'));
+      },
+
+      /**
+       * Remote station is ringing.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      onCallRinging: function() {
+         this.setStatus('ringing...', 0);
+      },
+
+      /**
+       * Called if we receive a remote stream.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param data
+       * @param sid Session id
+       */
+      onRemoteStreamAdded: function(event, data, sid) {
+         this.setStatus('Remote stream for session ' + sid + ' added.');
+
+         var stream = data.stream;
+         this.remoteStream = stream;
+
+         var sess = this.conn.jingle.sessions[sid];
+
+         var isVideoDevice = stream.getVideoTracks().length > 0;
+         var isAudioDevice = stream.getAudioTracks().length > 0;
+
+         sess.remoteDevices = {
+            video: isVideoDevice,
+            audio: isAudioDevice
+         };
+
+         this.setStatus(isVideoDevice ? 'Use remote video device.' : 'No remote video device');
+         this.setStatus(isAudioDevice ? 'Use remote audio device.' : 'No remote audio device');
+
+         if ($('.jsxc_remotevideo').length) {
+            RTC.attachMediaStream($('#jsxc_dialog .jsxc_remotevideo'), stream);
+
+            $('#jsxc_dialog .jsxc_' + (isVideoDevice ? 'remotevideo' : 'noRemoteVideo')).addClass('jsxc_deviceAvailable');
+         }
+      },
+
+      /**
+       * Called if the remote stream was removed.
+       * 
+       * @private
+       * @meberOf jsxc.webrtc
+       * @param event
+       * @param data
+       * @param sid Session id
+       */
+      onRemoteStreamRemoved: function(event, data, sid) {
+         this.setStatus('Remote stream for session ' + sid + ' removed.');
+      },
+
+      /**
+       * Extracts local and remote ip and display it to the user.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param sid session id
+       * @param sess
+       */
+      onIceConnectionStateChanged: function(event, sid, sess) {
+         var sigState = sess.peerconnection.signalingState;
+         var iceCon = sess.peerconnection.iceConnectionState;
+
+         jsxc.debug('iceGat state for ' + sid, sess.peerconnection.iceGatheringState);
+         jsxc.debug('iceCon state for ' + sid, iceCon);
+         jsxc.debug('sig state for ' + sid, sigState);
+
+         if (sigState === 'stable' && (iceCon === 'connected' || iceCon === 'completed')) {
+
+            $('#jsxc_dialog .jsxc_deviceAvailable').show();
+            $('#jsxc_dialog .bubblingG').hide();
+
+            var localSDP = sess.peerconnection.localDescription.sdp;
+            var remoteSDP = sess.peerconnection.remoteDescription.sdp;
+
+            sess.local_fp = SDPUtil.parse_fingerprint(SDPUtil.find_line(localSDP, 'a=fingerprint:')).fingerprint;
+            sess.remote_fp = SDPUtil.parse_fingerprint(SDPUtil.find_line(remoteSDP, 'a=fingerprint:')).fingerprint;
+
+            var ip_regex = "(\\d{1,3}\\.\\d{1,3}.\\d{1,3}\\.\\d{1,3}) \\d+ typ host";
+
+            sess.remote_ip = remoteSDP.match(new RegExp(ip_regex))[1];
+            sess.local_ip = localSDP.match(new RegExp(ip_regex))[1];
+
+            var regex = new RegExp(ip_regex, 'g');
+            var match;
+            while ((match = regex.exec(remoteSDP)) !== null) {
+               if (match[1] !== sess.remote_ip) {
+                  alert('!!! WARNING !!!\n\nPossible Man-in-the-middle attack detected!\n\nYou should close the connection.');
+                  return;
+               }
+            }
+
+            var text = '<p>';
+            text += '<b>' + jsxc.translate('%%Local IP%%: ') + '</b>' + sess.local_ip + '<br />';
+            text += '<b>' + jsxc.translate('%%Remote IP%%: ') + '</b>' + sess.remote_ip + '<br />';
+            text += '<b>' + jsxc.translate('%%Local Fingerprint%%: ') + '</b>' + sess.local_fp + '<br />';
+            text += '<b>' + jsxc.translate('%%Remote Fingerprint%%: ') + '</b>' + sess.remote_fp;
+            text += '</p>';
+
+            $('#jsxc_dialog .jsxc_infobar').html(text);
+         } else if (iceCon === 'failed') {
+            jsxc.gui.window.postMessage(jsxc.jidToBid(sess.peerjid), 'sys', jsxc.translate('%%ICE connection failure%%.'));
+
+            $(document).off('cleanup.dialog.jsxc');
+
+            sess.sendTerminate('failed-transport');
+            sess.terminate();
+
+            $(document).trigger('callterminated.jingle');
+         }
+      },
+
+      /**
+       * No STUN candidates found
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      noStunCandidates: function() {
+
+      },
+
+      /**
+       * Start a call to the specified jid.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param jid full jid
+       * @param um requested user media
+       */
+      startCall: function(jid, um) {
+         var self = this;
+
+         if (Strophe.getResourceFromJid(jid) === null) {
+            jsxc.debug('We need a full jid');
+            return;
+         }
+
+         self.last_caller = jid;
+
+         jsxc.switchEvents({
+            'finish.mediaready.jsxc': function() {
+               self.setStatus('Initiate call');
+
+               jsxc.gui.window.postMessage(jsxc.jidToBid(jid), 'sys', jsxc.translate('%%Call started.%%'));
+
+               $(document).one('error.jingle', function(e, sid, error) {
+                  if (error.source !== 'offer') {
+                     return;
+                  }
+
+                  $(document).off('cleanup.dialog.jsxc');
+                  setTimeout(function() {
+                     jsxc.gui.showAlert("Sorry, we couldn't establish a connection. Maybe your buddy is offline.");
+                  }, 500);
+               });
+
+               self.conn.jingle.initiate(jid, self.conn.jid.toLowerCase());
+            },
+            'mediafailure.jingle': function() {
+               jsxc.gui.dialog.close();
+            }
+         });
+
+         self.reqUserMedia(um);
+      },
+
+      /**
+       * Hang up the current call.
+       * 
+       * @memberOf jsxc.webrtc
+       */
+      hangUp: function(reason, text) {
+         $(document).off('cleanup.dialog.jsxc');
+
+         jsxc.webrtc.conn.jingle.terminate(null, reason, text);
+         $(document).trigger('callterminated.jingle');
+      },
+
+      /**
+       * Request video and audio from local user.
+       * 
+       * @memberOf jsxc.webrtc
+       */
+      reqUserMedia: function(um) {
+         if (this.localStream) {
+            $(document).trigger('mediaready.jingle', [ this.localStream ]);
+            return;
+         }
+
+         um = um || [ 'video', 'audio' ];
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('allowMediaAccess'), {
+            noClose: true
+         });
+         this.setStatus('please allow access to microphone and camera');
+
+         if (typeof MediaStreamTrack !== 'undefined' && typeof MediaStreamTrack.getSources !== 'undefined') {
+            MediaStreamTrack.getSources(function(sourceInfo) {
+               var availableDevices = sourceInfo.map(function(el) {
+
+                  return el.kind;
+               });
+
+               um = um.filter(function(el) {
+                  return availableDevices.indexOf(el) !== -1;
+               });
+
+               getUserMediaWithConstraints(um);
+            });
+         } else {
+            getUserMediaWithConstraints(um);
+         }
+      },
+
+      /**
+       * Make a snapshot from a video stream and display it.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param video Video stream
+       */
+      snapshot: function(video) {
+         if (!video) {
+            jsxc.debug('Missing video element');
+         }
+
+         $('.jsxc_snapshotbar p').remove();
+
+         var canvas = $('<canvas/>').css('display', 'none').appendTo('body').attr({
+            width: video.width(),
+            height: video.height()
+         }).get(0);
+         var ctx = canvas.getContext('2d');
+
+         ctx.drawImage(video[0], 0, 0);
+         var img = $('<img/>');
+         var url = null;
+
+         try {
+            url = canvas.toDataURL('image/jpeg');
+         } catch (err) {
+            jsxc.warn('Error', err);
+            return;
+         }
+
+         img[0].src = url;
+         var link = $('<a/>').attr({
+            target: '_blank',
+            href: url
+         });
+         link.append(img);
+         $('.jsxc_snapshotbar').append(link);
+
+         canvas.remove();
+      }
+   };
+
+   /**
+    * Display window for video call.
+    * 
+    * @memberOf jsxc.gui
+    */
+   jsxc.gui.showVideoWindow = function(jid) {
+      var self = jsxc.webrtc;
+
+      $(document).one('complete.dialog.jsxc', function() {
+
+         // mute own video element to avoid echoes
+         $('#jsxc_dialog .jsxc_localvideo')[0].muted = true;
+         $('#jsxc_dialog .jsxc_localvideo')[0].volume = 0;
+
+         var rv = $('#jsxc_dialog .jsxc_remotevideo');
+         var lv = $('#jsxc_dialog .jsxc_localvideo');
+
+         lv.draggable({
+            containment: "parent"
+         });
+
+         RTC.attachMediaStream(lv, self.localStream);
+
+         var w_dialog = $('#jsxc_dialog').width();
+         var w_remote = rv.width();
+
+         // fit in video
+         if (w_remote > w_dialog) {
+            var scale = w_dialog / w_remote;
+            var new_h = rv.height() * scale;
+            var new_w = w_dialog;
+            var vc = $('#jsxc_dialog .jsxc_videoContainer');
+
+            rv.height(new_h);
+            rv.width(new_w);
+
+            vc.height(new_h);
+            vc.width(new_w);
+
+            lv.height(lv.height() * scale);
+            lv.width(lv.width() * scale);
+         }
+
+         if (self.remoteStream) {
+            RTC.attachMediaStream(rv, self.remoteStream);
+
+            $('#jsxc_dialog .jsxc_' + (self.remoteStream.getVideoTracks().length > 0 ? 'remotevideo' : 'noRemoteVideo')).addClass('jsxc_deviceAvailable');
+         }
+
+         var toggleMulti = function(elem, open) {
+            $('#jsxc_dialog .jsxc_multi > div').not(elem).slideUp();
+
+            var opt = {
+               complete: jsxc.gui.dialog.resize
+            };
+
+            if (open) {
+               elem.slideDown(opt);
+            } else {
+               elem.slideToggle(opt);
+            }
+         };
+
+         var win = jsxc.gui.window.open(jsxc.jidToBid(jid));
+
+         win.find('.slimScrollDiv').resizable('disable');
+         win.find('.jsxc_textarea').slimScroll({
+            height: 413
+         });
+         win.find('.jsxc_emoticons').css('top', (413 + 6) + 'px');
+
+         $('#jsxc_dialog .jsxc_chatarea ul').append(win.detach());
+
+         $('#jsxc_dialog .jsxc_hangUp').click(function() {
+            jsxc.webrtc.hangUp();
+         });
+
+         $('#jsxc_dialog .jsxc_snapshot').click(function() {
+            jsxc.webrtc.snapshot(rv);
+            toggleMulti($('#jsxc_dialog .jsxc_snapshotbar'), true);
+         });
+
+         $('#jsxc_dialog .jsxc_snapshots').click(function() {
+            toggleMulti($('#jsxc_dialog .jsxc_snapshotbar'));
+         });
+
+         $('#jsxc_dialog .jsxc_showchat').click(function() {
+            var chatarea = $('#jsxc_dialog .jsxc_chatarea');
+
+            if (chatarea.is(':hidden')) {
+               chatarea.show();
+               $('#jsxc_dialog .jsxc_webrtc').width('900');
+               jsxc.gui.dialog.resize({
+                  width: '920px'
+               });
+            } else {
+               chatarea.hide();
+               $('#jsxc_dialog .jsxc_webrtc').width('650');
+               jsxc.gui.dialog.resize({
+                  width: '660px'
+               });
+            }
+         });
+
+         $('#jsxc_dialog .jsxc_info').click(function() {
+            toggleMulti($('#jsxc_dialog .jsxc_infobar'));
+         });
+
+         $('#jsxc_dialog .jsxc_fullscreen').click(function() {
+
+            if ($.support.fullscreen) {
+               // Reset position of localvideo
+               $(document).one('disabled.fullscreen', function() {
+                  lv.removeAttr('style');
+               });
+
+               $('#jsxc_dialog .jsxc_videoContainer').fullscreen();
+            }
+         });
+
+         $('#jsxc_dialog .jsxc_volume').change(function() {
+            rv[0].volume = $(this).val();
+         });
+
+         $('#jsxc_dialog .jsxc_volume').dblclick(function() {
+            $(this).val(0.5);
+         });
+      });
+
+      jsxc.gui.dialog.open(jsxc.gui.template.get('videoWindow'), {
+         noClose: true
+      });
+   };
+
+   $.extend(jsxc.CONST, {
+      KEYCODE_ENTER: 13,
+      KEYCODE_ESC: 27
+   });
+
+   $(document).ready(function() {
+      RTC = setupRTC();
+
+      if (RTC !== null) {
+         RTCPeerconnection = RTC.peerconnection;
+
+         $(document).on('add.roster.jsxc', jsxc.webrtc.onAddRosterItem);
+         $(document).on('init.window.jsxc', jsxc.webrtc.initWindow);
+         $(document).on('attached.jsxc', jsxc.webrtc.init);
+      }
+   });
+
+   $.extend(jsxc.l10n.en, {
+      Please_allow_access_to_microphone_and_camera: 'Please click the "Allow" button at the top, to allow access to microphone and camera.',
+      Incoming_call: 'Incoming call',
+      from: 'from',
+      Do_you_want_to_accept_the_call_from: 'Do you want to accept the call from',
+      Reject: 'Reject',
+      Accept: 'Accept',
+      hang_up: 'hang up',
+      snapshot: 'snapshot',
+      mute_my_audio: 'mute my audio',
+      pause_my_video: 'pause my video',
+      fullscreen: 'fullscreen',
+      Info: 'Info',
+      Local_IP: 'Local IP',
+      Remote_IP: 'Remote IP',
+      Local_Fingerprint: 'Local fingerprint',
+      Remote_Fingerprint: 'Remote fingerprint',
+      Video_call_not_possible: 'Video call not possible. Your buddy does not support video calls.',
+      Start_video_call: 'Start video call'
+   });
+
+   $.extend(jsxc.l10n.de, {
+      Please_allow_access_to_microphone_and_camera: 'Bitte klick auf den "Zulassen" Button oben, um den Zugriff auf Kamera und Mikrofon zu erlauben.',
+      Incoming_call: 'Eingehender Anruf',
+      from: 'von',
+      Do_you_want_to_accept_the_call_from: 'Möchtest Du den Anruf annehmen von',
+      Reject: 'Ablehnen',
+      Accept: 'Annehmen',
+      hang_up: 'Auflegen',
+      snapshot: 'Schnappschuss',
+      mute_my_audio: 'Mein Ton aus',
+      pause_my_video: 'Mein Video pausieren',
+      fullscreen: 'Vollbild',
+      Info: 'Info',
+      Local_IP: 'Lokale IP',
+      Remote_IP: 'Remote IP',
+      Local_Fingerprint: 'Lokaler Fingerprint',
+      Remote_Fingerprint: 'Remote Fingerprint',
+      Video_call_not_possible: 'Videoanruf nicht verfügbar. Dein Gesprächspartner unterstützt keine Videotelefonie.',
+      Start_video_call: 'Starte Videoanruf'
+   });
+
+   $.extend(jsxc.l10n.es, {
+      Please_allow_access_to_microphone_and_camera: 'Por favor, permitir el acceso al micrófono y la cámara.',
+      Incoming_call: 'Llamada entrante',
+      from: 'de',
+      Do_you_want_to_accept_the_call_from: 'Desea aceptar la llamada de',
+      Reject: 'Rechazar',
+      Accept: 'Aceptar',
+      hang_up: 'colgar',
+      snapshot: 'instantánea',
+      mute_my_audio: 'silenciar mi audio',
+      pause_my_video: 'pausar mi vídeo',
+      fullscreen: 'pantalla completa',
+      Info: 'Info',
+      Local_IP: 'IP local',
+      Remote_IP: 'IP remota',
+      Local_Fingerprint: 'Firma digital local',
+      Remote_Fingerprint: 'Firma digital remota',
+      Video_call_not_possible: 'Llamada de vídeo no es posible',
+      Start_video_call: 'Iniciar llamada de vídeo'
+   });
+}(jQuery));
diff --git a/jsxc.lib.js b/jsxc.lib.js
new file mode 100644
index 0000000..93492d2
--- /dev/null
+++ b/jsxc.lib.js
@@ -0,0 +1,5871 @@
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org> <br>
+ * Released under the MIT license
+ * 
+ * Please see http://www.jsxc.org/
+ * 
+ * @author Klaus Herberth <klaus at jsxc.org>
+ * @version 1.0.0
+ * @license MIT
+ */
+
+var jsxc;
+
+(function($) {
+   "use strict";
+
+   /**
+    * JavaScript Xmpp Chat namespace
+    * 
+    * @namespace jsxc
+    */
+   jsxc = {
+      /** Version of jsxc */
+      version: '1.0.0',
+
+      /** True if i'm the master */
+      master: false,
+
+      /** True if the role allocation is finished */
+      role_allocation: false,
+
+      /** Timeout for keepalive */
+      to: null,
+
+      /** Timeout after normal keepalive starts */
+      toBusy: null,
+
+      /** Timeout for notification */
+      toNotification: null,
+
+      /** Timeout delay for notification */
+      toNotificationDelay: 500,
+
+      /** Interval for keep-alive */
+      keepalive: null,
+
+      /** True if last activity was 10 min ago */
+      restore: false,
+
+      /** True if restore is complete */
+      restoreCompleted: false,
+
+      /** True if login through form */
+      triggeredFromForm: false,
+
+      /** True if login through box */
+      triggeredFromBox: false,
+
+      /** True if logout through element click */
+      triggeredFromElement: false,
+
+      /** True if logout through logout click */
+      triggeredFromLogout: false,
+
+      /** last values which we wrote into localstorage (IE workaround) */
+      ls: [],
+
+      /**
+       * storage event is even fired if I write something into storage (IE
+       * workaround) 0: conform, 1: not conform, 2: not shure
+       */
+      storageNotConform: null,
+
+      /** Timeout for storageNotConform test */
+      toSNC: null,
+
+      /** My bar id */
+      bid: null,
+
+      /** Some constants */
+      CONST: {
+         NOTIFICATION_DEFAULT: 'default',
+         NOTIFICATION_GRANTED: 'granted',
+         NOTIFICATION_DENIED: 'denied',
+         STATUS: [ 'offline', 'dnd', 'xa', 'away', 'chat', 'online' ],
+         SOUNDS: {
+            MSG: 'incomingMessage.wav',
+            CALL: 'Rotary-Phone6.mp3',
+            NOTICE: 'Ping1.mp3'
+         },
+         REGEX: {
+            JID: new RegExp('\\b[^"&\'\\/:<>@\\s]+@[\\w-_.]+\\b', 'ig'),
+            URL: new RegExp(/((?:https?:\/\/|www\.|([\w\-]+\.[a-zA-Z]{2,3})(?=\b))(?:(?:[\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*\([\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*\)([\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*[\-A-Za-z0-9+&@#\/%=~_|])?)|(?:[\-A-Za-z0-9+&@#\/%?=~_|!:,.;]*[\-A-Za-z0-9+&@#\/%=~_|]))?)/gi)
+         }
+      },
+
+      /**
+       * Write debug message to console and to log.
+       * 
+       * @memberOf jsxc
+       * @param {String} msg Debug message
+       * @param {Object} data
+       * @param {String} Could be warn|error|null
+       */
+      debug: function(msg, data, level) {
+         if (level) {
+            msg = '[' + level + '] ' + msg;
+         }
+
+         if (data) {
+            if (jsxc.storage.getItem('debug') === true) {
+               console.log(msg, data);
+            }
+
+            // try to convert data to string
+            var d;
+            try {
+               // clone html snippet
+               d = $("<span>").prepend($(data).clone()).html();
+            } catch (err) {
+               try {
+                  d = JSON.stringify(data);
+               } catch (err2) {
+                  d = 'see js console';
+               }
+            }
+
+            jsxc.log = jsxc.log + msg + ': ' + d + '\n';
+         } else {
+            console.log(msg);
+            jsxc.log = jsxc.log + msg + '\n';
+         }
+      },
+
+      /**
+       * Write warn message.
+       * 
+       * @memberOf jsxc
+       * @param {String} msg Warn message
+       * @param {Object} data
+       */
+      warn: function(msg, data) {
+         jsxc.debug(msg, data, 'WARN');
+      },
+
+      /**
+       * Write error message.
+       * 
+       * @memberOf jsxc
+       * @param {String} msg Error message
+       * @param {Object} data
+       */
+      error: function(msg, data) {
+         jsxc.debug(msg, data, 'ERROR');
+      },
+
+      /** debug log */
+      log: '',
+
+      /**
+       * Starts the action
+       * 
+       * @memberOf jsxc
+       * @param {object} options
+       */
+      init: function(options) {
+
+         if (options) {
+            // override default options
+            $.extend(jsxc.options, options);
+         }
+
+         /**
+          * Getter method for options. Saved options will override default one.
+          * 
+          * @param {string} key option key
+          * @returns default or saved option value
+          */
+         jsxc.options.get = function(key) {
+            var local = jsxc.storage.getUserItem('options') || {};
+
+            return local[key] || jsxc.options[key];
+         };
+
+         /**
+          * Setter method for options. Will write into localstorage.
+          * 
+          * @param {string} key option key
+          * @param {object} value option value
+          */
+         jsxc.options.set = function(key, value) {
+            jsxc.storage.updateItem('options', key, value, true);
+         };
+
+         jsxc.storageNotConform = jsxc.storage.getItem('storageNotConform');
+         if (jsxc.storageNotConform === null) {
+            jsxc.storageNotConform = 2;
+         }
+
+         // detect language
+         var lang;
+         if (jsxc.storage.getItem('lang') !== null) {
+            lang = jsxc.storage.getItem('lang');
+         } else if (jsxc.options.autoLang && navigator.language) {
+            lang = navigator.language.substr(0, 2);
+         } else {
+            lang = jsxc.options.defaultLang;
+         }
+
+         // set language
+         jsxc.l = jsxc.l10n.en;
+         $.extend(jsxc.l, jsxc.l10n[lang]);
+
+         // Check localStorage
+         if (typeof (localStorage) === 'undefined') {
+            jsxc.debug("Browser doesn't support localStorage.");
+            return;
+         }
+
+         // Register event listener for the storage event
+         window.addEventListener('storage', jsxc.storage.onStorage, false);
+
+         var lastActivity = jsxc.storage.getItem('lastActivity') || 0;
+
+         if ((new Date()).getTime() - lastActivity < jsxc.options.loginTimeout) {
+            jsxc.restore = true;
+         }
+
+         // Check if we have to establish a new connection
+         if (!jsxc.storage.getItem('rid') || !jsxc.storage.getItem('sid') || !jsxc.restore) {
+
+            // Looking for a login form
+            if (!jsxc.options.loginForm.form || !(jsxc.el_exists(jsxc.options.loginForm.form) && jsxc.el_exists(jsxc.options.loginForm.jid) && jsxc.el_exists(jsxc.options.loginForm.pass))) {
+
+               if (jsxc.options.displayRosterMinimized()) {
+                  // Show minimized roster
+                  jsxc.storage.setUserItem('roster', 'hidden');
+                  jsxc.gui.roster.init();
+                  jsxc.gui.roster.noConnection();
+               }
+
+               return;
+            }
+
+            if (typeof jsxc.options.formFound === 'function') {
+               jsxc.options.formFound.call();
+            }
+
+            // create jquery object
+            var form = jsxc.options.loginForm.form = $(jsxc.options.loginForm.form);
+            var events = form.data('events') || {
+               submit: []
+            };
+            var submits = [];
+
+            // save attached submit events and remove them. Will be reattached
+            // in jsxc.submitLoginForm
+            $.each(events.submit, function(index, val) {
+               submits.push(val.handler);
+            });
+
+            form.data('submits', submits);
+            form.off('submit');
+
+            // Add jsxc login action to form
+            form.submit(function() {
+
+               var settings = jsxc.prepareLogin();
+
+               if (settings !== false && (settings.xmpp.onlogin === "true" || settings.xmpp.onlogin === true)) {
+                  jsxc.triggeredFromForm = true;
+
+                  jsxc.xmpp.login();
+
+                  // Trigger submit in jsxc.xmpp.connected()
+                  return false;
+               }
+
+               return true;
+            });
+
+         } else {
+
+            // Restore old connection
+
+            jsxc.bid = jsxc.jidToBid(jsxc.storage.getItem('jid'));
+
+            jsxc.gui.init();
+
+            // Looking for logout element
+            if (jsxc.options.logoutElement !== null && jsxc.options.logoutElement.length > 0) {
+               jsxc.options.logoutElement.one('click', function() {
+                  jsxc.options.logoutElement = $(this);
+                  jsxc.triggeredFromLogout = true;
+                  return jsxc.xmpp.logout();
+               });
+            }
+
+            if (typeof (jsxc.storage.getItem('alive')) === 'undefined' || !jsxc.restore) {
+               jsxc.onMaster();
+            } else {
+               jsxc.checkMaster();
+            }
+         }
+      },
+
+      /**
+       * Load settings and prepare jid.
+       * 
+       * @memberOf jsxc
+       * @returns Loaded settings
+       */
+      prepareLogin: function() {
+         var username = $(jsxc.options.loginForm.jid).val();
+         var password = $(jsxc.options.loginForm.pass).val();
+
+         if (typeof jsxc.options.loadSettings !== 'function') {
+            jsxc.error('No loadSettings function given. Abort.');
+            return;
+         }
+
+         jsxc.gui.showWaitAlert(jsxc.l.Logging_in);
+
+         var settings = jsxc.options.loadSettings.call(this, username, password);
+
+         if (settings === false || settings === null || typeof settings === 'undefined') {
+            jsxc.warn('No settings provided');
+
+            return false;
+         }
+
+         if (typeof settings.xmpp.username === 'string') {
+            username = settings.xmpp.username;
+         }
+
+         var resource = (settings.xmpp.resource) ? '/' + settings.xmpp.resource : '';
+         var domain = settings.xmpp.domain;
+         var jid;
+
+         if (username.match(/@(.*)$/)) {
+            jid = (username.match(/\/(.*)$/)) ? username : username + resource;
+         } else {
+            jid = username + '@' + domain + resource;
+         }
+
+         if (typeof jsxc.options.loginForm.preJid === 'function') {
+            jid = jsxc.options.loginForm.preJid(jid);
+         }
+
+         jsxc.bid = jsxc.jidToBid(jid);
+
+         settings.xmpp.username = jid.split('@')[0];
+         settings.xmpp.domain = jid.split('@')[1].split('/')[0];
+         settings.xmpp.resource = jid.split('@')[1].split('/')[1] || "";
+
+         $.each(settings, function(key, val) {
+            jsxc.options.set(key, val);
+         });
+
+         jsxc.options.xmpp.jid = jid;
+         jsxc.options.xmpp.password = password;
+
+         return settings;
+      },
+
+      /**
+       * Called if the script is a slave
+       */
+      onSlave: function() {
+         jsxc.debug('I am the slave.');
+
+         jsxc.role_allocation = true;
+
+         jsxc.restoreRoster();
+         jsxc.restoreWindows();
+         jsxc.restoreCompleted = true;
+
+         $(document).trigger('restoreCompleted.jsxc');
+      },
+
+      /**
+       * Called if the script is the master
+       */
+      onMaster: function() {
+         jsxc.debug('I am master.');
+
+         jsxc.master = true;
+
+         // Init local storage
+         jsxc.storage.setItem('alive', 0);
+         jsxc.storage.setItem('alive_busy', 0);
+         if (!jsxc.storage.getUserItem('windowlist')) {
+            jsxc.storage.setUserItem('windowlist', []);
+         }
+
+         // Sending keepalive signal
+         jsxc.startKeepAlive();
+
+         // create or load DSA key and call _onMaster
+         jsxc.otr.createDSA();
+      },
+
+      /**
+       * Second half of the onMaster routine
+       */
+      _onMaster: function() {
+
+         // create otr objects, if we lost the master
+         if (jsxc.role_allocation) {
+            $.each(jsxc.storage.getUserItem('windowlist'), function(index, val) {
+               jsxc.otr.create(val);
+            });
+         }
+
+         jsxc.role_allocation = true;
+
+         if (jsxc.restore && !jsxc.restoreCompleted) {
+            jsxc.restoreRoster();
+            jsxc.restoreWindows();
+            jsxc.restoreCompleted = true;
+
+            $(document).trigger('restoreCompleted.jsxc');
+         }
+
+         // Prepare notifications
+         if (jsxc.restore) {
+            var noti = jsxc.storage.getUserItem('notification') || 2;
+            if (jsxc.options.notification && noti > 0 && jsxc.notification.hasSupport()) {
+               if (jsxc.notification.hasPermission()) {
+                  jsxc.notification.init();
+               } else {
+                  jsxc.notification.prepareRequest();
+               }
+            } else {
+               // No support => disable
+               jsxc.options.notification = false;
+            }
+         }
+
+         jsxc.xmpp.login();
+      },
+
+      /**
+       * Checks if there is a master
+       */
+      checkMaster: function() {
+         jsxc.debug('check master');
+
+         jsxc.to = window.setTimeout(jsxc.onMaster, 1000);
+         jsxc.storage.ink('alive');
+      },
+
+      /**
+       * Start sending keep-alive signal
+       */
+      startKeepAlive: function() {
+         jsxc.keepalive = window.setInterval(jsxc.keepAlive, jsxc.options.timeout - 1000);
+      },
+
+      /**
+       * Sends the keep-alive signal to signal that the master is still there.
+       */
+      keepAlive: function() {
+         jsxc.storage.ink('alive');
+
+         if (jsxc.role_allocation) {
+            jsxc.storage.setItem('lastActivity', (new Date()).getTime());
+         }
+      },
+
+      /**
+       * Send one keep-alive signal with higher timeout, and than resume with
+       * normal signal
+       */
+      keepBusyAlive: function() {
+         if (jsxc.toBusy) {
+            window.clearTimeout(jsxc.toBusy);
+         }
+
+         if (jsxc.keepalive) {
+            window.clearInterval(jsxc.keepalive);
+         }
+
+         jsxc.storage.ink('alive_busy');
+         jsxc.toBusy = window.setTimeout(jsxc.startKeepAlive, jsxc.options.busyTimeout - 1000);
+      },
+
+      /**
+       * Generates a random integer number between 0 and max
+       * 
+       * @param {Integer} max
+       * @return {Integer} random integer between 0 and max
+       */
+      random: function(max) {
+         return Math.floor(Math.random() * max);
+      },
+
+      /**
+       * Checks if there is a element with the given selector
+       * 
+       * @param {String} selector jQuery selector
+       * @return {Boolean}
+       */
+      el_exists: function(selector) {
+         return $(selector).length > 0;
+      },
+
+      /**
+       * Creates a CSS compatible string from a JID
+       * 
+       * @param {type} jid Valid Jabber ID
+       * @returns {String} css Compatible string
+       */
+      jidToCid: function(jid) {
+         jsxc.warn('jsxc.jidToCid is deprecated!');
+
+         var cid = Strophe.getBareJidFromJid(jid).replace('@', '-').replace(/\./g, '-').toLowerCase();
+
+         return cid;
+      },
+
+      /**
+       * Create comparable bar jid.
+       * 
+       * @memberOf jsxc
+       * @param jid
+       * @returns comparable bar jid
+       */
+      jidToBid: function(jid) {
+         return Strophe.getBareJidFromJid(jid).toLowerCase();
+      },
+
+      /**
+       * Restore roster
+       */
+      restoreRoster: function() {
+         var buddies = jsxc.storage.getUserItem('buddylist');
+
+         if (!buddies || buddies.length === 0) {
+            jsxc.debug('No saved buddylist.');
+
+            jsxc.gui.roster.empty();
+
+            return;
+         }
+
+         $.each(buddies, function(index, value) {
+            jsxc.gui.roster.add(value);
+         });
+
+         $(document).trigger('cloaded.roster.jsxc');
+      },
+
+      /**
+       * Restore all windows
+       */
+      restoreWindows: function() {
+         var windows = jsxc.storage.getUserItem('windowlist');
+
+         if (windows === null) {
+            return;
+         }
+
+         $.each(windows, function(index, bid) {
+            var window = jsxc.storage.getUserItem('window', bid);
+
+            if (!window) {
+               jsxc.debug('Associated window-element is missing: ' + bid);
+               return true;
+            }
+
+            jsxc.gui.window.init(bid);
+
+            if (!window.minimize) {
+               jsxc.gui.window.show(bid);
+            } else {
+               jsxc.gui.window.hide(bid);
+            }
+
+            jsxc.gui.window.setText(bid, window.text);
+         });
+      },
+
+      /**
+       * This method submits the specified login form.
+       */
+      submitLoginForm: function() {
+         var form = jsxc.options.loginForm.form.off('submit');
+
+         // Attach original events
+         var submits = form.data('submits') || [];
+         $.each(submits, function(index, val) {
+            form.submit(val);
+         });
+
+         if (form.find('#submit').length > 0) {
+            form.find('#submit').click();
+         } else {
+            form.submit();
+         }
+      },
+
+      /**
+       * Escapes some characters to HTML character
+       */
+      escapeHTML: function(text) {
+         text = text.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');
+         return text.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>');
+      },
+
+      /**
+       * Removes all html tags.
+       * 
+       * @memberOf jsxc
+       * @param text
+       * @returns stripped text
+       */
+      removeHTML: function(text) {
+         return $('<span>').html(text).text();
+      },
+
+      /**
+       * Executes only one of the given events
+       * 
+       * @param {string} obj.key event name
+       * @param {function} obj.value function to execute
+       * @returns {string} namespace of all events
+       */
+      switchEvents: function(obj) {
+         var ns = Math.random().toString(36).substr(2, 12);
+         var self = this;
+
+         $.each(obj, function(key, val) {
+            $(document).one(key + '.' + ns, function() {
+               $(document).off('.' + ns);
+
+               val.apply(self, arguments);
+            });
+         });
+
+         return ns;
+      },
+
+      /**
+       * Checks if tab is hidden.
+       * 
+       * @returns {boolean} True if tab is hidden
+       */
+      isHidden: function() {
+         var hidden = false;
+
+         if (typeof document.hidden !== 'undefined') {
+            hidden = document.hidden;
+         } else if (typeof document.webkitHidden !== 'undefined') {
+            hidden = document.webkitHidden;
+         } else if (typeof document.mozHidden !== 'undefined') {
+            hidden = document.mozHidden;
+         } else if (typeof document.msHidden !== 'undefined') {
+            hidden = document.msHidden;
+         }
+
+         // handle multiple tabs
+         if (hidden && jsxc.master) {
+            jsxc.storage.ink('hidden', 0);
+         } else if (!hidden && !jsxc.master) {
+            jsxc.storage.ink('hidden');
+         }
+
+         return hidden;
+      },
+
+      /**
+       * Replace %%tokens%% with correct translation.
+       * 
+       * @param {String} text Given text
+       * @returns {String} Translated string
+       */
+      translate: function(text) {
+         return text.replace(/%%([a-zA-Z0-9_-}{ .!,?/'@]+)%%/g, function(s, key) {
+            var k = key.replace(/ /gi, '_').replace(/[.!,?/'@]/g, '');
+
+            if (!jsxc.l[k]) {
+               jsxc.warn('No translation for: ' + k);
+            }
+
+            return jsxc.l[k] || key.replace(/_/g, ' ');
+         });
+      },
+   };
+
+   /**
+    * Set some options for the chat.
+    * 
+    * @namespace jsxc.options
+    */
+   jsxc.options = {
+
+      /** name of container application (e.g. owncloud or SOGo) */
+      app_name: 'web applications',
+
+      /** Timeout for the keepalive signal */
+      timeout: 3000,
+
+      /** Timeout for the keepalive signal if the master is busy */
+      busyTimeout: 15000,
+
+      /** OTR options */
+      otr: {
+         ERROR_START_AKE: true
+      },
+
+      /** xmpp options */
+      xmpp: {
+         url: null,
+         jid: null,
+         domain: null,
+         password: null,
+         overwrite: false,
+         onlogin: true
+      },
+
+      /** default xmpp priorities */
+      priority: {
+         online: 0,
+         chat: 0,
+         away: 0,
+         xa: 0,
+         dnd: 0
+      },
+
+      /** If all 3 properties are set, the login form is used */
+      loginForm: {
+         /** jquery object from form */
+         form: null,
+
+         /** jquery object from input element which contains the jid */
+         jid: null,
+
+         /** jquery object from input element which contains the password */
+         pass: null,
+
+         /** manipulate JID from input element */
+         preJid: function(jid) {
+            return jid;
+         }
+      },
+
+      /** jquery object from logout element */
+      logoutElement: null,
+
+      /** How many messages should be logged? */
+      numberOfMsg: 10,
+
+      /** Default language */
+      defaultLang: 'en',
+
+      /** auto language detection */
+      autoLang: true,
+
+      /** Place for roster */
+      rosterAppend: 'body',
+
+      /** Should we use the HTML5 notification API? */
+      notification: true,
+
+      /** duration for notification */
+      popupDuration: 6000,
+
+      /** Absolute path root of JSXC installation */
+      root: '',
+
+      /** Timeout for restore in ms */
+      loginTimeout: 1000 * 60 * 10,
+
+      /**
+       * This function decides wether the roster will be displayed or not if no
+       * connection is found.
+       */
+      displayRosterMinimized: function() {
+         return false;
+      },
+
+      /** Set to true if you want to hide offline buddies. */
+      hideOffline: false,
+
+      /** Mute notification sound? */
+      muteNotification: false,
+
+      /**
+       * If no avatar is found, this function is called.
+       * 
+       * @param jid Jid of that user.
+       * @this {jQuery} Elements to update with probable .jsxc_avatar elements
+       */
+      defaultAvatar: function() {
+
+      },
+
+      /**
+       * Returns permanent saved settings and overwrite default jsxc.options.
+       * 
+       * @memberOf jsxc.options
+       * @param username String username
+       * @param password String password
+       * @returns {object} at least xmpp.url
+       */
+      loadSettings: function() {
+
+      },
+
+      /**
+       * Call this function to save user settings permanent.
+       * 
+       * @memberOf jsxc.options
+       * @param data Holds all data as key/value
+       */
+      saveSettinsPermanent: function() {
+
+      }
+   };
+
+   /**
+    * Handle functions for chat window's and buddylist
+    * 
+    * @namespace jsxc.gui
+    */
+   jsxc.gui = {
+      /** Smilie token to file mapping */
+      emotions: [ [ 'O:-) O:)', 'angel' ], [ '>:-( >:( >:-( >:(', 'angry' ], [ ':-) :)', 'smile' ], [ ':-D :D', 'grin' ], [ ':-( :(', 'sad' ], [ ';-) ;)', 'wink' ], [ ':-P :P', 'tonguesmile' ], [ '=-O', 'surprised' ], [ ':kiss: :-*', 'kiss' ], [ '8-) :cool:', 'sunglassess' ], [ ':\'-( :\'( :&apos;-(', 'crysad' ], [ ':-/', 'doubt' ], [ ':-X :X', 'zip' ], [ ':yes:', 'thumbsup' ], [ ':no:', 'thumbsdown' ], [ ':beer:', 'beer' ], [ ':devil:', 'devil' ], [ ':kiss: :kissing:', 'kissin [...]
+
+      /**
+       * Different uri query actions as defined in XEP-0147.
+       * 
+       * @namespace jsxc.gui.queryActions
+       */
+      queryActions: {
+         /** xmpp:JID?message[;body=TEXT] */
+         message: function(jid, params) {
+            var win = jsxc.gui.window.open(jsxc.jidToBid(jid));
+
+            if (params && typeof params.body === 'string') {
+               win.find('.jsxc_textinput').val(params.body);
+            }
+         },
+
+         /** xmpp:JID?remove */
+         remove: function(jid) {
+            jsxc.gui.showRemoveDialog(jsxc.jidToBid(jid));
+         },
+
+         /** xmpp:JID?subscribe[;name=NAME] */
+         subscribe: function(jid, params) {
+            jsxc.gui.showContactDialog(jid);
+
+            if (params && typeof params.name) {
+               $('#jsxc_alias').val(params.name);
+            }
+         },
+
+         /** xmpp:JID?vcard */
+         vcard: function(jid) {
+            jsxc.gui.showVcard(jid);
+         }
+      },
+
+      /**
+       * Creates application skeleton.
+       * 
+       * @memberOf jsxc.gui
+       */
+      init: function() {
+         $('body').append($(jsxc.gui.template.get('windowList')));
+
+         $(window).resize(jsxc.gui.updateWindowListSB);
+         $('#jsxc_windowList').resize(jsxc.gui.updateWindowListSB);
+
+         $('#jsxc_windowListSB .jsxc_scrollLeft').click(function() {
+            jsxc.gui.scrollWindowListBy(-200);
+         });
+         $('#jsxc_windowListSB .jsxc_scrollRight').click(function() {
+            jsxc.gui.scrollWindowListBy(200);
+         });
+         $('#jsxc_windowList').on('wheel', function(ev) {
+            if ($('#jsxc_windowList').data('isOver')) {
+               jsxc.gui.scrollWindowListBy((ev.originalEvent.wheelDelta > 0) ? 200 : -200);
+            }
+         });
+
+         jsxc.gui.tooltip('#jsxc_windowList');
+
+         jsxc.gui.roster.init();
+
+         // prepare regexp for emotions
+         $.each(jsxc.gui.emotions, function(i, val) {
+            // escape characters
+            var reg = val[0].replace(/(\/|\||\*|\.|\+|\?|\^|\$|\(|\)|\[|\]|\{|\})/g, '\\$1');
+            reg = '(' + reg.split(' ').join('|') + ')';
+            jsxc.gui.emotions[i][2] = new RegExp(reg, 'g');
+         });
+
+         // We need this often, so we creates some template jquery objects
+         jsxc.gui.windowTemplate = $(jsxc.gui.template.get('chatWindow'));
+         jsxc.gui.buddyTemplate = $(jsxc.gui.template.get('rosterBuddy'));
+      },
+
+      /**
+       * Init tooltip plugin for given jQuery selector.
+       * 
+       * @param {String} selector jQuery selector
+       * @memberOf jsxc.gui
+       */
+      tooltip: function(selector) {
+         $(selector).tooltip({
+            show: {
+               delay: 600
+            },
+            content: function() {
+               return $(this).attr('title').replace(/\n/g, '<br />');
+            }
+         });
+      },
+
+      /**
+       * Updates Information in roster and chatbar
+       * 
+       * @param {String} bid bar jid
+       */
+      update: function(bid) {
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         if (!data) {
+            jsxc.debug('No data for ' + bid);
+            return;
+         }
+
+         var ri = jsxc.gui.roster.getItem(bid); // roster item from user
+         var we = jsxc.gui.window.get(bid); // window element from user
+         var ue = ri.add(we); // both
+         var spot = $('.jsxc_spot[data-bid="' + bid + '"]');
+
+         // Attach data to corresponding roster item
+         ri.data(data);
+
+         // Add online status
+         ue.add(spot).removeClass('jsxc_' + jsxc.CONST.STATUS.join(' jsxc_')).addClass('jsxc_' + jsxc.CONST.STATUS[data.status]);
+
+         // Change name and add title
+         ue.find('.jsxc_name').add(spot).text(data.name).attr('title', jsxc.l.is + ' ' + jsxc.CONST.STATUS[data.status]);
+
+         // Update gui according to encryption state
+         switch (data.msgstate) {
+            case 0:
+               we.find('.jsxc_transfer').removeClass('jsxc_enc jsxc_fin').attr('title', jsxc.l.your_connection_is_unencrypted);
+               we.find('.jsxc_settings .jsxc_verification').addClass('jsxc_disabled');
+               we.find('.jsxc_settings .jsxc_transfer').text(jsxc.l.start_private);
+               break;
+            case 1:
+               we.find('.jsxc_transfer').addClass('jsxc_enc').attr('title', jsxc.l.your_connection_is_encrypted);
+               we.find('.jsxc_settings .jsxc_verification').removeClass('jsxc_disabled');
+               we.find('.jsxc_settings .jsxc_transfer').text(jsxc.l.close_private);
+               break;
+            case 2:
+               we.find('.jsxc_settings .jsxc_verification').addClass('jsxc_disabled');
+               we.find('.jsxc_transfer').removeClass('jsxc_enc').addClass('jsxc_fin').attr('title', jsxc.l.your_buddy_closed_the_private_connection);
+               we.find('.jsxc_settings .jsxc_transfer').text(jsxc.l.close_private);
+               break;
+         }
+
+         // update gui according to verification state
+         if (data.trust) {
+            we.find('.jsxc_transfer').addClass('jsxc_trust').attr('title', jsxc.l.your_buddy_is_verificated);
+         } else {
+            we.find('.jsxc_transfer').removeClass('jsxc_trust');
+         }
+
+         // update gui according to subscription state
+         if (data.sub && data.sub !== 'both') {
+            ue.addClass('jsxc_oneway');
+         } else {
+            ue.removeClass('jsxc_oneway');
+         }
+
+         var info = '<b>' + Strophe.getBareJidFromJid(data.jid) + '</b>\n';
+         info += jsxc.translate('%%Subscription%%: %%' + data.sub + '%%\n');
+         info += jsxc.translate('%%Status%%: %%' + jsxc.CONST.STATUS[data.status] + '%%');
+
+         ri.find('.jsxc_name').attr('title', info);
+
+         if (data.avatar && data.avatar.length > 0) {
+            jsxc.gui.updateAvatar(ue, data.jid, data.avatar);
+         } else {
+            jsxc.options.defaultAvatar.call(ue, data.jid);
+         }
+      },
+
+      /**
+       * Update avatar on all given elements.
+       * 
+       * @memberOf jsxc.gui
+       * @param {jQuery} el Elements with subelement .jsxc_avatar
+       * @param {string} jid Jid
+       * @param {string} aid Avatar id (sha1 hash of image)
+       */
+      updateAvatar: function(el, jid, aid) {
+
+         if (typeof aid === 'undefined') {
+            if (typeof jsxc.options.defaultAvatar === 'function') {
+               jsxc.options.defaultAvatar.call(el, jid);
+            }
+            return;
+         }
+
+         var avatarSrc = jsxc.storage.getUserItem('avatar', aid);
+
+         var setAvatar = function(src) {
+            if (src === 0) {
+               jsxc.options.defaultAvatar.call(el, jid);
+               return;
+            }
+
+            el.find('.jsxc_avatar').removeAttr('style');
+
+            el.find('.jsxc_avatar').css({
+               'background-image': 'url(' + src + ')',
+               'text-indent': '999px'
+            });
+         };
+
+         if (avatarSrc !== null) {
+            setAvatar(avatarSrc);
+         } else {
+            jsxc.xmpp.conn.vcard.get(function(stanza) {
+               jsxc.debug('vCard', stanza);
+
+               var vCard = $(stanza).find("vCard > PHOTO");
+               var src;
+
+               if (vCard.length === 0) {
+                  jsxc.debug('No photo provided');
+                  src = 0;
+               } else {
+                  var img = vCard.find('BINVAL').text();
+                  var type = vCard.find('TYPE').text();
+                  src = 'data:' + type + ';base64,' + img;
+               }
+
+               jsxc.storage.setUserItem('avatar', aid, src);
+               setAvatar(src);
+            }, Strophe.getBareJidFromJid(jid), function(msg) {
+               jsxc.warn('Could not load vcard.', msg);
+
+               jsxc.storage.setUserItem('avatar', aid, 0);
+               setAvatar(0);
+            });
+         }
+      },
+
+      /**
+       * Updates scrollbar handlers.
+       * 
+       * @memberOf jsxc.gui
+       */
+      updateWindowListSB: function() {
+
+         if ($('#jsxc_windowList>ul').width() > $('#jsxc_windowList').width()) {
+            $('#jsxc_windowListSB > div').removeClass('jsxc_disabled');
+         } else {
+            $('#jsxc_windowListSB > div').addClass('jsxc_disabled');
+            $('#jsxc_windowList>ul').css('right', '0px');
+         }
+      },
+
+      /**
+       * Scroll window list by offset.
+       * 
+       * @memberOf jsxc.gui
+       * @param offset
+       */
+      scrollWindowListBy: function(offset) {
+
+         var scrollWidth = $('#jsxc_windowList>ul').width();
+         var width = $('#jsxc_windowList').width();
+         var el = $('#jsxc_windowList>ul');
+         var right = parseInt(el.css('right')) - offset;
+         var padding = $("#jsxc_windowListSB").width();
+
+         if (scrollWidth < width) {
+            return;
+         }
+
+         if (right > 0) {
+            right = 0;
+         }
+
+         if (right < width - scrollWidth - padding) {
+            right = width - scrollWidth - padding;
+         }
+
+         el.css('right', right + 'px');
+      },
+
+      /**
+       * Returns the window element
+       * 
+       * @param {String} bid
+       * @returns {jquery} jQuery object of the window element
+       */
+      getWindow: function(bid) {
+         jsxc.warn('jsxc.gui.getWindow is deprecated!');
+
+         return jsxc.gui.window.get(bid);
+      },
+
+      /**
+       * Toggle list with timeout, like menu or settings
+       * 
+       * @memberof jsxc.gui
+       */
+      toggleList: function() {
+         var self = $(this);
+
+         self.disableSelection();
+
+         var ul = self.find('ul');
+         var slideUp = null;
+
+         slideUp = function() {
+            ul.slideUp({
+               complete: function() {
+                  self.removeClass('jsxc_opened');
+               }
+            });
+
+            $('body').off('click', null, slideUp);
+         };
+
+         $(this).click(function() {
+
+            if (ul.is(":hidden")) {
+               // hide other lists
+               $('body').click();
+               $('body').one('click', slideUp);
+            } else {
+               $('body').off('click', null, slideUp);
+            }
+
+            ul.slideToggle();
+
+            window.clearTimeout(ul.data('timer'));
+
+            self.toggleClass('jsxc_opened');
+
+            return false;
+         }).mouseleave(function() {
+            ul.data('timer', window.setTimeout(slideUp, 2000));
+         }).mouseenter(function() {
+            window.clearTimeout(ul.data('timer'));
+         });
+      },
+
+      /**
+       * Creates and show loginbox
+       */
+      showLoginBox: function() {
+         // Set focus to password field
+         $(document).on("complete.dialog.jsxc", function() {
+            $('#jsxc_password').focus();
+         });
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('loginBox'));
+
+         $('#jsxc_dialog').find('form').submit(function() {
+
+            $(this).find('input[type=submit]').prop('disabled', true);
+
+            jsxc.options.loginForm.form = $(this);
+            jsxc.options.loginForm.jid = $(this).find('#jsxc_username');
+            jsxc.options.loginForm.pass = $(this).find('#jsxc_password');
+
+            var settings = jsxc.prepareLogin();
+
+            jsxc.triggeredFromBox = true;
+            jsxc.triggeredFromForm = false;
+
+            if (settings === false) {
+               jsxc.gui.showAuthFail();
+            } else {
+               jsxc.xmpp.login();
+            }
+
+            return false;
+         });
+      },
+
+      /**
+       * Creates and show the fingerprint dialog
+       * 
+       * @param {String} bid
+       */
+      showFingerprints: function(bid) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('fingerprintsDialog', bid));
+      },
+
+      /**
+       * Creates and show the verification dialog
+       * 
+       * @param {String} bid
+       */
+      showVerification: function(bid) {
+
+         // Check if there is a open dialog
+         if ($('#jsxc_dialog').length > 0) {
+            setTimeout(function() {
+               jsxc.gui.showVerification(bid);
+            }, 3000);
+            return;
+         }
+
+         // verification only possible if the connection is encrypted
+         if (jsxc.storage.getUserItem('buddy', bid).msgstate !== OTR.CONST.MSGSTATE_ENCRYPTED) {
+            jsxc.warn('Connection not encrypted');
+            return;
+         }
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('authenticationDialog', bid));
+
+         // Add handler
+
+         $('#jsxc_dialog > div:gt(0)').hide();
+         $('#jsxc_dialog select').change(function() {
+            $('#jsxc_dialog > div:gt(0)').hide();
+            $('#jsxc_dialog > div:eq(' + $(this).prop('selectedIndex') + ')').slideDown({
+               complete: function() {
+                  jsxc.gui.dialog.resize();
+               }
+            });
+         });
+
+         // Manual
+         $('#jsxc_dialog > div:eq(1) a.creation').click(function() {
+            if (jsxc.master) {
+               jsxc.otr.objects[bid].trust = true;
+            }
+
+            jsxc.storage.updateUserItem('buddy', bid, 'trust', true);
+
+            jsxc.gui.dialog.close();
+
+            jsxc.storage.updateUserItem('buddy', bid, 'trust', true);
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.conversation_is_now_verified);
+            jsxc.gui.update(bid);
+         });
+
+         // Question
+         $('#jsxc_dialog > div:eq(2) a.creation').click(function() {
+            var div = $('#jsxc_dialog > div:eq(2)');
+            var sec = div.find('#jsxc_secret2').val();
+            var quest = div.find('#jsxc_quest').val();
+
+            if (sec === '' || quest === '') {
+               // Add information for the user which form is missing
+               div.find('input[value=""]').addClass('jsxc_invalid').keyup(function() {
+                  if ($(this).val().match(/.*/)) {
+                     $(this).removeClass('jsxc_invalid');
+                  }
+               });
+               return;
+            }
+
+            if (jsxc.master) {
+               jsxc.otr.sendSmpReq(bid, sec, quest);
+            } else {
+               jsxc.storage.setUserItem('smp_' + bid, {
+                  sec: sec,
+                  quest: quest
+               });
+            }
+
+            jsxc.gui.dialog.close();
+
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.authentication_query_sent);
+         });
+
+         // Secret
+         $('#jsxc_dialog > div:eq(3) .creation').click(function() {
+            var div = $('#jsxc_dialog > div:eq(3)');
+            var sec = div.find('#jsxc_secret').val();
+
+            if (sec === '') {
+               // Add information for the user which form is missing
+               div.find('#jsxc_secret').addClass('jsxc_invalid').keyup(function() {
+                  if ($(this).val().match(/.*/)) {
+                     $(this).removeClass('jsxc_invalid');
+                  }
+               });
+               return;
+            }
+
+            if (jsxc.master) {
+               jsxc.otr.sendSmpReq(bid, sec);
+            } else {
+               jsxc.storage.setUserItem('smp_' + bid, {
+                  sec: sec,
+                  quest: null
+               });
+            }
+
+            jsxc.gui.dialog.close();
+
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.authentication_query_sent);
+         });
+      },
+
+      /**
+       * Create and show approve dialog
+       * 
+       * @param {type} from valid jid
+       */
+      showApproveDialog: function(from) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('approveDialog'), {
+            'noClose': true
+         });
+
+         $('#jsxc_dialog .jsxc_their_jid').text(Strophe.getBareJidFromJid(from));
+
+         $('#jsxc_dialog .jsxc_deny').click(function(ev) {
+            ev.stopPropagation();
+
+            jsxc.xmpp.resFriendReq(from, false);
+
+            jsxc.gui.dialog.close();
+         });
+
+         $('#jsxc_dialog .jsxc_approve').click(function(ev) {
+            ev.stopPropagation();
+
+            var data = jsxc.storage.getUserItem('buddy', jsxc.jidToBid(from));
+
+            jsxc.xmpp.resFriendReq(from, true);
+
+            // If friendship is not mutual show contact dialog
+            if (!data || data.sub === 'from') {
+               $(document).one('close.dialog.jsxc', function() {
+                  jsxc.gui.showContactDialog(from);
+               });
+            }
+
+            jsxc.gui.dialog.close();
+         });
+      },
+
+      /**
+       * Create and show dialog to add a buddy
+       * 
+       * @param {string} [username] jabber id
+       */
+      showContactDialog: function(username) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('contactDialog'));
+
+         // If we got a friendship request, we would display the username in our
+         // response
+         if (username) {
+            $('#jsxc_username').val(username);
+         }
+
+         $('#jsxc_dialog form').submit(function() {
+            var username = $('#jsxc_username').val();
+            var alias = $('#jsxc_alias').val();
+
+            if (!username.match(/@(.*)$/)) {
+               username += '@' + Strophe.getDomainFromJid(jsxc.storage.getItem('jid'));
+            }
+
+            // Check if the username is valid
+            if (!username || !username.match(jsxc.CONST.REGEX.JID)) {
+               // Add notification
+               $('#jsxc_username').addClass('jsxc_invalid').keyup(function() {
+                  if ($(this).val().match(jsxc.CONST.REGEX.JID)) {
+                     $(this).removeClass('jsxc_invalid');
+                  }
+               });
+               return false;
+            }
+            jsxc.xmpp.addBuddy(username, alias);
+
+            jsxc.gui.dialog.close();
+
+            return false;
+         });
+      },
+
+      /**
+       * Create and show dialog to remove a buddy
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      showRemoveDialog: function(bid) {
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('removeDialog', bid));
+
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         $('#jsxc_dialog .creation').click(function(ev) {
+            ev.stopPropagation();
+
+            if (jsxc.master) {
+               jsxc.xmpp.removeBuddy(data.jid);
+            } else {
+               // inform master
+               jsxc.storage.setUserItem('deletebuddy', bid, {
+                  jid: data.jid
+               });
+            }
+
+            jsxc.gui.dialog.close();
+         });
+      },
+
+      /**
+       * Create and show a wait dialog
+       * 
+       * @param {type} msg message to display to the user
+       * @returns {undefined}
+       */
+      showWaitAlert: function(msg) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('waitAlert', null, msg), {
+            'noClose': true
+         });
+      },
+
+      /**
+       * Create and show a wait dialog
+       * 
+       * @param {type} msg message to display to the user
+       * @returns {undefined}
+       */
+      showAlert: function(msg) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('alert', null, msg));
+      },
+
+      /**
+       * Create and show a auth fail dialog
+       * 
+       * @returns {undefined}
+       */
+      showAuthFail: function() {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('authFailDialog'));
+
+         if (jsxc.triggeredFromBox) {
+            $('#jsxc_dialog .jsxc_cancel').hide();
+         }
+
+         $('#jsxc_dialog .creation').click(function() {
+            jsxc.gui.dialog.close();
+         });
+
+         $('#jsxc_dialog .jsxc_cancel').click(function() {
+            jsxc.submitLoginForm();
+         });
+      },
+
+      /**
+       * Create and show a confirm dialog
+       * 
+       * @param {String} msg Message
+       * @param {function} confirm
+       * @param {function} dismiss
+       * @returns {undefined}
+       */
+      showConfirmDialog: function(msg, confirm, dismiss) {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('confirmDialog', null, msg), {
+            noClose: true
+         });
+
+         if (confirm) {
+            $('#jsxc_dialog .creation').click(confirm);
+         }
+
+         if (dismiss) {
+            $('#jsxc_dialog .jsxc_cancel').click(dismiss);
+         }
+      },
+
+      /**
+       * Show about dialog.
+       * 
+       * @memberOf jsxc.gui
+       */
+      showAboutDialog: function() {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('aboutDialog'));
+
+         $('#jsxc_dialog .jsxc_debuglog').click(function() {
+            jsxc.gui.showDebugLog();
+         });
+      },
+
+      /**
+       * Show debug log.
+       * 
+       * @memberOf jsxc.gui
+       */
+      showDebugLog: function() {
+         var userInfo = '<h3>User information</h3>';
+
+         if (navigator) {
+            var key;
+            for (key in navigator) {
+               if (navigator.hasOwnProperty(key) && typeof navigator[key] === 'string') {
+                  userInfo += '<b>' + key + ':</b> ' + navigator[key] + '<br />';
+               }
+            }
+         }
+
+         if (window.screen) {
+            userInfo += '<b>Height:</b> ' + window.screen.height + '<br />';
+            userInfo += '<b>Width:</b> ' + window.screen.width + '<br />';
+         }
+
+         userInfo += '<b>jsxc version:</b> ' + jsxc.version + '<br />';
+
+         jsxc.gui.dialog.open('<div class="jsxc_log">' + userInfo + '<h3>Log</h3><pre>' + jsxc.escapeHTML(jsxc.log) + '</pre></div>');
+      },
+
+      /**
+       * Show vCard of user with the given bar jid.
+       * 
+       * @memberOf jsxc.gui
+       * @param {String} jid
+       */
+      showVcard: function(jid) {
+         var bid = jsxc.jidToBid(jid);
+         jsxc.gui.dialog.open(jsxc.gui.template.get('vCard', bid));
+
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         if (data) {
+            // Display resources and corresponding information
+            var i, j, res, identities, identity = null, cap, client;
+            for (i = 0; i < data.res.length; i++) {
+               res = data.res[i];
+
+               identities = [];
+               cap = jsxc.xmpp.getCapabilitiesByJid(bid + '/' + res);
+
+               if (cap !== null && cap.identities !== null) {
+                  identities = cap.identities;
+               }
+
+               client = '';
+               for (j = 0; j < identities.length; j++) {
+                  identity = identities[j];
+                  if (identity.category === 'client') {
+                     if (client !== '') {
+                        client += ',\n';
+                     }
+
+                     client += identity.name + ' (' + identity.type + ')';
+                  }
+               }
+
+               var status = jsxc.storage.getUserItem('res', bid)[res];
+
+               $('#jsxc_dialog ul.jsxc_vCard').append('<li class="jsxc_sep"><strong>' + jsxc.translate('%%Resource%%') + ':</strong> ' + res + '</li>');
+               $('#jsxc_dialog ul.jsxc_vCard').append('<li><strong>' + jsxc.translate('%%Client%%') + ':</strong> ' + client + '</li>');
+               $('#jsxc_dialog ul.jsxc_vCard').append('<li>' + jsxc.translate('<strong>%%Status%%:</strong> %%' + jsxc.CONST.STATUS[status] + '%%') + '</li>');
+            }
+         }
+
+         var printProp = function(el, depth) {
+            var content = '';
+
+            el.each(function() {
+               var item = $(this);
+               var children = $(this).children();
+
+               content += '<li>';
+
+               var prop = jsxc.translate('%%' + item[0].tagName + '%%');
+
+               if (prop !== ' ') {
+                  content += '<strong>' + prop + ':</strong> ';
+               }
+
+               if (item[0].tagName === 'PHOTO') {
+
+               } else if (children.length > 0) {
+                  content += '<ul>';
+                  content += printProp(children, depth + 1);
+                  content += '</ul>';
+               } else if (item.text() !== '') {
+                  content += jsxc.escapeHTML(item.text());
+               }
+
+               content += '</li>';
+
+               if (depth === 0 && $('#jsxc_dialog ul.jsxc_vCard').length > 0) {
+                  if ($('#jsxc_dialog ul.jsxc_vCard li.jsxc_sep:first').length > 0) {
+                     $('#jsxc_dialog ul.jsxc_vCard li.jsxc_sep:first').before(content);
+                  } else {
+                     $('#jsxc_dialog ul.jsxc_vCard').append(content);
+                  }
+                  content = '';
+               }
+            });
+
+            if (depth > 0) {
+               return content;
+            }
+         };
+
+         var failedToLoad = function() {
+            if ($('#jsxc_dialog ul.jsxc_vCard').length === 0) {
+               return;
+            }
+
+            $('#jsxc_dialog p').remove();
+
+            var content = '<p>';
+            content += jsxc.translate('%%Sorry, your buddy doesn\'t provide any information.%%');
+            content += '</p>';
+
+            $('#jsxc_dialog').append(content);
+         };
+
+         jsxc.xmpp.loadVcard(bid, function(stanza) {
+
+            if ($('#jsxc_dialog ul.jsxc_vCard').length === 0) {
+               return;
+            }
+
+            $('#jsxc_dialog p').remove();
+
+            var photo = $(stanza).find("vCard > PHOTO");
+
+            if (photo.length > 0) {
+               var img = photo.find('BINVAL').text();
+               var type = photo.find('TYPE').text();
+               var src = 'data:' + type + ';base64,' + img;
+
+               $('#jsxc_dialog h3').before('<img class="jsxc_vCard" src="' + src + '" alt="avatar" />');
+            }
+
+            if ($(stanza).find('vCard').length === 0 || ($(stanza).find('vcard > *').length === 1 && photo.length === 1)) {
+               failedToLoad();
+               return;
+            }
+
+            printProp($(stanza).find('vcard > *'), 0);
+
+         }, failedToLoad);
+      },
+
+      showSettings: function() {
+         jsxc.gui.dialog.open(jsxc.gui.template.get('settings'));
+
+         if (jsxc.options.get('xmpp').overwrite === 'false' || jsxc.options.get('xmpp').overwrite === false) {
+            $('.jsxc_fieldsetXmpp').hide();
+         }
+
+         $('#jsxc_dialog form').each(function() {
+            var self = $(this);
+
+            self.find('input[type!="submit"]').each(function() {
+               var id = this.id.split("-");
+               var prop = id[0];
+               var key = id[1];
+               var type = this.type;
+
+               var data = jsxc.options.get(prop);
+
+               if (data && typeof data[key] !== 'undefined') {
+                  if (type === 'checkbox') {
+                     if (data[key] !== 'false' && data[key] !== false) {
+                        this.checked = 'checked';
+                     }
+                  } else {
+                     $(this).val(data[key]);
+                  }
+               }
+            });
+         });
+
+         $('#jsxc_dialog form').submit(function() {
+
+            var self = $(this);
+            var data = {};
+
+            self.find('input[type!="submit"]').each(function() {
+               var id = this.id.split("-");
+               var prop = id[0];
+               var key = id[1];
+               var val;
+               var type = this.type;
+
+               if (type === 'checkbox') {
+                  val = this.checked;
+               } else {
+                  val = $(this).val();
+               }
+
+               if (!data[prop]) {
+                  data[prop] = {};
+               }
+
+               data[prop][key] = val;
+            });
+
+            $.each(data, function(key, val) {
+               jsxc.options.set(key, val);
+            });
+
+            var err = jsxc.options.saveSettinsPermanent.call(this, data);
+
+            setTimeout(function() {
+               self.find('input[type="submit"]').effect('highlight', {
+                  color: (err) ? 'green' : 'red'
+               }, 4000);
+            }, 200);
+
+            return false;
+         });
+      },
+
+      /**
+       * Show prompt for notification permission.
+       * 
+       * @memberOf jsxc.gui
+       */
+      showRequestNotification: function() {
+         jsxc.gui.showConfirmDialog(jsxc.translate("%%Should we notify you_%%"), function() {
+            jsxc.gui.dialog.open(jsxc.gui.template.get('pleaseAccept'), {
+               noClose: true
+            });
+
+            jsxc.notification.requestPermission();
+         }, function() {
+            $(document).trigger('notificationfailure.jsxc');
+         });
+      },
+
+      showUnknownSender: function(bid) {
+         jsxc.gui.showConfirmDialog(jsxc.translate('%%You_received_a_message_from_an_unknown_sender%% (' + bid + '). %%Do_you_want_to_display_them%%'), function() {
+
+            jsxc.gui.dialog.close();
+
+            jsxc.storage.saveBuddy(bid, {
+               jid: bid,
+               name: bid,
+               status: 0,
+               sub: 'none',
+               res: []
+            });
+
+            jsxc.gui.window.open(bid);
+
+         }, function() {
+            // reset state
+            jsxc.storage.removeUserItem('chat', bid);
+         });
+      },
+
+      /**
+       * Change own presence to pres.
+       * 
+       * @memberOf jsxc.gui
+       * @param pres {CONST.STATUS} New presence state
+       * @param external {boolean} True if triggered from other tab.
+       */
+      changePresence: function(pres, external) {
+
+         if (external !== true) {
+            jsxc.storage.setUserItem('presence', pres);
+         }
+
+         if (jsxc.master) {
+            jsxc.xmpp.sendPres();
+         }
+
+         $('#jsxc_presence > span').text($('#jsxc_presence > ul .jsxc_' + pres).text());
+
+         jsxc.gui.updatePresence('own', pres);
+      },
+
+      /**
+       * Update all presence objects for given user.
+       * 
+       * @memberOf jsxc.gui
+       * @param bid bar jid of user.
+       * @param {CONST.STATUS} pres New presence state.
+       */
+      updatePresence: function(bid, pres) {
+
+         if (bid === 'own') {
+            if (pres === 'dnd') {
+               $('#jsxc_menu .jsxc_muteNotification').addClass('jsxc_disabled');
+               jsxc.notification.muteSound(true);
+            } else {
+               $('#jsxc_menu .jsxc_muteNotification').removeClass('jsxc_disabled');
+
+               if (!jsxc.options.get('muteNotification')) {
+                  jsxc.notification.unmuteSound(true);
+               }
+            }
+         }
+
+         $('.jsxc_presence[data-bid="' + bid + '"]').removeClass('jsxc_' + jsxc.CONST.STATUS.join(' jsxc_')).addClass('jsxc_' + pres);
+      },
+
+      /**
+       * Switch read state to UNread.
+       * 
+       * @memberOf jsxc.gui
+       * @param bid
+       */
+      unreadMsg: function(bid) {
+         var win = jsxc.gui.window.get(bid);
+
+         jsxc.gui.roster.getItem(bid).add(win).addClass('jsxc_unreadMsg');
+         jsxc.storage.updateUserItem('window', bid, 'unread', true);
+      },
+
+      /**
+       * Switch read state to read.
+       * 
+       * @memberOf jsxc.gui
+       * @param bid
+       */
+      readMsg: function(bid) {
+         var win = jsxc.gui.window.get(bid);
+
+         if (win.hasClass('jsxc_unreadMsg')) {
+            jsxc.gui.roster.getItem(bid).add(win).removeClass('jsxc_unreadMsg');
+            jsxc.storage.updateUserItem('window', bid, 'unread', false);
+         }
+      },
+
+      /**
+       * This function searches for URI scheme according to XEP-0147.
+       * 
+       * @memberOf jsxc.gui
+       * @param container In which element should we search?
+       */
+      detectUriScheme: function(container) {
+         container = (container) ? $(container) : $('body');
+
+         container.find("a[href^='xmpp:']").each(function() {
+
+            var element = $(this);
+            var href = element.attr('href').replace(/^xmpp:/, '');
+            var jid = href.split('?')[0];
+            var action, params = {};
+
+            if (href.indexOf('?') < 0) {
+               action = 'message';
+            } else {
+               var pairs = href.substring(href.indexOf('?') + 1).split(';');
+               action = pairs[0];
+
+               var i, key, value;
+               for (i = 1; i < pairs.length; i++) {
+                  key = pairs[i].split('=')[0];
+                  value = (pairs[i].indexOf('=') > 0) ? pairs[i].substring(pairs[i].indexOf('=') + 1) : null;
+
+                  params[decodeURIComponent(key)] = decodeURIComponent(value);
+               }
+            }
+
+            if (typeof jsxc.gui.queryActions[action] === 'function') {
+               element.addClass('jsxc_uriScheme jsxc_uriScheme_' + action);
+
+               element.off('click').click(function(ev) {
+                  ev.stopPropagation();
+
+                  jsxc.gui.queryActions[action].call(jsxc, jid, params);
+
+                  return false;
+               });
+            }
+         });
+      },
+
+      detectEmail: function(container) {
+         container = (container) ? $(container) : $('body');
+
+         container.find('a[href^="mailto:"]').each(function() {
+            var spot = $("<span>X</span>").addClass("jsxc_spot");
+            var href = $(this).attr("href").replace(/^ *mailto:/, "").trim();
+
+            if (href !== '' && href !== Strophe.getBareJidFromJid(jsxc.storage.getItem("jid"))) {
+               var bid = jsxc.jidToBid(href);
+               var self = $(this);
+               var s = self.prev();
+
+               if (!s.hasClass('jsxc_spot')) {
+                  s = spot.clone().attr('data-bid', bid);
+
+                  self.before(s);
+               }
+
+               s.off('click');
+
+               if (jsxc.storage.getUserItem('buddy', bid)) {
+                  jsxc.gui.update(bid);
+                  s.click(function() {
+                     jsxc.gui.window.open(bid);
+
+                     return false;
+                  });
+               } else {
+                  s.click(function() {
+                     jsxc.gui.showContactDialog(href);
+
+                     return false;
+                  });
+               }
+            }
+         });
+      }
+   };
+
+   /**
+    * Handle functions related to the gui of the roster
+    * 
+    * @namespace jsxc.gui.roster
+    */
+   jsxc.gui.roster = {
+
+      /**
+       * Init the roster skeleton
+       * 
+       * @memberOf jsxc.gui.roster
+       * @returns {undefined}
+       */
+      init: function() {
+         $(jsxc.options.rosterAppend + ':first').append($(jsxc.gui.template.get('roster')));
+
+         if (jsxc.options.get('hideOffline')) {
+            $('#jsxc_menu .jsxc_hideOffline').text(jsxc.translate('%%Show offline%%'));
+            $('#jsxc_buddylist').addClass('jsxc_hideOffline');
+         }
+
+         $('#jsxc_menu .jsxc_settings').click(function() {
+            jsxc.gui.showSettings();
+         });
+
+         $('#jsxc_menu .jsxc_hideOffline').click(function() {
+            var hideOffline = !jsxc.options.get('hideOffline');
+
+            if (hideOffline) {
+               $('#jsxc_buddylist').addClass('jsxc_hideOffline');
+            } else {
+               $('#jsxc_buddylist').removeClass('jsxc_hideOffline');
+            }
+
+            $(this).text(hideOffline ? jsxc.translate('%%Show offline%%') : jsxc.translate('%%Hide offline%%'));
+
+            jsxc.options.set('hideOffline', hideOffline);
+         });
+
+         if (jsxc.options.get('muteNotification')) {
+            jsxc.notification.muteSound();
+         }
+
+         $('#jsxc_menu .jsxc_muteNotification').click(function() {
+
+            if (jsxc.storage.getUserItem('presence') === 'dnd') {
+               return;
+            }
+
+            // invert current choice
+            var mute = !jsxc.options.get('muteNotification');
+
+            if (mute) {
+               jsxc.notification.muteSound();
+            } else {
+               jsxc.notification.unmuteSound();
+            }
+         });
+
+         $('#jsxc_roster .jsxc_addBuddy').click(function() {
+            jsxc.gui.showContactDialog();
+         });
+
+         $('#jsxc_roster .jsxc_onlineHelp').click(function() {
+            window.open("http://www.jsxc.org/manual.html", "onlineHelp");
+         });
+
+         $('#jsxc_roster .jsxc_about').click(function() {
+            jsxc.gui.showAboutDialog();
+         });
+
+         $('#jsxc_toggleRoster').click(function() {
+            jsxc.gui.roster.toggle();
+         });
+
+         $('#jsxc_presence > ul > li').click(function() {
+            var self = $(this);
+
+            jsxc.gui.changePresence(self.data('pres'));
+         });
+
+         $('#jsxc_buddylist').slimScroll({
+            distance: '3px',
+            height: ($('#jsxc_roster').height() - 31) + 'px',
+            width: $('#jsxc_buddylist').width() + 'px',
+            color: '#fff',
+            opacity: '0.5'
+         });
+
+         $('#jsxc_roster > .jsxc_bottom > div').each(function() {
+            jsxc.gui.toggleList.call($(this));
+         });
+
+         if (jsxc.storage.getUserItem('roster') === 'hidden') {
+            $('#jsxc_roster').css('right', '-200px');
+            $('#jsxc_windowList > ul').css('paddingRight', '10px');
+         }
+
+         var pres = jsxc.storage.getUserItem('presence') || 'online';
+         $('#jsxc_presence > span').text($('#jsxc_presence > ul .jsxc_' + pres).text());
+         jsxc.gui.updatePresence('own', pres);
+
+         $(document).on('cloaded.roster.jsxc', function() {
+            jsxc.gui.updateAvatar($('#jsxc_avatar'), jsxc.storage.getItem('jid'), 'own');
+         });
+
+         jsxc.gui.tooltip('#jsxc_roster');
+
+         jsxc.notice.load();
+
+         $(document).trigger('ready.roster.jsxc');
+      },
+
+      /**
+       * Create roster item and add it to the roster
+       * 
+       * @param {String} bid bar jid
+       */
+      add: function(bid) {
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var bud = jsxc.gui.buddyTemplate.clone().attr('data-bid', bid).attr('data-type', data.type || 'chat');
+
+         jsxc.gui.roster.insert(bid, bud);
+
+         bud.click(function() {
+            jsxc.gui.window.open(bid);
+         });
+
+         bud.find('.jsxc_chaticon').click(function() {
+            jsxc.gui.window.open(bid);
+         });
+
+         bud.find('.jsxc_rename').click(function() {
+            jsxc.gui.roster.rename(bid);
+            return false;
+         });
+
+         bud.find('.jsxc_delete').click(function() {
+            jsxc.gui.showRemoveDialog(bid);
+            return false;
+         });
+
+         var expandClick = function() {
+            bud.trigger('extra.jsxc');
+
+            bud.toggleClass('jsxc_expand');
+
+            jsxc.gui.updateAvatar(bud, data.jid, data.avatar);
+            return false;
+         };
+
+         bud.find('.jsxc_control').click(expandClick);
+         bud.dblclick(expandClick);
+
+         bud.find('.jsxc_vcardicon').click(function() {
+            jsxc.gui.showVcard(data.jid);
+            return false;
+         });
+
+         jsxc.gui.update(bid);
+
+         // update scrollbar
+         $('#jsxc_buddylist').slimScroll({
+            scrollTo: '0px'
+         });
+
+         $(document).trigger('add.roster.jsxc', [ bid, data, bud ]);
+      },
+
+      getItem: function(bid) {
+         return $("#jsxc_buddylist > li[data-bid='" + bid + "']");
+      },
+
+      /**
+       * Insert roster item. First order: online > away > offline. Second order:
+       * alphabetical of the name
+       * 
+       * @param {type} bid
+       * @param {jquery} li roster item which should be insert
+       * @returns {undefined}
+       */
+      insert: function(bid, li) {
+
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var listElements = $('#jsxc_buddylist > li');
+         var insert = false;
+
+         // Insert buddy with no mutual friendship to the end
+         var status = (data.sub === 'both') ? data.status : -1;
+
+         listElements.each(function() {
+
+            var thisStatus = ($(this).data('sub') === 'both') ? $(this).data('status') : -1;
+
+            if (($(this).data('name').toLowerCase() > data.name.toLowerCase() && thisStatus === status) || thisStatus < status) {
+
+               $(this).before(li);
+               insert = true;
+
+               return false;
+            }
+         });
+
+         if (!insert) {
+            li.appendTo('#jsxc_buddylist');
+         }
+      },
+
+      /**
+       * Initiate reorder of roster item
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      reorder: function(bid) {
+         jsxc.gui.roster.insert(bid, jsxc.gui.roster.remove(bid));
+      },
+
+      /**
+       * Removes buddy from roster
+       * 
+       * @param {String} bid bar jid
+       * @return {JQueryObject} Roster list element
+       */
+      remove: function(bid) {
+         return jsxc.gui.roster.getItem(bid).detach();
+      },
+
+      /**
+       * Removes buddy from roster and clean up
+       * 
+       * @param {String} bid bar compatible jid
+       */
+      purge: function(bid) {
+         if (jsxc.master) {
+            jsxc.storage.removeUserItem('buddy', bid);
+            jsxc.storage.removeUserItem('otr', bid);
+            jsxc.storage.removeUserItem('otr_version_' + bid);
+            jsxc.storage.removeUserItem('chat', bid);
+            jsxc.storage.removeUserItem('window', bid);
+            jsxc.storage.removeUserElement('buddylist', bid);
+            jsxc.storage.removeUserElement('windowlist', bid);
+         }
+
+         jsxc.gui.window._close(bid);
+         jsxc.gui.roster.remove(bid);
+      },
+
+      /**
+       * Create input element for rename action
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      rename: function(bid) {
+         var name = jsxc.gui.roster.getItem(bid).find('.jsxc_name');
+         var options = jsxc.gui.roster.getItem(bid).find('.jsxc_options, .jsxc_control');
+         var input = $('<input type="text" name="name"/>');
+
+         options.hide();
+         name = name.replaceWith(input);
+
+         input.val(name.text());
+         input.keypress(function(ev) {
+            if (ev.which !== 13) {
+               return;
+            }
+
+            options.show();
+            input.replaceWith(name);
+            jsxc.gui.roster._rename(bid, $(this).val());
+
+            $('html').off('click');
+         });
+
+         // Disable html click event, if click on input
+         input.click(function() {
+            return false;
+         });
+
+         $('html').one('click', function() {
+            options.show();
+            input.replaceWith(name);
+            jsxc.gui.roster._rename(bid, input.val());
+         });
+      },
+
+      /**
+       * Rename buddy
+       * 
+       * @param {type} bid
+       * @param {type} newname new name of buddy
+       * @returns {undefined}
+       */
+      _rename: function(bid, newname) {
+         if (jsxc.master) {
+            var d = jsxc.storage.getUserItem('buddy', bid);
+            var iq = $iq({
+               type: 'set'
+            }).c('query', {
+               xmlns: 'jabber:iq:roster'
+            }).c('item', {
+               jid: Strophe.getBareJidFromJid(d.jid),
+               name: newname
+            });
+            jsxc.xmpp.conn.sendIQ(iq);
+         }
+
+         jsxc.storage.updateUserItem('buddy', bid, 'name', newname);
+         jsxc.gui.update(bid);
+      },
+
+      /**
+       * Toogle complete roster
+       * 
+       * @param {Integer} d Duration in ms
+       */
+      toggle: function(d) {
+         var duration = d || 500;
+
+         var roster = $('#jsxc_roster');
+         var wl = $('#jsxc_windowList');
+
+         var roster_width = roster.innerWidth();
+         var roster_right = parseFloat($('#jsxc_roster').css('right'));
+         var state = (roster_right < 0) ? 'shown' : 'hidden';
+
+         jsxc.storage.setUserItem('roster', state);
+
+         roster.animate({
+            right: ((roster_width + roster_right) * -1) + 'px'
+         }, duration);
+         wl.animate({
+            right: (10 - roster_right) + 'px'
+         }, duration);
+
+         $(document).trigger('toggle.roster.jsxc', [ state, duration ]);
+      },
+
+      /**
+       * Shows a text with link to a login box that no connection exists.
+       */
+      noConnection: function() {
+         $('#jsxc_roster .slimScrollDiv').remove();
+         $('#jsxc_roster > .jsxc_bottom').remove();
+
+         $('#jsxc_roster').append($('<p>' + jsxc.l.no_connection + '</p>').append(' <a>' + jsxc.l.relogin + '</a>').click(function() {
+            jsxc.gui.showLoginBox();
+         }));
+      },
+
+      /**
+       * Shows a text with link to add a new buddy.
+       * 
+       * @memberOf jsxc.gui.roster
+       */
+      empty: function() {
+         var text = $('<p>' + jsxc.l.Your_roster_is_empty_add_a + '</p>');
+         var link = $('<a>' + jsxc.l.new_buddy + '</a>');
+
+         link.click(function() {
+            jsxc.gui.showContactDialog();
+         });
+         text.append(link);
+         text.append('.');
+
+         $('#jsxc_roster').prepend(text);
+      }
+   };
+
+   /**
+    * Wrapper for dialog
+    * 
+    * @namespace jsxc.gui.dialog
+    */
+   jsxc.gui.dialog = {
+      /**
+       * Open a Dialog.
+       * 
+       * @memberOf jsxc.gui.dialog
+       * @param {String} data Data of the dialog
+       * @param {Object} [o] Options for the dialog
+       * @param {Boolean} [o.noClose] If true, hide all default close options
+       * @returns {jQuery} Dialog object
+       */
+      open: function(data, o) {
+
+         var opt = o || {};
+
+         // default options
+         var options = {};
+         options = {
+            onComplete: function() {
+               $('#jsxc_dialog .jsxc_close').click(function(ev) {
+                  ev.preventDefault();
+
+                  jsxc.gui.dialog.close();
+               });
+
+               // workaround for old colorbox version (used by firstrunwizard)
+               if (options.closeButton === false) {
+                  $('#cboxClose').hide();
+               }
+
+               $.colorbox.resize();
+
+               $(document).trigger('complete.dialog.jsxc');
+            },
+            onClosed: function() {
+               $(document).trigger('close.dialog.jsxc');
+            },
+            onCleanup: function() {
+               $(document).trigger('cleanup.dialog.jsxc');
+            },
+            opacity: 0.5
+         };
+
+         if (opt.noClose) {
+            options.overlayClose = false;
+            options.escKey = false;
+            options.closeButton = false;
+            delete opt.noClose;
+         }
+
+         $.extend(options, opt);
+
+         options.html = '<div id="jsxc_dialog">' + data + '</div>';
+
+         $.colorbox(options);
+
+         return $('#jsxc_dialog');
+      },
+
+      /**
+       * Close current dialog.
+       */
+      close: function() {
+         jsxc.debug('close dialog');
+         $.colorbox.close();
+      },
+
+      /**
+       * Resizes current dialog.
+       * 
+       * @param {Object} options e.g. width and height
+       */
+      resize: function(options) {
+         $.colorbox.resize(options);
+      }
+   };
+
+   /**
+    * Handle functions related to the gui of the window
+    * 
+    * @namespace jsxc.gui.window
+    */
+   jsxc.gui.window = {
+      /**
+       * Init a window skeleton
+       * 
+       * @memberOf jsxc.gui.window
+       * @param {String} bid
+       * @returns {jQuery} Window object
+       */
+      init: function(bid) {
+         if (jsxc.gui.window.get(bid).length > 0) {
+            return jsxc.gui.window.get(bid);
+         }
+
+         var win = jsxc.gui.windowTemplate.clone().attr('data-bid', bid).hide().appendTo('#jsxc_windowList > ul').show('slow');
+         var data = jsxc.storage.getUserItem('buddy', bid);
+
+         // Attach jid to window
+         win.data('jid', data.jid);
+
+         // Add handler
+
+         jsxc.gui.toggleList.call(win.find('.jsxc_settings'));
+
+         win.find('.jsxc_verification').click(function() {
+            jsxc.gui.showVerification(bid);
+         });
+
+         win.find('.jsxc_fingerprints').click(function() {
+            jsxc.gui.showFingerprints(bid);
+         });
+
+         win.find('.jsxc_transfer').click(function() {
+            jsxc.otr.toggleTransfer(bid);
+         });
+
+         win.find('.jsxc_bar').click(function() {
+            jsxc.gui.window.toggle(bid);
+         });
+
+         win.find('.jsxc_close').click(function() {
+            jsxc.gui.window.close(bid);
+         });
+
+         win.find('.jsxc_clear').click(function() {
+            jsxc.gui.window.clear(bid);
+         });
+
+         win.find('.jsxc_tools').click(function() {
+            return false;
+         });
+
+         win.find('.jsxc_textinput').keyup(function(ev) {
+            var body = $(this).val();
+
+            if (ev.which === 13) {
+               body = '';
+            }
+
+            jsxc.storage.updateUserItem('window', bid, 'text', body);
+
+            if (ev.which === 27) {
+               jsxc.gui.window.close(bid);
+            }
+         }).keypress(function(ev) {
+            if (ev.which !== 13 || !$(this).val()) {
+               return;
+            }
+
+            jsxc.gui.window.postMessage(bid, 'out', $(this).val());
+
+            $(this).val('');
+         }).focus(function() {
+            // remove unread flag
+            jsxc.gui.readMsg(bid);
+         }).mouseenter(function() {
+            $('#jsxc_windowList').data('isOver', true);
+         }).mouseleave(function() {
+            $('#jsxc_windowList').data('isOver', false);
+         });
+
+         win.find('.jsxc_textarea').click(function() {
+            win.find('.jsxc_textinput').focus();
+         });
+
+         win.find('.jsxc_textarea').slimScroll({
+            height: '234px',
+            distance: '3px'
+         });
+
+         win.find('.jsxc_fade').hide();
+
+         win.find('.jsxc_name').disableSelection();
+
+         win.find('.slimScrollDiv').resizable({
+            handles: 'w, nw, n',
+            minHeight: 234,
+            minWidth: 250,
+            resize: function(event, ui) {
+               win.width(ui.size.width);
+               win.find('.jsxc_textarea').slimScroll({
+                  height: ui.size.height
+               });
+               win.find('.jsxc_emoticons').css('top', (ui.size.height + 6) + 'px');
+            }
+         });
+
+         if ($.inArray(bid, jsxc.storage.getUserItem('windowlist')) < 0) {
+
+            // add window to windowlist
+            var wl = jsxc.storage.getUserItem('windowlist');
+            wl.push(bid);
+            jsxc.storage.setUserItem('windowlist', wl);
+
+            // init window element in storage
+            jsxc.storage.setUserItem('window', bid, {
+               minimize: true,
+               text: '',
+               unread: false
+            });
+         } else {
+
+            if (jsxc.storage.getUserItem('window', bid).unread) {
+               jsxc.gui.unreadMsg(bid);
+            }
+         }
+
+         $.each(jsxc.gui.emotions, function(i, val) {
+            var ins = val[0].split(' ')[0];
+            var li = $('<li><div title="' + ins + '" class="jsxc_' + val[1] + '"/></li>');
+            li.click(function() {
+               win.find('input').val(win.find('input').val() + ins);
+               win.find('input').focus();
+            });
+            win.find('.jsxc_emoticons ul').append(li);
+         });
+
+         jsxc.gui.toggleList.call(win.find('.jsxc_emoticons'));
+
+         jsxc.gui.window.restoreChat(bid);
+
+         jsxc.gui.update(bid);
+
+         jsxc.gui.updateWindowListSB();
+
+         // create related otr object
+         if (jsxc.master && !jsxc.otr.objects[bid]) {
+            jsxc.otr.create(bid);
+         } else {
+            jsxc.otr.enable(bid);
+         }
+
+         $(document).trigger('init.window.jsxc', [ win ]);
+
+         return win;
+      },
+
+      /**
+       * Returns the window element
+       * 
+       * @param {String} bid
+       * @returns {jquery} jQuery object of the window element
+       */
+      get: function(id) {
+         return $("li.jsxc_windowItem[data-bid='" + jsxc.jidToBid(id) + "']");
+      },
+
+      /**
+       * Open a window, related to the bid. If the window doesn't exist, it will
+       * be created.
+       * 
+       * @param {String} bid
+       * @returns {jQuery} Window object
+       */
+      open: function(bid) {
+         var win = jsxc.gui.window.init(bid);
+         jsxc.gui.window.show(bid);
+         jsxc.gui.window.highlight(bid);
+
+         var padding = $("#jsxc_windowListSB").width();
+         var innerWidth = $('#jsxc_windowList>ul').width();
+         var outerWidth = $('#jsxc_windowList').width() - padding;
+
+         if (innerWidth > outerWidth) {
+            var offset = parseInt($('#jsxc_windowList>ul').css('right'));
+            var width = win.outerWidth(true);
+
+            var right = innerWidth - win.position().left - width + offset;
+            var left = outerWidth - (innerWidth - win.position().left) - offset;
+
+            if (left < 0) {
+               jsxc.gui.scrollWindowListBy(left * -1);
+            }
+
+            if (right < 0) {
+               jsxc.gui.scrollWindowListBy(right);
+            }
+         }
+
+         return win;
+      },
+
+      /**
+       * Close chatwindow and clean up
+       * 
+       * @param {String} bid bar jid
+       */
+      close: function(bid) {
+
+         if (jsxc.gui.window.get(bid).length === 0) {
+            jsxc.warn('Want to close a window, that is not open.');
+            return;
+         }
+
+         jsxc.storage.removeUserElement('windowlist', bid);
+         jsxc.storage.removeUserItem('window', bid);
+
+         if (jsxc.storage.getUserItem('buddylist').indexOf(bid) < 0) {
+            // delete data from unknown sender
+
+            jsxc.storage.removeUserItem('buddy', bid);
+            jsxc.storage.removeUserItem('chat', bid);
+         }
+
+         jsxc.gui.window._close(bid);
+      },
+
+      /**
+       * Close chatwindow
+       * 
+       * @param {String} bid
+       */
+      _close: function(bid) {
+         jsxc.gui.window.get(bid).hide('slow', function() {
+            $(this).remove();
+
+            jsxc.gui.updateWindowListSB();
+         });
+      },
+
+      /**
+       * Toggle between minimize and maximize of the text area
+       * 
+       * @param {String} bid bar jid
+       */
+      toggle: function(bid) {
+
+         var win = jsxc.gui.window.get(bid);
+
+         if (win.parents("#jsxc_windowList").length === 0) {
+            return;
+         }
+
+         if (win.find('.jsxc_fade').is(':hidden')) {
+            jsxc.gui.window.show(bid);
+         } else {
+            jsxc.gui.window.hide(bid);
+         }
+
+         jsxc.gui.updateWindowListSB();
+      },
+
+      /**
+       * Maximize text area and save
+       * 
+       * @param {String} bid
+       */
+      show: function(bid) {
+
+         jsxc.storage.updateUserItem('window', bid, 'minimize', false);
+
+         jsxc.gui.window._show(bid);
+      },
+
+      /**
+       * Maximize text area
+       * 
+       * @param {String} bid
+       * @returns {undefined}
+       */
+      _show: function(bid) {
+         var win = jsxc.gui.window.get(bid);
+         jsxc.gui.window.get(bid).find('.jsxc_fade').slideDown();
+         win.removeClass('jsxc_min');
+
+         // If the area is hidden, the scrolldown function doesn't work. So we
+         // call it here.
+         jsxc.gui.window.scrollDown(bid);
+
+         if (jsxc.restoreCompleted) {
+            win.find('.jsxc_textinput').focus();
+         }
+
+         win.trigger('show.window.jsxc');
+      },
+
+      /**
+       * Minimize text area and save
+       * 
+       * @param {String} bid
+       */
+      hide: function(bid) {
+         jsxc.storage.updateUserItem('window', bid, 'minimize', true);
+
+         jsxc.gui.window._hide(bid);
+      },
+
+      /**
+       * Minimize text area
+       * 
+       * @param {String} bid
+       */
+      _hide: function(bid) {
+         jsxc.gui.window.get(bid).addClass('jsxc_min').find(' .jsxc_fade').slideUp();
+
+         jsxc.gui.window.get(bid).trigger('hidden.window.jsxc');
+      },
+
+      /**
+       * Highlight window
+       * 
+       * @param {type} bid
+       */
+      highlight: function(bid) {
+         var el = jsxc.gui.window.get(bid).find(' .jsxc_bar');
+
+         if (!el.is(':animated')) {
+            el.effect('highlight', {
+               color: 'orange'
+            }, 2000);
+         }
+      },
+
+      /**
+       * Scroll chat area to the bottom
+       * 
+       * @param {String} bid bar jid
+       */
+      scrollDown: function(bid) {
+         var chat = jsxc.gui.window.get(bid).find('.jsxc_textarea');
+
+         // check if chat exist
+         if (chat.length === 0) {
+            return;
+         }
+
+         chat.slimScroll({
+            scrollTo: (chat.get(0).scrollHeight + 'px')
+         });
+      },
+
+      /**
+       * Write Message to chat area and save
+       * 
+       * @param {String} bid bar jid
+       * @param {String} direction 'in' message is received or 'out' message is
+       *        send
+       * @param {String} msg Message to display
+       */
+      postMessage: function(bid, direction, msg) {
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var html_msg = msg;
+
+         // remove html tags and reencode html tags
+         msg = jsxc.removeHTML(msg);
+         msg = jsxc.escapeHTML(msg);
+
+         // exceptions:
+
+         if (direction === 'out' && data.msgstate === 2) {
+            direction = 'sys';
+            msg = jsxc.l.your_message_wasnt_send_please_end_your_private_conversation;
+         }
+
+         if (direction === 'in' && data.msgstate === 2) {
+            direction = 'sys';
+            msg = jsxc.l.unencrypted_message_received + ' ' + msg;
+         }
+
+         if (direction === 'out' && data.sub === 'from') {
+            direction = 'sys';
+            msg = jsxc.l.your_message_wasnt_send_because_you_have_no_valid_subscription;
+         }
+
+         var post = jsxc.storage.saveMessage(bid, direction, msg);
+
+         if (direction === 'in') {
+            $(document).trigger('postmessagein.jsxc', [ bid, html_msg ]);
+         }
+
+         if (direction === 'out' && jsxc.master) {
+            jsxc.xmpp.sendMessage(bid, html_msg, post.uid);
+         }
+
+         jsxc.gui.window._postMessage(bid, post);
+
+         if (direction === 'out' && msg === '?') {
+            jsxc.gui.window.postMessage(bid, 'sys', '42');
+         }
+      },
+
+      /**
+       * Write Message to chat area
+       * 
+       * @param {String} bid bar jid
+       * @param {Object} post Post object with direction, msg, uid, received
+       * @param {Bool} restore If true no highlights are used and so unread flag
+       *        set
+       */
+      _postMessage: function(bid, post, restore) {
+         var win = jsxc.gui.window.get(bid);
+         var msg = post.msg;
+         var direction = post.direction;
+         var uid = post.uid;
+         var received = post.received || false;
+
+         if (win.find('.jsxc_textinput').is(':not(:focus)') && jsxc.restoreCompleted && direction === 'in' && !restore) {
+            jsxc.gui.window.highlight(bid);
+         }
+
+         msg = msg.replace(jsxc.CONST.REGEX.URL, function(url) {
+
+            var href = (url.match(/^https?:\/\//i)) ? url : 'http://' + url;
+
+            return '<a href="' + href + '" target="_blank">' + url + '</a>';
+         });
+
+         msg = msg.replace(new RegExp('(xmpp:)?(' + jsxc.CONST.REGEX.JID.source + ')(\\?[^\\s]+\\b)?', 'i'), function(match, protocol, jid, action) {
+            if (protocol === 'xmpp:') {
+               if (typeof action === 'string') {
+                  jid += action;
+               }
+
+               return '<a href="xmpp:' + jid + '">' + jid + '</a>';
+            }
+
+            return '<a href="mailto:' + jid + '" target="_blank">' + jid + '</a>';
+         });
+
+         $.each(jsxc.gui.emotions, function(i, val) {
+            msg = msg.replace(val[2], function(match, p1) {
+
+               // escape value for alt and title, this prevents double
+               // replacement
+               var esc = '', i;
+               for (i = 0; i < p1.length; i++) {
+                  esc += '&#' + p1.charCodeAt(i) + ';';
+               }
+
+               return '<div title="' + esc + '" class="jsxc_emoticon jsxc_' + val[1] + '"/>';
+            });
+         });
+
+         var msgDiv = $("<div>");
+         msgDiv.addClass('jsxc_chatmessage jsxc_' + direction);
+         msgDiv.attr('id', uid);
+         msgDiv.html(msg);
+
+         if (received) {
+            msgDiv.addClass('jsxc_received');
+         }
+
+         if (direction === 'sys') {
+            jsxc.gui.window.get(bid).find('.jsxc_textarea').append('<div style="clear:both"/>');
+         }
+
+         win.find('.jsxc_textarea').append(msgDiv);
+
+         jsxc.gui.detectUriScheme(win);
+         jsxc.gui.detectEmail(win);
+
+         jsxc.gui.window.scrollDown(bid);
+
+         // if window has no focus set unread flag
+         if (!win.find('.jsxc_textinput').is(':focus') && jsxc.restoreCompleted && !restore) {
+            jsxc.gui.unreadMsg(bid);
+         }
+      },
+
+      /**
+       * Set text into input area
+       * 
+       * @param {type} bid
+       * @param {type} text
+       * @returns {undefined}
+       */
+      setText: function(bid, text) {
+         jsxc.gui.window.get(bid).find('.jsxc_textinput').val(text);
+      },
+
+      /**
+       * Load old log into chat area
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      restoreChat: function(bid) {
+         var chat = jsxc.storage.getUserItem('chat', bid);
+
+         while (chat !== null && chat.length > 0) {
+            var c = chat.pop();
+            jsxc.gui.window._postMessage(bid, c, true);
+         }
+      },
+
+      /**
+       * Clear chat history
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      clear: function(bid) {
+         jsxc.storage.setUserItem('chat', bid, []);
+         jsxc.gui.window.get(bid).find('.jsxc_textarea').empty();
+      }
+   };
+
+   /**
+    * Hold all HTML templates.
+    * 
+    * @namespace jsxc.gui.template
+    */
+   jsxc.gui.template = {
+      /**
+       * Return requested template and replace all placeholder
+       * 
+       * @memberOf jsxc.gui.template;
+       * @param {type} name template name
+       * @param {type} bid
+       * @param {type} msg
+       * @returns {String} HTML Template
+       */
+      get: function(name, bid, msg) {
+
+         // common placeholder
+         var ph = {
+            my_priv_fingerprint: jsxc.storage.getUserItem('priv_fingerprint') ? jsxc.storage.getUserItem('priv_fingerprint').replace(/(.{8})/g, '$1 ') : jsxc.l.not_available,
+            my_jid: jsxc.storage.getItem('jid') || '',
+            my_node: Strophe.getNodeFromJid(jsxc.storage.getItem('jid') || '') || '',
+            root: jsxc.options.root,
+            app_name: jsxc.options.app_name
+         };
+
+         // placeholder depending on bid
+         if (bid) {
+            var data = jsxc.storage.getUserItem('buddy', bid);
+
+            $.extend(ph, {
+               bid_priv_fingerprint: (data && data.fingerprint) ? data.fingerprint.replace(/(.{8})/g, '$1 ') : jsxc.l.not_available,
+               bid_jid: bid,
+               bid_name: (data && data.name) ? data.name : bid
+            });
+         }
+
+         // placeholder depending on msg
+         if (msg) {
+            $.extend(ph, {
+               msg: msg
+            });
+         }
+
+         var ret = jsxc.gui.template[name];
+
+         if (typeof (ret) === 'string') {
+            ret = jsxc.translate(ret);
+
+            ret = ret.replace(/\{\{([a-zA-Z0-9_\-]+)\}\}/g, function(s, key) {
+               return (typeof ph[key] === 'string') ? ph[key] : s;
+            });
+
+            return ret;
+         }
+
+         jsxc.debug('Template not available: ' + name);
+         return name;
+      },
+      authenticationDialog: '<h3>Verification</h3>\
+            <p>%%Authenticating_a_buddy_helps_%%</p>\
+            <div>\
+              <p style="margin:0px;">%%How_do_you_want_to_authenticate_your_buddy%%</p>\
+              <select size="1">\
+                <option>%%Select_method%%</option>\
+                <option>%%Manual%%</option>\
+                <option>%%Question%%</option>\
+                <option>%%Secret%%</option>\
+              </select>\
+            </div>\
+            <div style="display:none">\
+              <p class=".jsxc_explanation">%%To_verify_the_fingerprint_%%</p>\
+              <p><strong>%%Your_fingerprint%%</strong><br />\
+              <span style="text-transform:uppercase">{{my_priv_fingerprint}}</span></p>\
+              <p><strong>%%Buddy_fingerprint%%</strong><br />\
+              <span style="text-transform:uppercase">{{bid_priv_fingerprint}}</span></p><br />\
+              <p class="jsxc_right"><a href="#" class="jsxc_close button">%%Close%%</a> <a href="#" class="button creation">%%Compared%%</a></p>\
+            </div>\
+            <div style="display:none">\
+              <p class=".jsxc_explanation">%%To_authenticate_using_a_question_%%</p>\
+              <p><label for="jsxc_quest">%%Question%%:</label><input type="text" name="quest" id="jsxc_quest" /></p>\
+              <p><label for="jsxc_secret2">%%Secret%%:</label><input type="text" name="secret2" id="jsxc_secret2" /></p>\
+              <p class="jsxc_right"><a href="#" class="button jsxc_close">%%Close%%</a> <a href="#" class="button creation">%%Ask%%</a></p>\
+            </div>\
+            <div style="display:none">\
+              <p class=".jsxc_explanation">%%To_authenticate_pick_a_secret_%%</p>\
+              <p><label for="jsxc_secret">%%Secret%%:</label><input type="text" name="secret" id="jsxc_secret" /></p>\
+              <p class="jsxc_right"><a href="#" class="button jsxc_close">%%Close%%</a> <a href="#" class="button creation">%%Compare%%</a></p>\
+            </div>',
+      fingerprintsDialog: '<div>\
+          <p class="jsxc_maxWidth">%%A_fingerprint_%%</p>\
+          <p><strong>%%Your_fingerprint%%</strong><br />\
+          <span style="text-transform:uppercase">{{my_priv_fingerprint}}</span></p>\
+          <p><strong>%%Buddy_fingerprint%%</strong><br />\
+          <span style="text-transform:uppercase">{{bid_priv_fingerprint}}</span></p><br />\
+          <p class="jsxc_right"><a href="#" class="button jsxc_close">%%Close%%</a></p>\
+        </div>',
+      chatWindow: '<li class="jsxc_min jsxc_windowItem">\
+            <div class="jsxc_window">\
+                <div class="jsxc_bar">\
+                     <div class="jsxc_avatar">☺</div>\
+                     <div class="jsxc_tools">\
+                           <div class="jsxc_settings">\
+                               <ul>\
+                                   <li class="jsxc_fingerprints jsxc_otr jsxc_disabled">%%Fingerprints%%</li>\
+                                   <li class="jsxc_verification">%%Authentication%%</li>\
+                                   <li class="jsxc_transfer jsxc_otr jsxc_disabled">%%start_private%%</li>\
+                                   <li class="jsxc_clear">%%clear_history%%</li>\
+                               </ul>\
+                           </div>\
+                           <div class="jsxc_transfer jsxc_otr jsxc_disabled"/>\
+                           <div class="jsxc_close">×</div>\
+                     </div>\
+                     <div class="jsxc_name"/>\
+                     <div class="jsxc_cycle"/>\
+                </div>\
+                <div class="jsxc_fade">\
+                   <div class="jsxc_gradient"/>\
+                   <div class="jsxc_textarea"/>\
+                   <div class="jsxc_emoticons"><ul/></div>\
+                   <input type="text" class="jsxc_textinput" placeholder="...%%Message%%" />\
+                </div>\
+            </div>\
+        </li>',
+      roster: '<div id="jsxc_roster">\
+           <ul id="jsxc_buddylist"></ul>\
+           <div class="jsxc_bottom jsxc_presence" data-bid="own">\
+              <div id="jsxc_avatar">\
+                 <div class="jsxc_avatar">☺</div>\
+              </div>\
+              <div id="jsxc_menu">\
+                 <span></span>\
+                 <ul>\
+                     <li class="jsxc_settings">%%Settings%%</li>\
+                     <li class="jsxc_muteNotification">%%Mute%%</li>\
+                     <li class="jsxc_addBuddy">%%Add_buddy%%</li>\
+                     <li class="jsxc_hideOffline">%%Hide offline%%</li>\
+                     <li class="jsxc_onlineHelp">%%Online help%%</li>\
+                     <li class="jsxc_about">%%About%%</li>\
+                 </ul>\
+              </div>\
+              <div id="jsxc_notice">\
+                 <span></span>\
+                 <ul></ul>\
+              </div>\
+              <div id="jsxc_presence">\
+                 <span>%%Online%%</span>\
+                 <ul>\
+                     <li data-pres="online" class="jsxc_online">%%Online%%</li>\
+                     <li data-pres="chat" class="jsxc_chat">%%Chatty%%</li>\
+                     <li data-pres="away" class="jsxc_away">%%Away%%</li>\
+                     <li data-pres="xa" class="jsxc_xa">%%Extended away%%</li>\
+                     <li data-pres="dnd" class="jsxc_dnd">%%dnd%%</li>\
+                     <!-- <li data-pres="offline" class="jsxc_offline">%%Offline%%</li> -->\
+                 </ul>\
+              </div>\
+           </div>\
+           <div id="jsxc_toggleRoster"></div>\
+       </div>',
+      windowList: '<div id="jsxc_windowList">\
+               <ul></ul>\
+            </div>\
+            <div id="jsxc_windowListSB">\
+               <div class="jsxc_scrollLeft jsxc_disabled"><</div>\
+               <div class="jsxc_scrollRight jsxc_disabled">></div>\
+            </div>',
+      rosterBuddy: '<li>\
+            <div class="jsxc_avatar">☺</div>\
+            <div class="jsxc_control"></div>\
+            <div class="jsxc_name"/>\
+            <div class="jsxc_options jsxc_right">\
+                <div class="jsxc_rename" title="%%rename_buddy%%">✎</div>\
+                <div class="jsxc_delete" title="%%delete_buddy%%">✘</div>\
+            </div>\
+            <div class="jsxc_options jsxc_left">\
+                <div class="jsxc_chaticon" title="%%send_message%%"/>\
+                <div class="jsxc_vcardicon" title="%%get_info%%">i</div>\
+            </div>\
+        </li>',
+      loginBox: '<h3>%%Login%%</h3>\
+        <form>\
+            <p><label for="jsxc_username">%%Username%%:</label>\
+               <input type="text" name="username" id="jsxc_username" required="required" value="{{my_node}}"/></p>\
+            <p><label for="jsxc_password">%%Password%%:</label>\
+               <input type="password" name="password" required="required" id="jsxc_password" /></p>\
+            <div class="bottom_submit_section">\
+                <input type="reset" class="button jsxc_close" name="clear" value="%%Cancel%%"/>\
+                <input type="submit" class="button creation" name="commit" value="%%Connect%%"/>\
+            </div>\
+        </form>',
+      contactDialog: '<h3>%%Add_buddy%%</h3>\
+         <p class=".jsxc_explanation">%%Type_in_the_full_username_%%</p>\
+         <form>\
+         <p><label for="jsxc_username">* %%Username%%:</label>\
+            <input type="text" name="username" id="jsxc_username" pattern="^[^\\x22&\'\\/:<>@\\s]+(@[.\\-_\\w]+)?" required="required" /></p>\
+         <p><label for="jsxc_alias">%%Alias%%:</label>\
+            <input type="text" name="alias" id="jsxc_alias" /></p>\
+         <p class="jsxc_right">\
+            <input class="button" type="submit" value="%%Add%%" />\
+         </p>\
+         <form>',
+      approveDialog: '<h3>%%Subscription_request%%</h3>\
+        <p>%%You_have_a_request_from%% <b class="jsxc_their_jid"></b>.</p>\
+        <p class="jsxc_right"><a href="#" class="button jsxc_deny">%%Deny%%</a> <a href="#" class="button creation jsxc_approve">%%Approve%%</a></p>',
+      removeDialog: '<h3>%%Remove buddy%%</h3>\
+        <p class="jsxc_maxWidth">%%You_are_about_to_remove_%%</p>\
+        <p class="jsxc_right"><a href="#" class="button jsxc_cancel jsxc_close">%%Cancel%%</a> <a href="#" class="button creation">%%Remove%%</a></p>',
+      waitAlert: '<h3>{{msg}}</h3>\
+        <p>%%Please_wait%%</p>\
+        <p class="jsxc_center"><img src="{{root}}/img/loading.gif" alt="wait" width="32px" height="32px" /></p>',
+      alert: '<h3>%%Alert%%</h3>\
+        <p>{{msg}}</p>\
+        <p class="jsxc_right"><a href="#" class="button jsxc_close jsxc_cancel">%%Ok%%</a></p>',
+      authFailDialog: '<h3>%%Login_failed%%</h3>\
+        <p>%%Sorry_we_cant_authentikate_%%</p>\
+        <p class="jsxc_right">\
+            <a class="button jsxc_cancel">%%Continue_without_chat%%</a>\
+            <a class="button creation">%%Retry%%</a>\
+        </p>',
+      confirmDialog: '<p>{{msg}}</p>\
+        <p class="jsxc_right">\
+            <a class="button jsxc_cancel jsxc_close">%%Dismiss%%</a>\
+            <a class="button creation">%%Confirm%%</a>\
+        </p>',
+      pleaseAccept: '<p>%%Please_accept_%%</p>',
+      aboutDialog: '<h3>JavaScript XMPP Chat</h3>\
+         <p><b>Version: </b>' + jsxc.version + '<br />\
+         <a href="http://jsxc.org/" target="_blank">www.jsxc.org</a><br />\
+         <br />\
+         <i>Released under the MIT license</i><br />\
+         <br />\
+         Real-time chat app for {{app_name}} and more.<br />\
+         Requires an external <a href="https://xmpp.org/xmpp-software/servers/" target="_blank">XMPP server</a>.<br />\
+         <br />\
+         <b>Credits: </b> <a href="http://www.beepzoid.com/old-phones/" target="_blank">David English (Ringtone)</a>,\
+         <a href="https://soundcloud.com/freefilmandgamemusic/ping-1?in=freefilmandgamemusic/sets/free-notification-sounds-and" target="_blank">CameronMusic (Ping)</a></p>\
+         <p class="jsxc_right"><a class="button jsxc_debuglog" href="#">Show debug log</a></p>',
+      vCard: '<h3>%%Info_about%% {{bid_name}}</h3>\
+         <ul class="jsxc_vCard"></ul>\
+         <p><img src="{{root}}/img/loading.gif" alt="wait" width="32px" height="32px" /> %%Please_wait%%...</p>',
+      settings: '<h3>%%User_settings%%</h3>\
+         <p></p>\
+         <form>\
+            <fieldset class="jsxc_fieldsetXmpp jsxc_fieldset">\
+               <legend>%%Login options%%</legend>\
+               <label for="xmpp-url">%%BOSH url%%</label><input type="text" id="xmpp-url" readonly="readonly"/><br />\
+               <label for="xmpp-username">%%Username%%</label><input type="text" id="xmpp-username"/><br />\
+               <label for="xmpp-domain">%%Domain%%</label><input type="text" id="xmpp-domain"/><br />\
+               <label for="xmpp-resource">%%Resource%%</label><input type="text" id="xmpp-resource"/><br />\
+               <label for="xmpp-onlogin">%%On login%%</label><input type="checkbox" id="xmpp-onlogin" /><br />\
+               <input type="submit" value="%%Save%%"/>\
+            </fieldset>\
+         </form>\
+         <p></p>\
+         <form>\
+            <fieldset class="jsxc_fieldsetPriority jsxc_fieldset">\
+               <legend>%%Priority%%</legend>\
+               <label for="priority-online">%%Online%%</label><input type="number" value="0" id="priority-online" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-chat">%%Chatty%%</label><input type="number" value="0" id="priority-chat" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-away">%%Away%%</label><input type="number" value="0" id="priority-away" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-xa">%%Extended_away%%</label><input type="number" value="0" id="priority-xa" min="-128" max="127" step="1" required="required"/><br />\
+               <label for="priority-dnd">%%dnd%%</label><input type="number" value="0" id="priority-dnd" min="-128" max="127" step="1" required="required"/><br />\
+               <input type="submit" value="%%Save%%"/>\
+            </fieldset>\
+         </form>'
+   };
+
+   /**
+    * Handle XMPP stuff.
+    * 
+    * @namespace jsxc.xmpp
+    */
+   jsxc.xmpp = {
+      conn: null, // connection
+
+      /**
+       * Create new connection or attach to old
+       * 
+       * @memberOf jsxc.xmpp
+       */
+      login: function() {
+
+         var sid = jsxc.storage.getItem('sid');
+         var rid = jsxc.storage.getItem('rid');
+         var jid = jsxc.storage.getItem('jid');
+         var url = jsxc.options.get('xmpp').url;
+
+         // Register eventlistener
+         $(document).on('connected.jsxc', jsxc.xmpp.connected);
+         $(document).on('attached.jsxc', jsxc.xmpp.attached);
+         $(document).on('disconnected.jsxc', jsxc.xmpp.disconnected);
+         $(document).on('ridChange', jsxc.xmpp.onRidChange);
+
+         Strophe.addNamespace('RECEIPTS', 'urn:xmpp:receipts');
+
+         // Create new connection (no login)
+         jsxc.xmpp.conn = new Strophe.Connection(url);
+
+         // Override default function to preserve unique id
+         var stropheGetUniqueId = jsxc.xmpp.conn.getUniqueId;
+         jsxc.xmpp.conn.getUniqueId = function(suffix) {
+            var uid = stropheGetUniqueId.call(jsxc.xmpp.conn, suffix);
+            jsxc.storage.setItem('_uniqueId', jsxc.xmpp.conn._uniqueId);
+
+            return uid;
+         };
+
+         if (jsxc.storage.getItem('debug') === true) {
+            jsxc.xmpp.conn.xmlInput = function(data) {
+               console.log('<', data);
+            };
+            jsxc.xmpp.conn.xmlOutput = function(data) {
+               console.log('>', data);
+            };
+         }
+
+         var callback = function(status, condition) {
+
+            jsxc.debug(Object.getOwnPropertyNames(Strophe.Status)[status] + ': ' + condition);
+
+            switch (status) {
+               case Strophe.Status.CONNECTED:
+                  jsxc.bid = jsxc.jidToBid(jsxc.xmpp.conn.jid.toLowerCase());
+                  $(document).trigger('connected.jsxc');
+                  break;
+               case Strophe.Status.ATTACHED:
+                  $(document).trigger('attached.jsxc');
+                  break;
+               case Strophe.Status.DISCONNECTED:
+                  $(document).trigger('disconnected.jsxc');
+                  break;
+               case Strophe.Status.CONNFAIL:
+                  jsxc.xmpp.onConnfail(condition);
+                  break;
+               case Strophe.Status.AUTHFAIL:
+                  jsxc.gui.showAuthFail();
+                  break;
+            }
+         };
+
+         if (jsxc.xmpp.conn.caps) {
+            jsxc.xmpp.conn.caps.node = 'http://jsxc.org/';
+         }
+
+         if (jsxc.restore && sid && rid) {
+            jsxc.debug('Try to attach');
+            jsxc.debug('SID: ' + sid);
+            jsxc.xmpp.conn.attach(jid, sid, rid, callback);
+         } else {
+            jsxc.debug('New connection');
+
+            jsxc.xmpp.conn.connect(jsxc.options.xmpp.jid, jsxc.options.xmpp.password, callback);
+         }
+      },
+
+      /**
+       * Logs user out of his xmpp session and does some clean up.
+       * 
+       * @returns {Boolean}
+       */
+      logout: function() {
+
+         // instruct all tabs
+         jsxc.storage.removeItem('sid');
+
+         // clean up
+         jsxc.storage.removeUserItem('buddylist');
+         jsxc.storage.removeUserItem('windowlist');
+         jsxc.storage.removeItem('_uniqueId');
+
+         if (!jsxc.master) {
+            $('#jsxc_roster').remove();
+            $('#jsxc_windowlist').remove();
+            return true;
+         }
+
+         if (jsxc.xmpp.conn === null) {
+            return true;
+         }
+
+         // Hide dropdown menu
+         $('body').click();
+
+         jsxc.triggeredFromElement = true;
+
+         // restore all otr objects
+         $.each(jsxc.storage.getUserItem('otrlist'), function(i, val) {
+            jsxc.otr.create(val);
+         });
+
+         var numOtr = Object.keys(jsxc.otr.objects).length + 1;
+         var disReady = function() {
+            if (--numOtr <= 0) {
+               jsxc.xmpp.conn.flush();
+
+               setTimeout(function() {
+                  jsxc.xmpp.conn.disconnect();
+               }, 600);
+            }
+         };
+
+         // end all private conversations
+         $.each(jsxc.otr.objects, function(key, obj) {
+            if (obj.msgstate === OTR.CONST.MSGSTATE_ENCRYPTED) {
+               obj.endOtr.call(obj, function() {
+                  obj.init.call(obj);
+                  jsxc.otr.backup(key);
+
+                  disReady();
+               });
+            } else {
+               disReady();
+            }
+         });
+
+         disReady();
+
+         // Trigger real logout in jsxc.xmpp.disconnected()
+         return false;
+      },
+
+      /**
+       * Triggered if connection is established
+       * 
+       * @private
+       */
+      connected: function() {
+
+         jsxc.xmpp.conn.pause();
+
+         var nomJid = Strophe.getBareJidFromJid(jsxc.xmpp.conn.jid).toLowerCase() + '/' + Strophe.getResourceFromJid(jsxc.xmpp.conn.jid);
+
+         // Save sid and jid
+         jsxc.storage.setItem('sid', jsxc.xmpp.conn._proto.sid);
+         jsxc.storage.setItem('jid', nomJid);
+
+         jsxc.storage.setItem('lastActivity', (new Date()).getTime());
+
+         // make shure roster will be reloaded
+         jsxc.storage.removeUserItem('buddylist');
+
+         jsxc.storage.removeUserItem('windowlist');
+         jsxc.storage.removeUserItem('own');
+         jsxc.storage.removeUserItem('avatar', 'own');
+         jsxc.storage.removeUserItem('otrlist');
+
+         // submit login form
+         if (jsxc.triggeredFromForm) {
+            // Trigger normal submit
+            jsxc.submitLoginForm();
+            return;
+         }
+
+         // reload page after login from login box
+         if (jsxc.triggeredFromBox) {
+            window.location.reload();
+            return;
+         }
+
+         jsxc.xmpp.connectionReady();
+      },
+
+      /**
+       * Triggered if connection is attached
+       * 
+       * @private
+       */
+      attached: function() {
+
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onRosterChanged, 'jabber:iq:roster', 'iq', 'set');
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onMessage, null, 'message', 'chat');
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onReceived, null, 'message');
+         jsxc.xmpp.conn.addHandler(jsxc.xmpp.onPresence, null, 'presence');
+
+         // Only load roaster if necessary
+         if (!jsxc.restore || !jsxc.storage.getUserItem('buddylist')) {
+            // in order to not overide existing presence information, we send
+            // pres first after roster is ready
+            $(document).one('cloaded.roster.jsxc', jsxc.xmpp.sendPres);
+
+            $('#jsxc_roster > p:first').remove();
+
+            var iq = $iq({
+               type: 'get'
+            }).c('query', {
+               xmlns: 'jabber:iq:roster'
+            });
+
+            jsxc.xmpp.conn.sendIQ(iq, jsxc.xmpp.onRoster);
+         } else {
+            jsxc.xmpp.sendPres();
+         }
+
+         jsxc.xmpp.connectionReady();
+      },
+
+      /**
+       * Triggered if the connection is ready
+       */
+      connectionReady: function() {
+
+         // Load saved unique id
+         jsxc.xmpp.conn._uniqueId = jsxc.storage.getItem('_uniqueId') || new Date().getTime();
+
+         $(document).trigger('connectionReady.jsxc');
+      },
+
+      /**
+       * Sends presence stanza to server.
+       */
+      sendPres: function() {
+         // disco stuff
+         if (jsxc.xmpp.conn.disco) {
+            jsxc.xmpp.conn.disco.addIdentity('client', 'web', 'JSXC');
+            jsxc.xmpp.conn.disco.addFeature(Strophe.NS.DISCO_INFO);
+            jsxc.xmpp.conn.disco.addFeature(Strophe.NS.RECEIPTS);
+         }
+
+         // create presence stanza
+         var pres = $pres();
+
+         if (jsxc.xmpp.conn.caps) {
+            // attach caps
+            pres.c('c', jsxc.xmpp.conn.caps.generateCapsAttrs()).up();
+         }
+
+         var presState = jsxc.storage.getUserItem('presence') || 'online';
+         if (presState !== 'online') {
+            pres.c('show').t(presState).up();
+         }
+
+         var priority = jsxc.options.get('priority');
+         if (priority && typeof priority[presState] !== 'undefined' && parseInt(priority[presState]) !== 0) {
+            pres.c('priority').t(priority[presState]).up();
+         }
+
+         jsxc.debug('Send presence', pres.toString());
+         jsxc.xmpp.conn.send(pres);
+      },
+
+      /**
+       * Triggered if lost connection
+       * 
+       * @private
+       */
+      disconnected: function() {
+         jsxc.debug('disconnected');
+
+         jsxc.storage.removeItem('sid');
+         jsxc.storage.removeItem('rid');
+         jsxc.storage.removeItem('lastActivity');
+         jsxc.storage.removeItem('hidden');
+         jsxc.storage.removeUserItem('avatar', 'own');
+         jsxc.storage.removeUserItem('otrlist');
+
+         jsxc.xmpp.conn = null;
+
+         $('#jsxc_windowList').remove();
+
+         if (jsxc.triggeredFromElement) {
+            $('#jsxc_roster').remove();
+
+            if (jsxc.triggeredFromLogout) {
+               window.location = jsxc.options.logoutElement.attr('href');
+            }
+         } else {
+            jsxc.gui.roster.noConnection();
+         }
+
+         window.clearInterval(jsxc.keepalive);
+      },
+
+      /**
+       * Triggered on connection fault
+       * 
+       * @param {String} condition information why we lost the connection
+       * @private
+       */
+      onConnfail: function(condition) {
+         jsxc.debug('XMPP connection failed: ' + condition);
+
+         if (jsxc.triggeredFromForm) {
+            jsxc.submitLoginForm();
+         }
+      },
+
+      /**
+       * Triggered on initial roster load
+       * 
+       * @param {dom} iq
+       * @private
+       */
+      onRoster: function(iq) {
+         /*
+          * <iq from='' type='get' id=''> <query xmlns='jabber:iq:roster'> <item
+          * jid='' name='' subscription='' /> ... </query> </iq>
+          */
+
+         jsxc.debug('Load roster', iq);
+
+         var buddies = [];
+
+         $(iq).find('item').each(function() {
+            var jid = $(this).attr('jid');
+            var name = $(this).attr('name') || jid;
+            var bid = jsxc.jidToBid(jid);
+            var sub = $(this).attr('subscription');
+
+            buddies.push(bid);
+
+            jsxc.storage.removeUserItem('res', bid);
+
+            jsxc.storage.saveBuddy(bid, {
+               jid: jid,
+               name: name,
+               status: 0,
+               sub: sub,
+               res: []
+            });
+
+            jsxc.gui.roster.add(bid);
+         });
+
+         if (buddies.length === 0) {
+            jsxc.gui.roster.empty();
+         }
+
+         jsxc.storage.setUserItem('buddylist', buddies);
+
+         jsxc.debug('Roster loaded');
+         $(document).trigger('cloaded.roster.jsxc');
+      },
+
+      /**
+       * Triggerd on roster changes
+       * 
+       * @param {dom} iq
+       * @returns {Boolean} True to preserve handler
+       * @private
+       */
+      onRosterChanged: function(iq) {
+         /*
+          * <iq from='' type='set' id=''> <query xmlns='jabber:iq:roster'> <item
+          * jid='' name='' subscription='' /> </query> </iq>
+          */
+
+         jsxc.debug('onRosterChanged', iq);
+
+         $(iq).find('item').each(function() {
+            var jid = $(this).attr('jid');
+            var name = $(this).attr('name') || jid;
+            var bid = jsxc.jidToBid(jid);
+            var sub = $(this).attr('subscription');
+            // var ask = $(this).attr('ask');
+
+            if (sub === 'remove') {
+               jsxc.gui.roster.purge(bid);
+            } else {
+               var bl = jsxc.storage.getUserItem('buddylist');
+
+               if (bl.indexOf(bid) < 0) {
+                  bl.push(bid); // (INFO) push returns the new length
+                  jsxc.storage.setUserItem('buddylist', bl);
+               }
+
+               var temp = jsxc.storage.saveBuddy(bid, {
+                  jid: jid,
+                  name: name,
+                  sub: sub
+               });
+
+               if (temp === 'updated') {
+
+                  jsxc.gui.update(bid);
+                  jsxc.gui.roster.reorder(bid);
+               } else {
+                  jsxc.gui.roster.add(bid);
+               }
+            }
+
+            // Remove pending friendship request from notice list
+            if (sub === 'from' || sub === 'both') {
+               var notices = jsxc.storage.getUserItem('notices');
+               var noticeKey = null, notice;
+
+               for (noticeKey in notices) {
+                  notice = notices[noticeKey];
+
+                  if (notice.fnName === 'gui.showApproveDialog' && notice.fnParams[0] === jid) {
+                     jsxc.debug('Remove notice with key ' + noticeKey);
+
+                     jsxc.notice.remove(noticeKey);
+                  }
+               }
+            }
+         });
+
+         if (!jsxc.storage.getUserItem('buddylist') || jsxc.storage.getUserItem('buddylist').length === 0) {
+            jsxc.gui.roster.empty();
+         } else {
+            $('#jsxc_roster > p:first').remove();
+         }
+
+         // preserve handler
+         return true;
+      },
+
+      /**
+       * Triggered on incoming presence stanzas
+       * 
+       * @param {dom} presence
+       * @private
+       */
+      onPresence: function(presence) {
+         /*
+          * <presence xmlns='jabber:client' type='unavailable' from='' to=''/>
+          * 
+          * <presence xmlns='jabber:client' from='' to=''> <priority>5</priority>
+          * <c xmlns='http://jabber.org/protocol/caps'
+          * node='http://psi-im.org/caps' ver='caps-b75d8d2b25' ext='ca cs
+          * ep-notify-2 html'/> </presence>
+          * 
+          * <presence xmlns='jabber:client' from='' to=''> <show>chat</show>
+          * <status></status> <priority>5</priority> <c
+          * xmlns='http://jabber.org/protocol/caps'
+          * node='http://psi-im.org/caps' ver='caps-b75d8d2b25' ext='ca cs
+          * ep-notify-2 html'/> </presence>
+          */
+         jsxc.debug('onPresence', presence);
+
+         var ptype = $(presence).attr('type');
+         var from = $(presence).attr('from');
+         var jid = Strophe.getBareJidFromJid(from).toLowerCase();
+         var to = $(presence).attr('to');
+         to = (to) ? Strophe.getBareJidFromJid(to).toLowerCase() : jid;
+         var r = Strophe.getResourceFromJid(from);
+         var bid = jsxc.jidToBid(jid);
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var res = jsxc.storage.getUserItem('res', bid) || {};
+         var status = null;
+         var xVCard = $(presence).find('x[xmlns="vcard-temp:x:update"]');
+
+         if (jid === to) {
+            return true;
+         }
+
+         if (ptype === 'error') {
+            jsxc.error('[XMPP] ' + $(presence).attr('code'));
+            return true;
+         }
+
+         // incoming friendship request
+         if (ptype === 'subscribe') {
+            jsxc.storage.setUserItem('friendReq', {
+               jid: jid,
+               approve: -1
+            });
+            jsxc.notice.add('%%Friendship request%%', '%%from%% ' + jid, 'gui.showApproveDialog', [ jid ]);
+
+            return true;
+         } else if (ptype === 'unavailable' || ptype === 'unsubscribed') {
+            status = jsxc.CONST.STATUS.indexOf('offline');
+         } else {
+            var show = $(presence).find('show').text();
+            if (show === '') {
+               status = jsxc.CONST.STATUS.indexOf('online');
+            } else {
+               status = jsxc.CONST.STATUS.indexOf(show);
+            }
+         }
+
+         if (status === 0) {
+            delete res[r];
+         } else {
+            res[r] = status;
+         }
+
+         var maxVal = [];
+         var max = 0, prop = null;
+         for (prop in res) {
+            if (res.hasOwnProperty(prop)) {
+               if (max <= res[prop]) {
+                  if (max !== res[prop]) {
+                     maxVal = [];
+                     max = res[prop];
+                  }
+                  maxVal.push(prop);
+               }
+            }
+         }
+
+         if (data.status === 0 && max > 0) {
+            // buddy has come online
+            jsxc.notification.notify(data.name, jsxc.translate('%%has come online%%.'));
+         }
+
+         data.status = max;
+         data.res = maxVal;
+         data.jid = jid;
+
+         // Looking for avatar
+         if (xVCard.length > 0) {
+            var photo = xVCard.find('photo');
+
+            if (photo.length > 0 && photo.text() !== data.avatar) {
+               jsxc.storage.removeUserItem('avatar', data.avatar);
+               data.avatar = photo.text();
+            }
+         }
+
+         // Reset jid
+         if (jsxc.gui.window.get(bid).length > 0) {
+            jsxc.gui.window.get(bid).data('jid', jid);
+         }
+
+         jsxc.storage.setUserItem('buddy', bid, data);
+         jsxc.storage.setUserItem('res', bid, res);
+
+         jsxc.debug('Presence (' + from + '): ' + status);
+
+         jsxc.gui.update(bid);
+         jsxc.gui.roster.reorder(bid);
+
+         $(document).trigger('presence.jsxc', [ from, status, presence ]);
+
+         // preserve handler
+         return true;
+      },
+
+      /**
+       * Triggered on incoming message stanzas
+       * 
+       * @param {dom} presence
+       * @returns {Boolean}
+       * @private
+       */
+      onMessage: function(message) {
+         /*
+          * <message xmlns='jabber:client' type='chat' to='' id='' from=''>
+          * <body>...</body> <active
+          * xmlns='http://jabber.org/protocol/chatstates'/> </message>
+          */
+
+         jsxc.debug('Incoming message', message);
+
+         var type = $(message).attr('type');
+         var from = $(message).attr('from');
+         var mid = $(message).attr('id');
+         var jid = Strophe.getBareJidFromJid(from);
+         var bid = jsxc.jidToBid(jid);
+         var data = jsxc.storage.getUserItem('buddy', bid);
+         var body = $(message).find('body:first').text();
+         var request = $(message).find("request[xmlns='urn:xmpp:receipts']");
+
+         if (!body) {
+            return true;
+         }
+
+         if (data === null) {
+            // jid not in roster
+
+            var chat = jsxc.storage.getUserItem('chat', bid) || [];
+
+            if (chat.length === 0) {
+               jsxc.notice.add('%%Unknown sender%%', '%%You received a message from an unknown sender%% (' + bid + ').', 'gui.showUnknownSender', [ bid ]);
+            }
+
+            var msg = jsxc.removeHTML(body);
+            msg = jsxc.escapeHTML(msg);
+
+            jsxc.storage.saveMessage(bid, 'in', msg);
+
+            return true;
+         }
+
+         var win = jsxc.gui.window.init(bid);
+
+         // If we now the full jid, we use it
+         if (type === 'chat') {
+            win.data('jid', from);
+            jsxc.storage.updateUserItem('buddy', bid, {
+               jid: from
+            });
+         }
+
+         $(document).trigger('message.jsxc', [ from, body ]);
+
+         // create related otr object
+         if (jsxc.master && !jsxc.otr.objects[bid]) {
+            jsxc.otr.create(bid);
+         }
+
+         if (mid !== null && request.length && data !== null && (data.sub === 'both' || data.sub === 'from') && type === 'chat') {
+            // Send received according to XEP-0184
+            jsxc.xmpp.conn.send($msg({
+               to: from
+            }).c('received', {
+               xmlns: 'urn:xmpp:receipts',
+               id: mid
+            }));
+         }
+
+         if (jsxc.otr.objects.hasOwnProperty(bid)) {
+            jsxc.otr.objects[bid].receiveMsg(body);
+         } else {
+            jsxc.gui.window.postMessage(bid, 'in', body);
+         }
+
+         // preserve handler
+         return true;
+      },
+
+      /**
+       * Triggerd if the rid changed
+       * 
+       * @param {event} ev
+       * @param {obejct} data
+       * @private
+       */
+      onRidChange: function(ev, data) {
+         jsxc.storage.setItem('rid', data.rid);
+      },
+
+      /**
+       * response to friendship request
+       * 
+       * @param {string} from jid from original friendship req
+       * @param {boolean} approve
+       */
+      resFriendReq: function(from, approve) {
+         if (jsxc.master) {
+            jsxc.xmpp.conn.send($pres({
+               to: from,
+               type: (approve) ? 'subscribed' : 'unsubscribed'
+            }));
+
+            jsxc.storage.removeUserItem('friendReq');
+            jsxc.gui.dialog.close();
+
+         } else {
+            jsxc.storage.updateUserItem('friendReq', 'approve', approve);
+         }
+      },
+
+      /**
+       * Add buddy to my friends
+       * 
+       * @param {string} username jid
+       * @param {string} alias
+       */
+      addBuddy: function(username, alias) {
+         var bid = jsxc.jidToBid(username);
+
+         if (jsxc.master) {
+            // add buddy to roster (trigger onRosterChanged)
+            var iq = $iq({
+               type: 'set'
+            }).c('query', {
+               xmlns: 'jabber:iq:roster'
+            }).c('item', {
+               jid: username,
+               name: alias || ''
+            });
+            jsxc.xmpp.conn.sendIQ(iq);
+
+            // send subscription request to buddy (trigger onRosterChanged)
+            jsxc.xmpp.conn.send($pres({
+               to: username,
+               type: 'subscribe'
+            }));
+
+            jsxc.storage.removeUserItem('add_' + bid);
+         } else {
+            jsxc.storage.setUserItem('add_' + bid, {
+               username: username,
+               alias: alias || null
+            });
+         }
+      },
+
+      /**
+       * Remove buddy from my friends
+       * 
+       * @param {type} jid
+       */
+      removeBuddy: function(jid) {
+         var bid = jsxc.jidToBid(jid);
+
+         // Shortcut to remove buddy from roster and cancle all subscriptions
+         var iq = $iq({
+            type: 'set'
+         }).c('query', {
+            xmlns: 'jabber:iq:roster'
+         }).c('item', {
+            jid: Strophe.getBareJidFromJid(jid),
+            subscription: 'remove'
+         });
+         jsxc.xmpp.conn.sendIQ(iq);
+
+         jsxc.gui.roster.purge(bid);
+      },
+
+      onReceived: function(message) {
+         var from = $(message).attr('from');
+         var jid = Strophe.getBareJidFromJid(from);
+         var bid = jsxc.jidToBid(jid);
+         var received = $(message).find("received[xmlns='urn:xmpp:receipts']");
+
+         if (received.length) {
+            var receivedId = received.attr('id').replace(/:/, '-');
+            var chat = jsxc.storage.getUserItem('chat', bid);
+            var i;
+
+            for (i = chat.length - 1; i >= 0; i--) {
+               if (chat[i].uid === receivedId) {
+                  chat[i].received = true;
+
+                  $('#' + receivedId).addClass('jsxc_received');
+
+                  jsxc.storage.setUserItem('chat', bid, chat);
+                  break;
+               }
+            }
+         }
+
+         return true;
+      },
+
+      /**
+       * Public function to send message.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param bid css jid of user
+       * @param msg message
+       * @param uid unique id
+       */
+      sendMessage: function(bid, msg, uid) {
+         if (jsxc.otr.objects.hasOwnProperty(bid)) {
+            jsxc.otr.objects[bid].sendMsg(msg, uid);
+         } else {
+            jsxc.xmpp._sendMessage(jsxc.gui.window.get(bid).data('jid'), msg, uid);
+         }
+      },
+
+      /**
+       * Create message stanza and send it.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param jid Jabber id
+       * @param msg Message
+       * @param uid unique id
+       * @private
+       */
+      _sendMessage: function(jid, msg, uid) {
+         var data = jsxc.storage.getUserItem('buddy', jsxc.jidToBid(jid)) || {};
+         var isBar = (Strophe.getBareJidFromJid(jid) === jid);
+         var type = data.type || 'chat';
+
+         var xmlMsg = $msg({
+            to: jid,
+            type: type,
+            id: uid
+         }).c('body').t(msg);
+
+         if (type === 'chat' && (isBar || jsxc.xmpp.conn.caps.hasFeatureByJid(jid, Strophe.NS.RECEIPTS))) {
+            // Add request according to XEP-0184
+            xmlMsg.up().c('request', {
+               xmlns: 'urn:xmpp:receipts'
+            });
+         }
+
+         jsxc.xmpp.conn.send(xmlMsg);
+      },
+
+      /**
+       * This function loads a vcard.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param bid
+       * @param cb
+       * @param error_cb
+       */
+      loadVcard: function(bid, cb, error_cb) {
+         if (jsxc.master) {
+            jsxc.xmpp.conn.vcard.get(cb, bid, error_cb);
+         } else {
+            jsxc.storage.setUserItem('vcard', bid, 'request:' + (new Date()).getTime());
+
+            $(document).one('loaded.vcard.jsxc', function(ev, result) {
+               if (result && result.state === 'success') {
+                  cb($(result.data).get(0));
+               } else {
+                  error_cb();
+               }
+            });
+         }
+      },
+
+      /**
+       * Retrieves capabilities.
+       * 
+       * @memberOf jsxc.xmpp
+       * @param jid
+       * @returns List of known capabilities
+       */
+      getCapabilitiesByJid: function(jid) {
+         if (jsxc.xmpp.conn) {
+            return jsxc.xmpp.conn.caps.getCapabilitiesByJid(jid);
+         }
+
+         var jidVerIndex = JSON.parse(localStorage.getItem('strophe.caps._jidVerIndex')) || {};
+         var knownCapabilities = JSON.parse(localStorage.getItem('strophe.caps._knownCapabilities')) || {};
+
+         if (jidVerIndex[jid]) {
+            return knownCapabilities[jidVerIndex[jid]];
+         }
+
+         return null;
+      }
+   };
+
+   /**
+    * Handle long-live data
+    * 
+    * @namespace jsxc.storage
+    */
+   jsxc.storage = {
+      /**
+       * Prefix for localstorage
+       * 
+       * @privat
+       */
+      PREFIX: 'jsxc',
+
+      SEP: ':',
+
+      /**
+       * @param {type} uk Should we generate a user prefix?
+       * @returns {String} prefix
+       * @memberOf jsxc.storage
+       */
+      getPrefix: function(uk) {
+         var self = jsxc.storage;
+
+         return self.PREFIX + self.SEP + ((uk && jsxc.bid) ? jsxc.bid + self.SEP : '');
+      },
+
+      /**
+       * Save item to storage
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {Object} value value
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      setItem: function(key, value, uk) {
+
+         // Workaround for non-conform browser
+         if (jsxc.storageNotConform > 0 && key !== 'rid' && key !== 'lastActivity') {
+            if (jsxc.storageNotConform > 1 && jsxc.toSNC === null) {
+               jsxc.toSNC = window.setTimeout(function() {
+                  jsxc.storageNotConform = 0;
+                  jsxc.storage.setItem('storageNotConform', 0);
+               }, 1000);
+            }
+
+            jsxc.ls.push(JSON.stringify({
+               key: key,
+               value: value
+            }));
+         }
+
+         if (typeof (value) === 'object') {
+            value = JSON.stringify(value);
+         }
+
+         localStorage.setItem(jsxc.storage.getPrefix(uk) + key, value);
+      },
+
+      setUserItem: function(type, key, value) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 2) {
+            value = key;
+            key = type;
+            type = '';
+         } else if (arguments.length === 3) {
+            key = type + self.SEP + key;
+         }
+
+         return jsxc.storage.setItem(key, value, true);
+      },
+
+      /**
+       * Load item from storage
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      getItem: function(key, uk) {
+         key = jsxc.storage.getPrefix(uk) + key;
+
+         var value = localStorage.getItem(key);
+         try {
+            return JSON.parse(value);
+         } catch (e) {
+            return value;
+         }
+      },
+
+      /**
+       * Get a user item from storage.
+       * 
+       * @param key
+       * @returns user item
+       */
+      getUserItem: function(type, key) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 1) {
+            key = type;
+         } else if (arguments.length === 2) {
+            key = type + self.SEP + key;
+         }
+
+         return jsxc.storage.getItem(key, true);
+      },
+
+      /**
+       * Remove item from storage
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      removeItem: function(key, uk) {
+
+         // Workaround for non-conform browser
+         if (jsxc.storageNotConform && key !== 'rid' && key !== 'lastActivity') {
+            jsxc.ls.push(JSON.stringify({
+               key: jsxc.storage.prefix + key,
+               value: ''
+            }));
+         }
+
+         localStorage.removeItem(jsxc.storage.getPrefix(uk) + key);
+      },
+
+      /**
+       * Remove user item from storage.
+       * 
+       * @param key
+       */
+      removeUserItem: function(type, key) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 1) {
+            key = type;
+         } else if (arguments.length === 2) {
+            key = type + self.SEP + key;
+         }
+
+         jsxc.storage.removeItem(key, true);
+      },
+
+      /**
+       * Updates value of a variable in a saved object.
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String|object} variable variablename in object or object with
+       *        variable/key pairs
+       * @param {Object} [value] value
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      updateItem: function(key, variable, value, uk) {
+
+         var data = jsxc.storage.getItem(key, uk) || {};
+
+         if (typeof (variable) === 'object') {
+
+            $.each(variable, function(key, val) {
+               if (typeof (data[key]) === 'undefined') {
+                  jsxc.debug('Variable ' + key + ' doesn\'t exist in ' + variable + '. It was created.');
+               }
+
+               data[key] = val;
+            });
+         } else {
+            if (typeof (data[variable]) === 'undefined') {
+               jsxc.debug('Variable ' + variable + ' doesn\'t exist. It was created.');
+            }
+
+            data[variable] = value;
+         }
+
+         jsxc.storage.setItem(key, data, uk);
+      },
+
+      /**
+       * Updates value of a variable in a saved user object.
+       * 
+       * @param {String} key variablename
+       * @param {String|object} variable variablename in object or object with
+       *        variable/key pairs
+       * @param {Object} [value] value
+       */
+      updateUserItem: function(type, key, variable, value) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 4 || (arguments.length === 3 && typeof variable === 'object')) {
+            key = type + self.SEP + key;
+         } else {
+            value = variable;
+            variable = key;
+            key = type;
+         }
+
+         return jsxc.storage.updateItem(key, variable, value, true);
+      },
+
+      /**
+       * Inkrements value
+       * 
+       * @function
+       * @param {String} key variablename
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       */
+      ink: function(key, uk) {
+
+         jsxc.storage.setItem(key, Number(jsxc.storage.getItem(key, uk)) + 1, uk);
+      },
+
+      /**
+       * Remove element from array or object
+       * 
+       * @param {string} key name of array or object
+       * @param {string} name name of element in array or object
+       * @param {String} uk Userkey? Should we add the bid as prefix?
+       * @returns {undefined}
+       */
+      removeElement: function(key, name, uk) {
+         var item = jsxc.storage.getItem(key, uk);
+
+         if ($.isArray(item)) {
+            item = $.grep(item, function(e) {
+               return e !== name;
+            });
+         } else if (typeof (item) === 'object') {
+            delete item[name];
+         }
+
+         jsxc.storage.setItem(key, item, uk);
+      },
+
+      removeUserElement: function(type, key, name) {
+         var self = jsxc.storage;
+
+         if (arguments.length === 2) {
+            name = key;
+            key = type;
+         } else if (arguments.length === 3) {
+            key = type + self.SEP + key;
+         }
+
+         return jsxc.storage.removeElement(key, name, true);
+      },
+
+      /**
+       * Triggered if changes are recognized
+       * 
+       * @function
+       * @param {event} e Storageevent
+       * @param {String} e.key Keyname which triggered event
+       * @param {Object} e.oldValue Old Value for key
+       * @param {Object} e.newValue New Value for key
+       * @param {String} e.url
+       */
+      onStorage: function(e) {
+
+         // skip
+         if (e.key === jsxc.storage.PREFIX + jsxc.storage.SEP + 'rid' || e.key === jsxc.storage.PREFIX + jsxc.storage.SEP + 'lastActivity') {
+            return;
+         }
+
+         var re = new RegExp('^' + jsxc.storage.PREFIX + jsxc.storage.SEP + '(?:[^' + jsxc.storage.SEP + ']+@[^' + jsxc.storage.SEP + ']+' + jsxc.storage.SEP + ')?(.*)', 'i');
+         var key = e.key.replace(re, '$1');
+
+         // Workaround for non-conform browser: Triggered event on every page
+         // (own)
+         if (jsxc.storageNotConform > 0 && jsxc.ls.length > 0) {
+
+            var val = e.newValue;
+            try {
+               val = JSON.parse(val);
+            } catch (err) {
+            }
+
+            var index = $.inArray(JSON.stringify({
+               key: key,
+               value: val
+            }), jsxc.ls);
+
+            if (index >= 0) {
+
+               // confirm that the storage event is not fired regularly
+               if (jsxc.storageNotConform > 1) {
+                  window.clearTimeout(jsxc.toSNC);
+                  jsxc.storageNotConform = 1;
+                  jsxc.storage.setItem('storageNotConform', 1);
+               }
+
+               jsxc.ls.splice(index, 1);
+               return;
+            }
+         }
+
+         // Workaround for non-conform browser
+         if (e.oldValue === e.newValue) {
+            return;
+         }
+
+         var n, o;
+         var bid = key.replace(new RegExp('[^' + jsxc.storage.SEP + ']+' + jsxc.storage.SEP + '(.*)', 'i'), '$1');
+
+         // react if someone ask, if there is a master
+         if (jsxc.master && key === 'alive') {
+            jsxc.debug('Master request.');
+
+            jsxc.storage.ink('alive');
+            return;
+         }
+
+         // master alive
+         if (!jsxc.master && (key === 'alive' || key === 'alive_busy') && !jsxc.triggeredFromElement) {
+
+            // reset timeout
+            window.clearTimeout(jsxc.to);
+            jsxc.to = window.setTimeout(jsxc.checkMaster, ((key === 'alive') ? jsxc.options.timeout : jsxc.options.busyTimeout) + jsxc.random(60));
+
+            // only call the first time
+            if (!jsxc.role_allocation) {
+               jsxc.onSlave();
+            }
+
+            return;
+         }
+
+         if (key.match(/^notices/)) {
+            jsxc.notice.load();
+         }
+
+         if (key.match(/^presence/)) {
+            jsxc.gui.changePresence(e.newValue, true);
+         }
+
+         if (key.match(/^options/) && e.newValue) {
+            n = JSON.parse(e.newValue);
+
+            if (typeof n.muteNotification !== 'undefined' && n.muteNotification) {
+               jsxc.notification.muteSound(true);
+            } else {
+               jsxc.notification.unmuteSound(true);
+            }
+         }
+
+         if (key.match(/^hidden/)) {
+            if (jsxc.master) {
+               clearTimeout(jsxc.toNotification);
+            } else {
+               jsxc.isHidden();
+            }
+         }
+
+         if (key.match(new RegExp('^chat' + jsxc.storage.SEP))) {
+
+            var posts = JSON.parse(e.newValue);
+            var data, el;
+
+            while (posts.length > 0) {
+               data = posts.pop();
+               el = $('#' + data.uid);
+
+               if (el.length === 0) {
+                  if (jsxc.master && data.direction === 'out') {
+                     jsxc.xmpp.sendMessage(bid, data.msg, data.uid);
+                  }
+
+                  jsxc.gui.window._postMessage(bid, data);
+               } else if (data.received) {
+                  el.addClass('jsxc_received');
+               }
+            }
+            return;
+         }
+
+         if (key.match(new RegExp('^window' + jsxc.storage.SEP))) {
+
+            if (!e.newValue) {
+               jsxc.gui.window._close(bid);
+               return;
+            }
+
+            if (!e.oldValue) {
+               jsxc.gui.window.open(bid);
+               return;
+            }
+
+            n = JSON.parse(e.newValue);
+
+            if (n.minimize) {
+               jsxc.gui.window._hide(bid);
+            } else {
+               jsxc.gui.window._show(bid);
+            }
+
+            jsxc.gui.window.setText(bid, n.text);
+
+            return;
+         }
+
+         if (key.match(new RegExp('^smp' + jsxc.storage.SEP))) {
+
+            if (!e.newValue) {
+
+               jsxc.gui.dialog.close();
+
+               if (jsxc.master) {
+                  jsxc.otr.objects[bid].sm.abort();
+               }
+
+               return;
+            }
+
+            n = JSON.parse(e.newValue);
+
+            if (typeof (n.data) !== 'undefined') {
+
+               jsxc.otr.onSmpQuestion(bid, n.data);
+
+            } else if (jsxc.master && n.sec) {
+               jsxc.gui.dialog.close();
+
+               jsxc.otr.sendSmpReq(bid, n.sec, n.quest);
+            }
+         }
+
+         if (!jsxc.master && key.match(new RegExp('^buddy' + jsxc.storage.SEP))) {
+
+            if (!e.newValue) {
+               jsxc.gui.roster.purge(bid);
+               return;
+            }
+            if (!e.oldValue) {
+               jsxc.gui.roster.add(bid);
+               return;
+            }
+
+            n = JSON.parse(e.newValue);
+            o = JSON.parse(e.oldValue);
+
+            jsxc.gui.update(bid);
+
+            if (o.status !== n.status || o.sub !== n.sub) {
+               jsxc.gui.roster.reorder(bid);
+            }
+         }
+
+         if (jsxc.master && key.match(new RegExp('^deletebuddy' + jsxc.storage.SEP)) && e.newValue) {
+            n = JSON.parse(e.newValue);
+
+            jsxc.xmpp.removeBuddy(n.jid);
+            jsxc.storage.removeUserItem(key);
+         }
+
+         if (jsxc.master && key.match(new RegExp('^buddy' + jsxc.storage.SEP))) {
+
+            n = JSON.parse(e.newValue);
+            o = JSON.parse(e.oldValue);
+
+            if (o.transferReq !== n.transferReq) {
+               jsxc.storage.updateUserItem('buddy', bid, 'transferReq', -1);
+
+               if (n.transferReq === 0) {
+                  jsxc.otr.goPlain(bid);
+               }
+               if (n.transferReq === 1) {
+                  jsxc.otr.goEncrypt(bid);
+               }
+            }
+
+            if (o.name !== n.name) {
+               jsxc.gui.roster._rename(bid, n.name);
+            }
+         }
+
+         // logout
+         if (key === 'sid') {
+            if (!e.newValue) {
+               // if (jsxc.master && jsxc.xmpp.conn) {
+               // jsxc.xmpp.conn.disconnect();
+               // jsxc.triggeredFromElement = true;
+               // }
+               jsxc.xmpp.logout();
+
+            }
+            return;
+         }
+
+         if (key === 'friendReq') {
+            n = JSON.parse(e.newValue);
+
+            if (jsxc.master && n.approve >= 0) {
+               jsxc.xmpp.resFriendReq(n.jid, n.approve);
+            }
+         }
+
+         if (jsxc.master && key.match(new RegExp('^add' + jsxc.storage.SEP))) {
+            n = JSON.parse(e.newValue);
+
+            jsxc.xmpp.addBuddy(n.username, n.alias);
+         }
+
+         if (key === 'roster') {
+            jsxc.gui.roster.toggle();
+         }
+
+         if (jsxc.master && key.match(new RegExp('^vcard' + jsxc.storage.SEP)) && e.newValue !== null && e.newValue.match(/^request:/)) {
+
+            jsxc.xmpp.loadVcard(bid, function(stanza) {
+               jsxc.storage.setUserItem('vcard', bid, {
+                  state: 'success',
+                  data: $('<div>').append(stanza).html()
+               });
+            }, function() {
+               jsxc.storage.setUserItem('vcard', bid, {
+                  state: 'error'
+               });
+            });
+         }
+
+         if (!jsxc.master && key.match(new RegExp('^vcard' + jsxc.storage.SEP)) && e.newValue !== null && !e.newValue.match(/^request:/)) {
+            n = JSON.parse(e.newValue);
+
+            if (typeof n.state !== 'undefined') {
+               $(document).trigger('loaded.vcard.jsxc', n);
+            }
+
+            jsxc.storage.removeUserItem('vcard', bid);
+         }
+      },
+
+      /**
+       * Save message to storage.
+       * 
+       * @memberOf jsxc.storage
+       * @param bid
+       * @param direction
+       * @param msg
+       * @return post
+       */
+      saveMessage: function(bid, direction, msg) {
+         var chat = jsxc.storage.getUserItem('chat', bid) || [];
+
+         var uid = new Date().getTime() + ':msg';
+
+         if (chat.length > jsxc.options.get('numberOfMsg')) {
+            chat.pop();
+         }
+
+         var post = {
+            direction: direction,
+            msg: msg,
+            uid: uid.replace(/:/, '-'),
+            received: false
+         };
+
+         chat.unshift(post);
+         jsxc.storage.setUserItem('chat', bid, chat);
+
+         return post;
+      },
+
+      /**
+       * Save or update buddy data.
+       * 
+       * @memberOf jsxc.storage
+       * @param bid
+       * @param data
+       * @returns {String} Updated or created
+       */
+      saveBuddy: function(bid, data) {
+
+         if (jsxc.storage.getUserItem('buddy', bid)) {
+            jsxc.storage.updateUserItem('buddy', bid, data);
+
+            return 'updated';
+         }
+
+         jsxc.storage.setUserItem('buddy', bid, $.extend({
+            jid: '',
+            name: '',
+            status: 0,
+            sub: 'none',
+            msgstate: 0,
+            transferReq: -1,
+            trust: false,
+            fingerprint: null,
+            res: [],
+            type: 'chat'
+         }, data));
+
+         return 'created';
+      }
+   };
+
+   /**
+    * @namespace jsxc.otr
+    */
+   jsxc.otr = {
+      /** list of otr objects */
+      objects: {},
+
+      dsaFallback: null,
+      /**
+       * Handler for otr receive event
+       * 
+       * @memberOf jsxc.otr
+       * @param {string} bid
+       * @param {string} msg received message
+       * @param {string} encrypted True, if msg was encrypted.
+       */
+      receiveMessage: function(bid, msg, encrypted) {
+
+         if (jsxc.otr.objects[bid].msgstate !== OTR.CONST.MSGSTATE_PLAINTEXT) {
+            jsxc.otr.backup(bid);
+         }
+
+         if (jsxc.otr.objects[bid].msgstate !== OTR.CONST.MSGSTATE_PLAINTEXT && !encrypted) {
+            jsxc.gui.window.postMessage(bid, 'sys', jsxc.translate('%%Received an unencrypted message.%% [') + msg + ']');
+         } else {
+            jsxc.gui.window.postMessage(bid, 'in', msg);
+         }
+      },
+
+      /**
+       * Handler for otr send event
+       * 
+       * @param {string} jid
+       * @param {string} msg message to be send
+       */
+      sendMessage: function(jid, msg, uid) {
+         if (jsxc.otr.objects[jsxc.jidToBid(jid)].msgstate !== 0) {
+            jsxc.otr.backup(jsxc.jidToBid(jid));
+         }
+
+         jsxc.xmpp._sendMessage(jid, msg, uid);
+      },
+
+      /**
+       * Create new otr instance
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      create: function(bid) {
+
+         if (jsxc.otr.objects.hasOwnProperty(bid)) {
+            return;
+         }
+
+         if (!jsxc.options.otr.priv) {
+            return;
+         }
+
+         // save list of otr objects
+         var ol = jsxc.storage.getUserItem('otrlist') || [];
+         if (ol.indexOf(bid) < 0) {
+            ol.push(bid);
+            jsxc.storage.setUserItem('otrlist', ol);
+         }
+
+         jsxc.otr.objects[bid] = new OTR(jsxc.options.otr);
+
+         if (jsxc.options.otr.SEND_WHITESPACE_TAG) {
+            jsxc.otr.objects[bid].SEND_WHITESPACE_TAG = true;
+         }
+
+         if (jsxc.options.otr.WHITESPACE_START_AKE) {
+            jsxc.otr.objects[bid].WHITESPACE_START_AKE = true;
+         }
+
+         jsxc.otr.objects[bid].on('status', function(status) {
+            var data = jsxc.storage.getUserItem('buddy', bid);
+
+            if (data === null) {
+               return;
+            }
+
+            switch (status) {
+               case OTR.CONST.STATUS_SEND_QUERY:
+                  jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.trying_to_start_private_conversation);
+                  break;
+               case OTR.CONST.STATUS_AKE_SUCCESS:
+                  data.fingerprint = jsxc.otr.objects[bid].their_priv_pk.fingerprint();
+                  data.msgstate = OTR.CONST.MSGSTATE_ENCRYPTED;
+
+                  var msg = (jsxc.otr.objects[bid].trust ? jsxc.l.Verified : jsxc.l.Unverified) + ' ' + jsxc.l.private_conversation_started;
+                  jsxc.gui.window.postMessage(bid, 'sys', msg);
+                  break;
+               case OTR.CONST.STATUS_END_OTR:
+                  data.fingerprint = null;
+
+                  if (jsxc.otr.objects[bid].msgstate === OTR.CONST.MSGSTATE_PLAINTEXT) {
+                     // we abort the private conversation
+
+                     data.msgstate = OTR.CONST.MSGSTATE_PLAINTEXT;
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.private_conversation_aborted);
+
+                  } else {
+                     // the buddy abort the private conversation
+
+                     data.msgstate = OTR.CONST.MSGSTATE_FINISHED;
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.your_buddy_closed_the_private_conversation_you_should_do_the_same);
+                  }
+                  break;
+               case OTR.CONST.STATUS_SMP_HANDLE:
+                  jsxc.keepBusyAlive();
+                  break;
+            }
+
+            jsxc.storage.setUserItem('buddy', bid, data);
+
+            // for encryption and verification state
+            jsxc.gui.update(bid);
+         });
+
+         jsxc.otr.objects[bid].on('smp', function(type, data) {
+            switch (type) {
+               case 'question': // verification request received
+                  jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.Authentication_request_received);
+
+                  if ($('#jsxc_dialog').length > 0) {
+                     jsxc.otr.objects[bid].sm.abort();
+                     break;
+                  }
+
+                  jsxc.otr.onSmpQuestion(bid, data);
+                  jsxc.storage.setUserItem('smp_' + bid, {
+                     data: data || null
+                  });
+
+                  break;
+               case 'trust': // verification completed
+                  jsxc.otr.objects[bid].trust = data;
+                  jsxc.storage.updateUserItem('buddy', bid, 'trust', data);
+                  jsxc.otr.backup(bid);
+                  jsxc.gui.update(bid);
+
+                  if (data) {
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.conversation_is_now_verified);
+                  } else {
+                     jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.authentication_failed);
+                  }
+                  jsxc.storage.removeUserItem('smp_' + bid);
+                  jsxc.gui.dialog.close();
+                  break;
+               case 'abort':
+                  jsxc.gui.window.postMessage(bid, 'sys', jsxc.l.Authentication_aborted);
+                  break;
+               default:
+                  jsxc.debug('[OTR] sm callback: Unknown type: ' + type);
+            }
+         });
+
+         // Receive message
+         jsxc.otr.objects[bid].on('ui', function(msg, encrypted) {
+            jsxc.otr.receiveMessage(bid, msg, encrypted === true);
+         });
+
+         // Send message
+         jsxc.otr.objects[bid].on('io', function(msg, uid) {
+            var jid = jsxc.gui.window.get(bid).data('jid') || jsxc.otr.objects[bid].jid;
+
+            jsxc.otr.objects[bid].jid = jid;
+
+            jsxc.otr.sendMessage(jid, msg, uid);
+         });
+
+         jsxc.otr.objects[bid].on('error', function(err) {
+            // Handle this case in jsxc.otr.receiveMessage
+            if (err !== 'Received an unencrypted message.') {
+               jsxc.gui.window.postMessage(bid, 'sys', '[OTR] ' + jsxc.translate('%%' + err + '%%'));
+            }
+
+            jsxc.error('[OTR] ' + err);
+         });
+
+         jsxc.otr.restore(bid);
+      },
+
+      /**
+       * show verification dialog with related part (secret or question)
+       * 
+       * @param {type} bid
+       * @param {string} [data]
+       * @returns {undefined}
+       */
+      onSmpQuestion: function(bid, data) {
+         jsxc.gui.showVerification(bid);
+
+         $('#jsxc_dialog select').prop('selectedIndex', (data ? 2 : 3)).change();
+         $('#jsxc_dialog > div:eq(0)').hide();
+
+         if (data) {
+            $('#jsxc_dialog > div:eq(2)').find('#jsxc_quest').val(data).prop('disabled', true);
+            $('#jsxc_dialog > div:eq(2)').find('.creation').text('Answer');
+            $('#jsxc_dialog > div:eq(2)').find('.jsxc_explanation').text(jsxc.l.your_buddy_is_attempting_to_determine_ + ' ' + jsxc.l.to_authenticate_to_your_buddy + jsxc.l.enter_the_answer_and_click_answer);
+         } else {
+            $('#jsxc_dialog > div:eq(3)').find('.jsxc_explanation').text(jsxc.l.your_buddy_is_attempting_to_determine_ + ' ' + jsxc.l.to_authenticate_to_your_buddy + jsxc.l.enter_the_secret);
+         }
+
+         $('#jsxc_dialog .jsxc_close').click(function() {
+            jsxc.storage.removeUserItem('smp_' + bid);
+
+            if (jsxc.master) {
+               jsxc.otr.objects[bid].sm.abort();
+            }
+         });
+      },
+
+      /**
+       * Send verification request to buddy
+       * 
+       * @param {string} bid
+       * @param {string} sec secret
+       * @param {string} [quest] question
+       * @returns {undefined}
+       */
+      sendSmpReq: function(bid, sec, quest) {
+         jsxc.keepBusyAlive();
+
+         jsxc.otr.objects[bid].smpSecret(sec, quest || '');
+      },
+
+      /**
+       * Toggle encryption state
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      toggleTransfer: function(bid) {
+         if (jsxc.storage.getUserItem('buddy', bid).msgstate === 0) {
+            jsxc.otr.goEncrypt(bid);
+         } else {
+            jsxc.otr.goPlain(bid);
+         }
+      },
+
+      /**
+       * Send request to encrypt the session
+       * 
+       * @param {type} bid
+       * @returns {undefined}
+       */
+      goEncrypt: function(bid) {
+         if (jsxc.master) {
+            jsxc.otr.objects[bid].sendQueryMsg();
+         } else {
+            jsxc.storage.updateUserItem('buddy', bid, 'transferReq', 1);
+         }
+      },
+
+      /**
+       * Abort encryptet session
+       * 
+       * @param {type} bid
+       * @param cb callback
+       * @returns {undefined}
+       */
+      goPlain: function(bid, cb) {
+         if (jsxc.master) {
+            jsxc.otr.objects[bid].endOtr.call(jsxc.otr.objects[bid], cb);
+            jsxc.otr.objects[bid].init.call(jsxc.otr.objects[bid]);
+
+            jsxc.otr.backup(bid);
+         } else {
+            jsxc.storage.updateUserItem('buddy', bid, 'transferReq', 0);
+         }
+      },
+
+      /**
+       * Backups otr session
+       * 
+       * @param {string} bid
+       */
+      backup: function(bid) {
+         var o = jsxc.otr.objects[bid]; // otr object
+         var r = {}; // return value
+
+         if (o === null) {
+            return;
+         }
+
+         // all variables which should be saved
+         var savekey = [ 'jid', 'our_instance_tag', 'msgstate', 'authstate', 'fragment', 'their_y', 'their_old_y', 'their_keyid', 'their_instance_tag', 'our_dh', 'our_old_dh', 'our_keyid', 'sessKeys', 'storedMgs', 'oldMacKeys', 'trust', 'transmittedRS', 'ssid', 'receivedPlaintext', 'authstate', 'send_interval' ];
+
+         var i;
+         for (i = 0; i < savekey.length; i++) {
+            r[savekey[i]] = JSON.stringify(o[savekey[i]]);
+         }
+
+         if (o.their_priv_pk !== null) {
+            r.their_priv_pk = JSON.stringify(o.their_priv_pk.packPublic());
+         }
+
+         if (o.ake.otr_version && o.ake.otr_version !== '') {
+            r.otr_version = JSON.stringify(o.ake.otr_version);
+         }
+
+         jsxc.storage.setUserItem('otr', bid, r);
+      },
+
+      /**
+       * Restore old otr session
+       * 
+       * @param {string} bid
+       */
+      restore: function(bid) {
+         var o = jsxc.otr.objects[bid];
+         var d = jsxc.storage.getUserItem('otr', bid);
+
+         if (o !== null || d !== null) {
+            var key;
+            for (key in d) {
+               if (d.hasOwnProperty(key)) {
+                  var val = JSON.parse(d[key]);
+                  if (key === 'their_priv_pk' && val !== null) {
+                     val = DSA.parsePublic(val);
+                  }
+                  if (key === 'otr_version' && val !== null) {
+                     o.ake.otr_version = val;
+                  } else {
+                     o[key] = val;
+                  }
+               }
+            }
+
+            jsxc.otr.objects[bid] = o;
+
+            if (o.msgstate === 1 && o.their_priv_pk !== null) {
+               o._smInit.call(jsxc.otr.objects[bid]);
+            }
+         }
+
+         jsxc.otr.enable(bid);
+      },
+
+      /**
+       * Create or load DSA key
+       * 
+       * @returns {unresolved}
+       */
+      createDSA: function() {
+         if (jsxc.options.otr.priv) {
+            return;
+         }
+
+         if (jsxc.storage.getUserItem('key') === null) {
+            var msg = jsxc.l.Creating_your_private_key_;
+            var worker = null;
+
+            if (Worker) {
+               // try to create web-worker
+
+               try {
+                  worker = new Worker(jsxc.options.root + '/lib/otr/build/dsa-webworker.js');
+               } catch (err) {
+                  jsxc.warn('Couldn\'t create web-worker.', err);
+               }
+            }
+
+            jsxc.otr.dsaFallback = (worker === null);
+
+            if (!jsxc.otr.dsaFallback) {
+               // create DSA key in background
+
+               jsxc._onMaster();
+
+               worker.onmessage = function(e) {
+                  var type = e.data.type;
+                  var val = e.data.val;
+
+                  if (type === 'debug') {
+                     jsxc.debug(val);
+                  } else if (type === 'data') {
+                     jsxc.otr.DSAready(DSA.parsePrivate(val));
+                  }
+               };
+
+               // start worker
+               worker.postMessage({
+                  imports: [ jsxc.options.root + '/lib/otr/vendor/salsa20.js', jsxc.options.root + '/lib/otr/vendor/bigint.js', jsxc.options.root + '/lib/otr/vendor/crypto.js', jsxc.options.root + '/lib/otr/vendor/eventemitter.js', jsxc.options.root + '/lib/otr/lib/const.js', jsxc.options.root + '/lib/otr/lib/helpers.js', jsxc.options.root + '/lib/otr/lib/dsa.js' ],
+                  seed: BigInt.getSeed(),
+                  debug: true
+               });
+
+            } else {
+               // fallback
+
+               jsxc.gui.dialog.open(jsxc.gui.template.get('waitAlert', null, msg), {
+                  noClose: true
+               });
+
+               jsxc.debug('DSA key creation started.');
+
+               // wait until the wait alert is opened
+               setTimeout(function() {
+                  var dsa = new DSA();
+                  jsxc.otr.DSAready(dsa);
+               }, 500);
+            }
+         } else {
+            jsxc.debug('DSA key loaded');
+            jsxc.options.otr.priv = DSA.parsePrivate(jsxc.storage.getUserItem('key'));
+
+            jsxc.otr._createDSA();
+         }
+      },
+
+      /**
+       * Ending of createDSA().
+       */
+      _createDSA: function() {
+
+         jsxc.storage.setUserItem('priv_fingerprint', jsxc.options.otr.priv.fingerprint());
+
+         if (jsxc.otr.dsaFallback !== false) {
+            jsxc._onMaster();
+         }
+      },
+
+      /**
+       * Ending of DSA key generation.
+       * 
+       * @param {DSA} dsa DSA object
+       */
+      DSAready: function(dsa) {
+         jsxc.storage.setUserItem('key', dsa.packPrivate());
+         jsxc.options.otr.priv = dsa;
+
+         // close wait alert
+         if (jsxc.otr.dsaFallback) {
+            jsxc.gui.dialog.close();
+         } else {
+            $.each(jsxc.storage.getUserItem('windowlist'), function(index, val) {
+               jsxc.otr.create(val);
+            });
+         }
+
+         jsxc.otr._createDSA();
+      },
+
+      enable: function(bid) {
+         jsxc.gui.window.get(bid).find('.jsxc_otr').removeClass('jsxc_disabled');
+      }
+   };
+
+   /**
+    * This namespace handles the Notification API.
+    * 
+    * @namespace jsxc.notification
+    */
+   jsxc.notification = {
+
+      /** Current audio file. */
+      audio: null,
+
+      /**
+       * Register notification on incoming messages.
+       * 
+       * @memberOf jsxc.notification
+       */
+      init: function() {
+         $(document).on('postmessagein.jsxc', function(event, bid, msg) {
+            msg = (msg.match(/^\?OTR/)) ? jsxc.translate('%%Encrypted message%%') : msg;
+            var data = jsxc.storage.getUserItem('buddy', bid);
+
+            jsxc.notification.notify(jsxc.translate('%%New message from%% ') + data.name, msg, undefined, undefined, jsxc.CONST.SOUNDS.MSG);
+         });
+
+         $(document).on('callincoming.jingle', function() {
+            jsxc.notification.playSound(jsxc.CONST.SOUNDS.CALL, true, true);
+         });
+
+         $(document).on('accept.call.jsxc reject.call.jsxc', function() {
+            jsxc.notification.stopSound();
+         });
+      },
+
+      /**
+       * Shows a pop up notification and optional play sound.
+       * 
+       * @param title Title
+       * @param msg Message
+       * @param d Duration
+       * @param force Should message also shown, if tab is visible?
+       * @param soundFile Playing given sound file
+       * @param loop Loop sound file?
+       */
+      notify: function(title, msg, d, force, soundFile, loop) {
+         if (!jsxc.options.notification || !jsxc.notification.hasPermission()) {
+            return; // notifications disabled
+         }
+
+         if (!jsxc.isHidden() && !force) {
+            return; // Tab is visible
+         }
+
+         jsxc.toNotification = setTimeout(function() {
+
+            if (typeof soundFile === 'string') {
+               jsxc.notification.playSound(soundFile, loop, force);
+            }
+
+            var popup = new Notification(jsxc.translate(title), {
+               body: jsxc.translate(msg),
+               icon: jsxc.options.root + '/img/XMPP_logo.png'
+            });
+
+            var duration = d || jsxc.options.popupDuration;
+
+            if (duration > 0) {
+               setTimeout(function() {
+                  popup.close();
+               }, duration);
+            }
+         }, jsxc.toNotificationDelay);
+      },
+
+      /**
+       * Checks if browser has support for notifications and add on chrome to
+       * the default api.
+       * 
+       * @returns {Boolean} True if the browser has support.
+       */
+      hasSupport: function() {
+         if (window.webkitNotifications) {
+            // prepare chrome
+
+            window.Notification = function(title, opt) {
+               var popup = window.webkitNotifications.createNotification(null, title, opt.body);
+               popup.show();
+
+               popup.close = function() {
+                  popup.cancel();
+               };
+
+               return popup;
+            };
+
+            var permission;
+            switch (window.webkitNotifications.checkPermission()) {
+               case 0:
+                  permission = jsxc.CONST.NOTIFICATION_GRANTED;
+                  break;
+               case 2:
+                  permission = jsxc.CONST.NOTIFICATION_DENIED;
+                  break;
+               default: // 1
+                  permission = jsxc.CONST.NOTIFICATION_DEFAULT;
+            }
+            window.Notification.permission = permission;
+
+            window.Notification.requestPermission = function(func) {
+               window.webkitNotifications.requestPermission(func);
+            };
+
+            return true;
+         } else if (window.Notification) {
+            return true;
+         } else {
+            return false;
+         }
+      },
+
+      /**
+       * Ask user on first incoming message if we should inform him about new
+       * messages.
+       */
+      prepareRequest: function() {
+
+         $(document).one('postmessagein.jsxc', function() {
+            jsxc.switchEvents({
+               'notificationready.jsxc': function() {
+                  jsxc.gui.dialog.close();
+                  jsxc.notification.init();
+                  jsxc.storage.setUserItem('notification', true);
+               },
+               'notificationfailure.jsxc': function() {
+                  jsxc.gui.dialog.close();
+                  jsxc.options.notification = false;
+                  jsxc.storage.setUserItem('notification', false);
+               }
+            });
+
+            setTimeout(function() {
+               jsxc.notice.add('%%Notifications%%?', '%%Should_we_notify_you_%%', 'gui.showRequestNotification');
+            }, 1000);
+         });
+      },
+
+      /**
+       * Request notification permission.
+       */
+      requestPermission: function() {
+         window.Notification.requestPermission(function(status) {
+            if (window.Notification.permission !== status) {
+               window.Notification.permission = status;
+            }
+
+            if (jsxc.notification.hasPermission()) {
+               $(document).trigger('notificationready.jsxc');
+            } else {
+               $(document).trigger('notificationfailure.jsxc');
+            }
+         });
+      },
+
+      /**
+       * Check permission.
+       * 
+       * @returns {Boolean} True if we have the permission
+       */
+      hasPermission: function() {
+         return window.Notification.permission === jsxc.CONST.NOTIFICATION_GRANTED;
+      },
+
+      /**
+       * Plays the given file.
+       * 
+       * @memberOf jsxc.notification
+       * @param {string} soundFile File relative to the sound directory
+       * @param {boolean} loop True for loop
+       * @param {boolean} force Play even if a tab is visible. Default: false.
+       */
+      playSound: function(soundFile, loop, force) {
+         if (!jsxc.master) {
+            // only master plays sound
+            return;
+         }
+
+         if (jsxc.options.get('muteNotification') || jsxc.storage.getUserItem('presence') === 'dnd') {
+            // sound mute or own presence is dnd
+            return;
+         }
+
+         if (!jsxc.isHidden() && !force) {
+            // tab is visible
+            return;
+         }
+
+         // stop current audio file
+         jsxc.notification.stopSound();
+
+         var audio = new Audio(jsxc.options.root + '/sound/' + soundFile);
+         audio.loop = loop || false;
+         audio.play();
+
+         jsxc.notification.audio = audio;
+      },
+
+      /**
+       * Stop/remove current sound.
+       * 
+       * @memberOf jsxc.notification
+       */
+      stopSound: function() {
+         var audio = jsxc.notification.audio;
+
+         if (typeof audio !== 'undefined' && audio !== null) {
+            audio.pause();
+            jsxc.notification.audio = null;
+         }
+      },
+
+      /**
+       * Mute sound.
+       * 
+       * @memberOf jsxc.notification
+       * @param {boolean} external True if triggered from external tab. Default:
+       *        false.
+       */
+      muteSound: function(external) {
+         $('#jsxc_menu .jsxc_muteNotification').text(jsxc.translate('%%Unmute%%'));
+
+         if (external !== true) {
+            jsxc.options.set('muteNotification', true);
+         }
+      },
+
+      /**
+       * Unmute sound.
+       * 
+       * @memberOf jsxc.notification
+       * @param {boolean} external True if triggered from external tab. Default:
+       *        false.
+       */
+      unmuteSound: function(external) {
+         $('#jsxc_menu .jsxc_muteNotification').text(jsxc.translate('%%Mute%%'));
+
+         if (external !== true) {
+            jsxc.options.set('muteNotification', false);
+         }
+      }
+   };
+
+   /**
+    * This namespace handle the notice system.
+    * 
+    * @namspace jsxc.notice
+    * @memberOf jsxc
+    */
+   jsxc.notice = {
+      /** Number of notices. */
+      _num: 0,
+
+      /**
+       * Loads the saved notices.
+       * 
+       * @memberOf jsxc.notice
+       */
+      load: function() {
+         // reset list
+         $('#jsxc_notice ul li').remove();
+         $('#jsxc_notice > span').text('');
+         jsxc.notice._num = 0;
+
+         var saved = jsxc.storage.getUserItem('notices') || [];
+         var key = null;
+
+         for (key in saved) {
+            if (saved.hasOwnProperty(key)) {
+               var val = saved[key];
+
+               jsxc.notice.add(val.msg, val.description, val.fnName, val.fnParams, key);
+            }
+         }
+      },
+
+      /**
+       * Add a new notice to the stack;
+       * 
+       * @memberOf jsxc.notice
+       * @param msg Header message
+       * @param description Notice description
+       * @param fnName Function name to be called if you open the notice
+       * @param fnParams Array of params for function
+       * @param id Notice id
+       */
+      add: function(msg, description, fnName, fnParams, id) {
+         var nid = id || Date.now();
+         var list = $('#jsxc_notice ul');
+         var notice = $('<li/>');
+
+         notice.click(function() {
+            jsxc.notice.remove(nid);
+
+            var fnList = fnName.split('.');
+            var fn = jsxc[fnList[0]];
+            var i;
+            for (i = 1; i < fnList.length; i++) {
+               fn = fn[fnList[i]];
+            }
+
+            if (typeof fn === 'function') {
+               fn.apply(null, fnParams);
+            }
+
+            return false;
+         });
+
+         notice.text(jsxc.translate(msg));
+         notice.attr('title', jsxc.translate(description) || '');
+         notice.attr('data-nid', nid);
+         list.append(notice);
+
+         $('#jsxc_notice > span').text(++jsxc.notice._num);
+
+         if (!id) {
+            var saved = jsxc.storage.getUserItem('notices') || {};
+            saved[nid] = {
+               msg: msg,
+               description: description,
+               fnName: fnName,
+               fnParams: fnParams
+            };
+            jsxc.storage.setUserItem('notices', saved);
+
+            jsxc.notification.notify(msg, description || '', null, true, jsxc.CONST.SOUNDS.NOTICE);
+         }
+      },
+
+      /**
+       * Removes notice from stack
+       * 
+       * @memberOf jsxc.notice
+       * @param nid The notice id
+       */
+      remove: function(nid) {
+         var el = $('#jsxc_notice li[data-nid=' + nid + ']');
+
+         el.remove();
+         $('#jsxc_notice > span').text(--jsxc.notice._num || '');
+
+         var s = jsxc.storage.getUserItem('notices');
+         delete s[nid];
+         jsxc.storage.setUserItem('notices', s);
+      }
+   };
+
+   /**
+    * Contains all available translations
+    * 
+    * @namespace jsxc.l10n
+    * @memberOf jsxc
+    */
+   jsxc.l10n = {
+      en: {
+         Logging_in: 'Logging in…',
+         your_connection_is_unencrypted: 'Your connection is unencrypted.',
+         your_connection_is_encrypted: 'Your connection is encrypted.',
+         your_buddy_closed_the_private_connection: 'Your buddy closed the private connection.',
+         start_private: 'Start private',
+         close_private: 'Close private',
+         your_buddy_is_verificated: 'Your buddy is verified.',
+         you_have_only_a_subscription_in_one_way: 'You only have a one-way subscription.',
+         authentication_query_sent: 'Authentication query sent.',
+         your_message_wasnt_send_please_end_your_private_conversation: 'Your message was not sent. Please end your private conversation.',
+         unencrypted_message_received: 'Unencrypted message received:',
+         your_message_wasnt_send_because_you_have_no_valid_subscription: 'Your message was not sent because you have no valid subscription.',
+         not_available: 'Not available',
+         no_connection: 'No connection!',
+         relogin: 'relogin',
+         trying_to_start_private_conversation: 'Trying to start private conversation!',
+         Verified: 'Verified',
+         Unverified: 'Unverified',
+         private_conversation_started: 'Private conversation started.',
+         private_conversation_aborted: 'Private conversation aborted!',
+         your_buddy_closed_the_private_conversation_you_should_do_the_same: 'Your buddy closed the private conversation! You should do the same.',
+         conversation_is_now_verified: 'Conversation is now verified.',
+         authentication_failed: 'Authentication failed.',
+         your_buddy_is_attempting_to_determine_: 'You buddy is attempting to determine if he or she is really talking to you.',
+         to_authenticate_to_your_buddy: 'To authenticate to your buddy, ',
+         enter_the_answer_and_click_answer: 'enter the answer and click Answer.',
+         enter_the_secret: 'enter the secret.',
+         Creating_your_private_key_: 'Creating your private key; this may take a while.',
+         Authenticating_a_buddy_helps_: 'Authenticating a buddy helps ensure that the person you are talking to is really the one he or she claims to be.',
+         How_do_you_want_to_authenticate_your_buddy: 'How do you want to authenticate {{bid_name}} (<b>{{bid_jid}}</b>)?',
+         Select_method: 'Select method...',
+         Manual: 'Manual',
+         Question: 'Question',
+         Secret: 'Secret',
+         To_verify_the_fingerprint_: 'To verify the fingerprint, contact your buddy via some other trustworthy channel, such as the telephone.',
+         Your_fingerprint: 'Your fingerprint',
+         Buddy_fingerprint: 'Buddy fingerprint',
+         Close: 'Close',
+         Compared: 'Compared',
+         To_authenticate_using_a_question_: 'To authenticate using a question, pick a question whose answer is known only you and your buddy.',
+         Ask: 'Ask',
+         To_authenticate_pick_a_secret_: 'To authenticate, pick a secret known only to you and your buddy.',
+         Compare: 'Compare',
+         Fingerprints: 'Fingerprints',
+         Authentication: 'Authentication',
+         Message: 'Message',
+         Add_buddy: 'Add buddy',
+         rename_buddy: 'rename buddy',
+         delete_buddy: 'delete buddy',
+         Login: 'Login',
+         Username: 'Username',
+         Password: 'Password',
+         Cancel: 'Cancel',
+         Connect: 'Connect',
+         Type_in_the_full_username_: 'Type in the full username and an optional alias.',
+         Alias: 'Alias',
+         Add: 'Add',
+         Subscription_request: 'Subscription request',
+         You_have_a_request_from: 'You have a request from',
+         Deny: 'Deny',
+         Approve: 'Approve',
+         Remove_buddy: 'Remove buddy',
+         You_are_about_to_remove_: 'You are about to remove {{bid_name}} (<b>{{bid_jid}}</b>) from your buddy list. All related chats will be closed.',
+         Continue_without_chat: 'Continue without chat',
+         Please_wait: 'Please wait',
+         Login_failed: 'Chat login failed',
+         Sorry_we_cant_authentikate_: 'Authentication failed with the chat server. Maybe the password is wrong?',
+         Retry: 'Back',
+         clear_history: 'Clear history',
+         New_message_from: 'New message from',
+         Should_we_notify_you_: 'Should we notify you about new messages in the future?',
+         Please_accept_: 'Please click the "Allow" button at the top.',
+         Hide_offline: 'Hide offline contacts',
+         Show_offline: 'Show offline contacts',
+         About: 'About',
+         dnd: 'Do Not Disturb',
+         Mute: 'Mute',
+         Unmute: 'Unmute',
+         Subscription: 'Subscription',
+         both: 'both',
+         Status: 'Status',
+         online: 'online',
+         chat: 'chat',
+         away: 'away',
+         xa: 'extended away',
+         offline: 'offline',
+         none: 'none',
+         Unknown_instance_tag: 'Unknown instance tag.',
+         Not_one_of_our_latest_keys: 'Not one of our latest keys.',
+         Received_an_unreadable_encrypted_message: 'Received an unreadable encrypted message.',
+         Online: 'Online',
+         Chatty: 'Chatty',
+         Away: 'Away',
+         Extended_away: 'Extended away',
+         Offline: 'Offline',
+         Friendship_request: 'Friendship request',
+         Confirm: 'Confirm',
+         Dismiss: 'Dismiss',
+         Remove: 'Remove',
+         Online_help: 'Online help',
+         FN: 'Full name',
+         N: ' ',
+         FAMILY: 'Family name',
+         GIVEN: 'Given name',
+         NICKNAME: 'Nickname',
+         URL: 'URL',
+         ADR: 'Address',
+         STREET: 'Street Address',
+         EXTADD: 'Extended Address',
+         LOCALITY: 'Locality',
+         REGION: 'Region',
+         PCODE: 'Postal Code',
+         CTRY: 'Country',
+         TEL: 'Telephone',
+         NUMBER: 'Number',
+         EMAIL: 'Email',
+         USERID: ' ',
+         ORG: 'Organization',
+         ORGNAME: 'Name',
+         ORGUNIT: 'Unit',
+         TITLE: 'Job title',
+         ROLE: 'Role',
+         BDAY: 'Birthday',
+         DESC: 'Description',
+         PHOTO: ' ',
+         send_message: 'Send message',
+         get_info: 'Show information',
+         Settings: 'Settings',
+         Priority: 'Priority',
+         Save: 'Save',
+         User_settings: 'User settings',
+         A_fingerprint_: 'A fingerprint is used to make sure that the person you are talking to is who he or she is saying.',
+         Your_roster_is_empty_add_a: 'Your roster is empty, add a ',
+         new_buddy: 'new buddy',
+         is: 'is',
+         Login_options: 'Login options',
+         BOSH_url: 'BOSH URL',
+         Domain: 'Domain',
+         Resource: 'Resource',
+         On_login: 'On login',
+         Received_an_unencrypted_message: 'Received an unencrypted message',
+         Sorry_your_buddy_doesnt_provide_any_information: 'Sorry, your buddy does not provide any information.',
+         Info_about: 'Info about',
+         Authentication_aborted: 'Authentication aborted.',
+         Authentication_request_received: 'Authentication request received.',
+         Do_you_want_to_display_them: 'Do you want to display them?',
+         Log_in_without_chat: 'Log in without chat',
+         has_come_online: 'has come online',
+         Unknown_sender: 'Unknown sender',
+         You_received_a_message_from_an_unknown_sender: 'You received a message from an unknown sender'
+      },
+      de: {
+         Logging_in: 'Login läuft…',
+         your_connection_is_unencrypted: 'Deine Verbindung ist UNverschlüsselt.',
+         your_connection_is_encrypted: 'Deine Verbindung ist verschlüsselt.',
+         your_buddy_closed_the_private_connection: 'Dein Freund hat die private Verbindung getrennt.',
+         start_private: 'Privat starten',
+         close_private: 'Privat abbrechen',
+         your_buddy_is_verificated: 'Dein Freund ist verifiziert.',
+         you_have_only_a_subscription_in_one_way: 'Die Freundschaft ist nur einseitig.',
+         authentication_query_sent: 'Authentifizierungsanfrage gesendet.',
+         your_message_wasnt_send_please_end_your_private_conversation: 'Deine Nachricht wurde nicht gesendet. Bitte beende die private Konversation.',
+         unencrypted_message_received: 'Unverschlüsselte Nachricht erhalten.',
+         your_message_wasnt_send_because_you_have_no_valid_subscription: 'Deine Nachricht wurde nicht gesandt, da die Freundschaft einseitig ist.',
+         not_available: 'Nicht verfügbar.',
+         no_connection: 'Keine Verbindung.',
+         relogin: 'Neu anmelden.',
+         trying_to_start_private_conversation: 'Versuche private Konversation zu starten.',
+         Verified: 'Verifiziert',
+         Unverified: 'Unverifiziert',
+         private_conversation_started: 'Private Konversation gestartet.',
+         private_conversation_aborted: 'Private Konversation abgebrochen.',
+         your_buddy_closed_the_private_conversation_you_should_do_the_same: 'Dein Freund hat die private Konversation beendet. Das solltest du auch tun!',
+         conversation_is_now_verified: 'Konversation ist jetzt verifiziert',
+         authentication_failed: 'Authentifizierung fehlgeschlagen.',
+         your_buddy_is_attempting_to_determine_: 'Dein Freund versucht herauszufinden ob er wirklich mit dir redet.',
+         to_authenticate_to_your_buddy: 'Um dich gegenüber deinem Freund zu verifizieren ',
+         enter_the_answer_and_click_answer: 'gib die Antwort ein und klick auf Antworten.',
+         enter_the_secret: 'gib das Geheimnis ein.',
+         Creating_your_private_key_: 'Wir werden jetzt deinen privaten Schlüssel generieren. Das kann einige Zeit in Anspruch nehmen.',
+         Authenticating_a_buddy_helps_: 'Einen Freund zu authentifizieren hilft sicher zustellen, dass die Person mit der du sprichst auch die ist die sie sagt.',
+         How_do_you_want_to_authenticate_your_buddy: 'Wie willst du {{bid_name}} (<b>{{bid_jid}}</b>) authentifizieren?',
+         Select_method: 'Wähle...',
+         Manual: 'Manual',
+         Question: 'Frage',
+         Secret: 'Geheimnis',
+         To_verify_the_fingerprint_: 'Um den Fingerprint zu verifizieren kontaktiere dein Freund über einen anderen Kommunikationsweg. Zum Beispiel per Telefonanruf.',
+         Your_fingerprint: 'Dein Fingerprint',
+         Buddy_fingerprint: 'Sein/Ihr Fingerprint',
+         Close: 'Schließen',
+         Compared: 'Verglichen',
+         To_authenticate_using_a_question_: 'Um die Authentifizierung per Frage durchzuführen, wähle eine Frage bei welcher nur dein Freund die Antwort weiß.',
+         Ask: 'Frage',
+         To_authenticate_pick_a_secret_: 'Um deinen Freund zu authentifizieren, wähle ein Geheimnis welches nur deinem Freund und dir bekannt ist.',
+         Compare: 'Vergleiche',
+         Fingerprints: 'Fingerprints',
+         Authentication: 'Authentifizierung',
+         Message: 'Nachricht',
+         Add_buddy: 'Freund hinzufügen',
+         rename_buddy: 'Freund umbenennen',
+         delete_buddy: 'Freund löschen',
+         Login: 'Anmeldung',
+         Username: 'Benutzername',
+         Password: 'Passwort',
+         Cancel: 'Abbrechen',
+         Connect: 'Verbinden',
+         Type_in_the_full_username_: 'Gib bitte den vollen Benutzernamen und optional ein Alias an.',
+         Alias: 'Alias',
+         Add: 'Hinzufügen',
+         Subscription_request: 'Freundschaftsanfrage',
+         You_have_a_request_from: 'Du hast eine Anfrage von',
+         Deny: 'Ablehnen',
+         Approve: 'Bestätigen',
+         Remove_buddy: 'Freund entfernen',
+         You_are_about_to_remove_: 'Du bist gerade dabei {{bid_name}} (<b>{{bid_jid}}</b>) von deiner Kontaktliste zu entfernen. Alle Chats werden geschlossen.',
+         Continue_without_chat: 'Weiter ohne Chat',
+         Please_wait: 'Bitte warten',
+         Login_failed: 'Chat-Anmeldung fehlgeschlagen',
+         Sorry_we_cant_authentikate_: 'Der Chatserver hat die Anmeldung abgelehnt. Falsches Passwort?',
+         Retry: 'Zurück',
+         clear_history: 'Lösche Verlauf',
+         New_message_from: 'Neue Nachricht von',
+         Should_we_notify_you_: 'Sollen wir dich in Zukunft über eingehende Nachrichten informieren, auch wenn dieser Tab nicht im Vordergrund ist?',
+         Please_accept_: 'Bitte klick auf den "Zulassen" Button oben.',
+         Menu: 'Menü',
+         Hide_offline: 'Offline ausblenden',
+         Show_offline: 'Offline einblenden',
+         About: 'Über',
+         dnd: 'Beschäftigt',
+         Mute: 'Ton aus',
+         Unmute: 'Ton an',
+         Subscription: 'Bezug',
+         both: 'beidseitig',
+         Status: 'Status',
+         online: 'online',
+         chat: 'chat',
+         away: 'abwesend',
+         xa: 'länger abwesend',
+         offline: 'offline',
+         none: 'keine',
+         Unknown_instance_tag: 'Unbekannter instance tag.',
+         Not_one_of_our_latest_keys: 'Nicht einer unserer letzten Schlüssel.',
+         Received_an_unreadable_encrypted_message: 'Eine unlesbare verschlüsselte Nachricht erhalten.',
+         Online: 'Online',
+         Chatty: 'Gesprächig',
+         Away: 'Abwesend',
+         Extended_away: 'Länger abwesend',
+         Offline: 'Offline',
+         Friendship_request: 'Freundschaftsanfrage',
+         Confirm: 'Bestätigen',
+         Dismiss: 'Ablehnen',
+         Remove: 'Löschen',
+         Online_help: 'Online Hilfe',
+         FN: 'Name',
+         N: ' ',
+         FAMILY: 'Familienname',
+         GIVEN: 'Vorname',
+         NICKNAME: 'Spitzname',
+         URL: 'URL',
+         ADR: 'Adresse',
+         STREET: 'Straße',
+         EXTADD: 'Zusätzliche Adresse',
+         LOCALITY: 'Ortschaft',
+         REGION: 'Region',
+         PCODE: 'Postleitzahl',
+         CTRY: 'Land',
+         TEL: 'Telefon',
+         NUMBER: 'Nummer',
+         EMAIL: 'E-Mail',
+         USERID: ' ',
+         ORG: 'Organisation',
+         ORGNAME: 'Name',
+         ORGUNIT: 'Abteilung',
+         TITLE: 'Titel',
+         ROLE: 'Rolle',
+         BDAY: 'Geburtstag',
+         DESC: 'Beschreibung',
+         PHOTO: ' ',
+         send_message: 'Sende Nachricht',
+         get_info: 'Benutzerinformationen',
+         Settings: 'Einstellungen',
+         Priority: 'Priorität',
+         Save: 'Speichern',
+         User_settings: 'Benutzereinstellungen',
+         A_fingerprint_: 'Ein Fingerabdruck wird dazu benutzt deinen Gesprächspartner zu identifizieren.',
+         Your_roster_is_empty_add_a: 'Deine Freundesliste ist leer, füge einen neuen Freund ',
+         new_buddy: 'hinzu',
+         is: 'ist',
+         Login_options: 'Anmeldeoptionen',
+         BOSH_url: 'BOSH url',
+         Domain: 'Domain',
+         Resource: 'Ressource',
+         On_login: 'Beim Anmelden',
+         Received_an_unencrypted_message: 'Unverschlüsselte Nachricht empfangen',
+         Sorry_your_buddy_doesnt_provide_any_information: 'Dein Freund stellt leider keine Informationen bereit.',
+         Info_about: 'Info über',
+         Authentication_aborted: 'Authentifizierung abgebrochen.',
+         Authentication_request_received: 'Authentifizierunganfrage empfangen.',
+         Log_in_without_chat: 'Anmelden ohne Chat',
+         Do_you_want_to_display_them: 'Möchtest du sie sehen?',
+         has_come_online: 'ist online gekommen',
+         Unknown_sender: 'Unbekannter Sender',
+         You_received_a_message_from_an_unknown_sender: 'Du hast eine Nachricht von einem unbekannten Sender erhalten'
+      },
+      es: {
+         Logging_in: 'Por favor, espere...',
+         your_connection_is_unencrypted: 'Su conexión no está cifrada.',
+         your_connection_is_encrypted: 'Su conexión está cifrada.',
+         your_buddy_closed_the_private_connection: 'Su amigo ha cerrado la conexión privada.',
+         start_private: 'Iniciar privado',
+         close_private: 'Cerrar privado',
+         your_buddy_is_verificated: 'Tu amigo está verificado.',
+         you_have_only_a_subscription_in_one_way: 'Sólo tienes una suscripción de un modo.',
+         authentication_query_sent: 'Consulta de verificación enviada.',
+         your_message_wasnt_send_please_end_your_private_conversation: 'Su mensaje no fue enviado. Por favor, termine su conversación privada.',
+         unencrypted_message_received: 'Mensaje no cifrado recibido:',
+         your_message_wasnt_send_because_you_have_no_valid_subscription: 'Su mensaje no se ha enviado, porque usted no tiene suscripción válida.',
+         not_available: 'No disponible',
+         no_connection: 'Sin conexión!',
+         relogin: 'iniciar sesión nuevamente',
+         trying_to_start_private_conversation: 'Intentando iniciar una conversación privada!',
+         Verified: 'Verificado',
+         Unverified: 'No verificado',
+         private_conversation_started: 'se inició una conversación privada.',
+         private_conversation_aborted: 'Conversación privada abortada!',
+         your_buddy_closed_the_private_conversation_you_should_do_the_same: 'Su amigo cerró la conversación privada! Usted debería hacer lo mismo.',
+         conversation_is_now_verified: 'La conversación es ahora verificada.',
+         authentication_failed: 'Fallo la verificación.',
+         your_buddy_is_attempting_to_determine_: 'Tu amigo está tratando de determinar si él o ella está realmente hablando con usted.',
+         to_authenticate_to_your_buddy: 'Para autenticar a su amigo, ',
+         enter_the_answer_and_click_answer: 'introduce la respuesta y haga clic en Contestar.',
+         enter_the_secret: 'especifique el secreto.',
+         Creating_your_private_key_: 'Ahora vamos a crear su clave privada. Esto puede tomar algún tiempo.',
+         Authenticating_a_buddy_helps_: 'Autenticación de un amigo ayuda a garantizar que la persona que está hablando es quien él o ella está diciendo.',
+         How_do_you_want_to_authenticate_your_buddy: '¿Cómo desea autenticar {{bid_name}} (<b>{{bid_jid}}</b>)?',
+         Select_method: 'Escoja un método...',
+         Manual: 'Manual',
+         Question: 'Pregunta',
+         Secret: 'Secreto',
+         To_verify_the_fingerprint_: 'Para verificar la firma digital, póngase en contacto con su amigo a través de algún otro canal autenticado, como el teléfono.',
+         Your_fingerprint: 'Tu firma digital',
+         Buddy_fingerprint: 'firma digital de tu amigo',
+         Close: 'Cerrar',
+         Compared: 'Comparado',
+         To_authenticate_using_a_question_: 'Para autenticar mediante una pregunta, elegir una pregunta cuya respuesta se conoce sólo usted y su amigo.',
+         Ask: 'Preguntar',
+         To_authenticate_pick_a_secret_: 'Para autenticar, elija un secreto conocido sólo por usted y su amigo.',
+         Compare: 'Comparar',
+         Fingerprints: 'Firmas digitales',
+         Authentication: 'Autenticación',
+         Message: 'Mensaje',
+         Add_buddy: 'Añadir amigo',
+         rename_buddy: 'renombrar amigo',
+         delete_buddy: 'eliminar amigo',
+         Login: 'Iniciar Sesión',
+         Username: 'Usuario',
+         Password: 'Contraseña',
+         Cancel: 'Cancelar',
+         Connect: 'Conectar',
+         Type_in_the_full_username_: 'Escriba el usuario completo y un alias opcional.',
+         Alias: 'Alias',
+         Add: 'Añadir',
+         Subscription_request: 'Solicitud de suscripción',
+         You_have_a_request_from: 'Tienes una petición de',
+         Deny: 'Rechazar',
+         Approve: 'Aprobar',
+         Remove_buddy: 'Eliminar amigo',
+         You_are_about_to_remove_: 'Vas a eliminar a {{bid_name}} (<b>{{bid_jid}}</b>) de tu lista de amigos. Todas las conversaciones relacionadas serán cerradas.',
+         Continue_without_chat: 'Continuar',
+         Please_wait: 'Espere por favor',
+         Login_failed: 'Fallo el inicio de sesión',
+         Sorry_we_cant_authentikate_: 'Lo sentimos, no podemos autentificarlo en nuestro servidor de chat. ¿Tal vez la contraseña es incorrecta?',
+         Retry: 'Reintentar',
+         clear_history: 'Borrar el historial',
+         New_message_from: 'Nuevo mensaje de',
+         Should_we_notify_you_: '¿Debemos notificarle sobre nuevos mensajes en el futuro?',
+         Please_accept_: 'Por favor, haga clic en el botón "Permitir" en la parte superior.',
+         dnd: 'No Molestar',
+         Mute: 'Desactivar sonido',
+         Unmute: 'Activar sonido',
+         Subscription: 'Suscripción',
+         both: 'ambos',
+         Status: 'Estado',
+         online: 'en línea',
+         chat: 'chat',
+         away: 'ausente',
+         xa: 'mas ausente',
+         offline: 'desconectado',
+         none: 'nadie',
+         Unknown_instance_tag: 'Etiqueta de instancia desconocida.',
+         Not_one_of_our_latest_keys: 'No de nuestra ultima tecla.',
+         Received_an_unreadable_encrypted_message: 'Se recibió un mensaje cifrado ilegible.',
+         Online: 'En linea',
+         Chatty: 'Hablador',
+         Away: 'Ausente',
+         Extended_away: 'Mas ausente',
+         Offline: 'Desconectado',
+         Friendship_request: 'Solicitud de amistad',
+         Confirm: 'Confirmar',
+         Dismiss: 'Rechazar',
+         Remove: 'Eliminar',
+         Online_help: 'Ayuda en línea',
+         FN: 'Nombre completo ',
+         N: ' ',
+         FAMILY: 'Apellido',
+         GIVEN: 'Nombre',
+         NICKNAME: 'Apodar',
+         URL: 'URL',
+         ADR: 'Dirección',
+         STREET: 'Calle',
+         EXTADD: 'Extendido dirección',
+         LOCALITY: 'Población',
+         REGION: 'Región',
+         PCODE: 'Código postal',
+         CTRY: 'País',
+         TEL: 'Teléfono',
+         NUMBER: 'Número',
+         EMAIL: 'Emilio',
+         USERID: ' ',
+         ORG: 'Organización',
+         ORGNAME: 'Nombre',
+         ORGUNIT: 'Departamento',
+         TITLE: 'Título',
+         ROLE: 'Rol',
+         BDAY: 'Cumpleaños',
+         DESC: 'Descripción',
+         PHOTO: ' ',
+         send_message: 'mandar un texto',
+         get_info: 'obtener información',
+         Settings: 'Ajustes',
+         Priority: 'Prioridad',
+         Save: 'Guardar',
+         User_settings: 'Configuración de usuario',
+         A_fingerprint_: 'La huella digital se utiliza para que puedas estar seguro que la persona con la que estas hablando es quien realmente dice ser',
+         Your_roster_is_empty_add_a: 'Tu lista de amigos esta vacia',
+         new_buddy: 'Nuevo amigo',
+         is: 'es',
+         Login_options: 'Opciones de login',
+         BOSH_url: 'BOSH url',
+         Domain: 'Dominio',
+         Resource: 'Recurso',
+         On_login: 'Iniciar sesión',
+         Received_an_unencrypted_message: 'Recibe un mensaje no cifrado'
+      }
+   };
+}(jQuery));
diff --git a/jsxc.lib.webrtc.js b/jsxc.lib.webrtc.js
new file mode 100644
index 0000000..6b1ea17
--- /dev/null
+++ b/jsxc.lib.webrtc.js
@@ -0,0 +1,1051 @@
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org> <br>
+ * Released under the MIT license
+ * 
+ * Please see http://www.jsxc.org/
+ * 
+ * @author Klaus Herberth <klaus at jsxc.org>
+ * @version 1.0.0
+ * @license MIT
+ */
+
+/* global jsxc, Strophe, SDPUtil, getUserMediaWithConstraints, setupRTC, jQuery, MediaStreamTrack */
+
+var RTC = null, RTCPeerconnection = null;
+
+jsxc.gui.template.incomingCall = '<h3>%%Incoming_call%%</h3>\
+        <p>%%Do_you_want_to_accept_the_call_from%% {{bid_name}}?</p>\
+        <p class="jsxc_right">\
+            <a href="#" class="button jsxc_reject">%%Reject%%</a> <a href="#" class="button creation jsxc_accept">%%Accept%%</a>\
+         </p>';
+
+jsxc.gui.template.allowMediaAccess = '<p>%%Please_allow_access_to_microphone_and_camera%%</p>';
+
+jsxc.gui.template.videoWindow = '<div class="jsxc_webrtc">\
+            <div class="jsxc_chatarea">\
+                <ul></ul>\
+            </div>\
+            <div class="jsxc_videoContainer">\
+                <video class="jsxc_localvideo" autoplay></video>\
+                <video class="jsxc_remotevideo" autoplay></video>\
+                <div class="jsxc_status"></div>\
+               <div class="bubblingG">\
+                  <span id="bubblingG_1">\
+                  </span>\
+                  <span id="bubblingG_2">\
+                  </span>\
+                  <span id="bubblingG_3">\
+                  </span>\
+               </div>\
+                <div class="jsxc_noRemoteVideo">\
+                   <div>\
+                     <div></div>\
+                     <p>%%No_video_signal%%</p>\
+                     <div></div>\
+                   </div>\
+                </div>\
+            </div>\
+            <div class="jsxc_controlbar">\
+                <button type="button" class="jsxc_hangUp">%%hang_up%%</button>\
+                <input type="range" class="jsxc_volume" min="0.0" max="1.0" step="0.05" value="0.5" />\
+                <div class="jsxc_buttongroup">\
+                    <button type="button" class="jsxc_snapshot">%%snapshot%%</button><button type="button" class="jsxc_snapshots">▼</button>\
+                </div>\
+                <!-- <button type="button" class="jsxc_mute_local">%%mute_my_audio%%</button>\
+                <button type="button" class="jsxc_pause_local">%%pause_my_video%%</button> --> \
+                <button type="button" class="jsxc_showchat">%%chat%%</button>\
+                <button type="button" class="jsxc_fullscreen">%%fullscreen%%</button>\
+                <button type="button" class="jsxc_info">%%Info%%</button>\
+            </div>\
+            <div class="jsxc_multi">\
+               <div class="jsxc_snapshotbar">\
+                   <p>No pictures yet!</p>\
+               </div>\n\
+               <!--<div class="jsxc_chatarea">\
+                   <ul></ul>\
+               </div>-->\
+               <div class="jsxc_infobar"></div>\
+            </div>\
+        </div>';
+
+(function($) {
+   "use strict";
+
+   /**
+    * WebRTC namespace for jsxc.
+    * 
+    * @namespace jsxc.webrtc
+    */
+   jsxc.webrtc = {
+      /** strophe connection */
+      conn: null,
+
+      /** local video stream */
+      localStream: null,
+
+      /** remote video stream */
+      remoteStream: null,
+
+      /** jid of the last caller */
+      last_caller: null,
+
+      /** should we auto accept incoming calls? */
+      AUTO_ACCEPT: false,
+
+      /** required disco features */
+      reqVideoFeatures: [ 'urn:xmpp:jingle:apps:rtp:video', 'urn:xmpp:jingle:apps:rtp:audio', 'urn:xmpp:jingle:transports:ice-udp:1', 'urn:xmpp:jingle:apps:dtls:0' ],
+
+      /** bare jid to current jid mapping */
+      chatJids: {},
+
+      /**
+       * Initialize webrtc plugin.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      init: function() {
+         var self = jsxc.webrtc;
+
+         // shortcut
+         self.conn = jsxc.xmpp.conn;
+
+         if (RTC.browser === 'firefox') {
+            self.conn.jingle.media_constraints.mandatory.MozDontOfferDataChannel = true;
+         }
+
+         if (!self.conn.jingle) {
+            jsxc.error('No jingle plugin found!');
+            return;
+         }
+
+         // jingle configuration
+         self.conn.jingle.PRANSWER = false;
+         self.conn.jingle.AUTOACCEPT = false;
+         self.conn.jingle.ice_config = jsxc.storage.getUserItem('iceConfig');
+         self.conn.jingle.MULTIPARTY = false;
+         self.conn.jingle.pc_constraints = RTC.pc_constraints;
+
+         $(document).on('message.jsxc', $.proxy(self.onMessage, self));
+         $(document).on('presence.jsxc', $.proxy(self.onPresence, self));
+
+         $(document).on('mediaready.jingle', $.proxy(self.onMediaReady, self));
+         $(document).on('mediafailure.jingle', $.proxy(self.onMediaFailure, self));
+         $(document).on('callincoming.jingle', $.proxy(self.onCallIncoming, self));
+         $(document).on('callterminated.jingle', $.proxy(self.onCallTerminated, self));
+         $(document).on('ringing.jingle', $.proxy(self.onCallRinging, self));
+
+         $(document).on('remotestreamadded.jingle', $.proxy(self.onRemoteStreamAdded, self));
+         $(document).on('remotestreamremoved.jingle', $.proxy(self.onRemoteStreamRemoved, self));
+         $(document).on('iceconnectionstatechange.jingle', $.proxy(self.onIceConnectionStateChanged, self));
+         $(document).on('nostuncandidates.jingle', $.proxy(self.noStunCandidates, self));
+
+         $(document).on('error.jingle', function(ev, sid, error) {
+            jsxc.error('[JINGLE]', error);
+         });
+
+         if (self.conn.disco) {
+            self.conn.disco.addFeature('urn:xmpp:jingle:apps:dtls:0');
+         }
+
+         if (self.conn.caps) {
+            $(document).on('caps.strophe', $.proxy(self.onCaps, self));
+         }
+
+         self.getTurnCrendentials();
+      },
+
+      /**
+       * Checks if cached configuration is valid and if necessary update it.
+       * 
+       * @memberOf jsxc.webrtc
+       */
+      getTurnCrendentials: function() {
+
+         if (!jsxc.options.turnCredentialsPath) {
+            jsxc.debug('No path for TURN credentials defined!');
+            return;
+         }
+
+         var ttl = (jsxc.storage.getUserItem('iceValidity') || 0) - (new Date()).getTime();
+         if (ttl > 0) {
+            // credentials valid
+
+            window.setTimeout(jsxc.webrtc.getTurnCrendentials, ttl + 500);
+            return;
+         }
+
+         $.ajax(jsxc.options.turnCredentialsPath, {
+            async: true,
+            success: function(data) {
+               var iceConfig = {
+                  iceServers: [ {
+                     url: 'turn:' + data.url,
+                     credential: data.credential,
+                     username: data.username
+                  } ]
+               };
+
+               jsxc.webrtc.conn.jingle.ice_config = iceConfig;
+               jsxc.storage.setUserItem('iceConfig', iceConfig);
+               jsxc.storage.setUserItem('iceValidity', (new Date()).getTime() + 1000 * data.ttl);
+            },
+            dataType: 'json'
+         });
+      },
+
+      /**
+       * Return list of video capable resources.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param jid
+       * @returns {Array}
+       */
+      getCapableRes: function(jid) {
+         var self = jsxc.webrtc;
+         var bid = jsxc.jidToBid(jid);
+         var res = jsxc.storage.getUserItem('res', bid) || [];
+
+         var available = [];
+         $.each(res, function(r) {
+            if (self.conn.caps.hasFeatureByJid(bid + '/' + r, self.reqVideoFeatures)) {
+               available.push(r);
+            }
+         });
+
+         return available;
+      },
+
+      /**
+       * Add "video" button to roster
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param bid bid of roster item
+       * @param data data wich belongs to bid
+       * @param el the roster item
+       */
+      onAddRosterItem: function(event, bid, data, el) {
+         var self = jsxc.webrtc;
+
+         if (!self.conn) {
+            $(document).one('connectionReady.jsxc', function() {
+               self.onAddRosterItem(null, bid, data, el);
+            });
+            return;
+         }
+
+         var videoIcon = $('<div class="jsxc_video jsxc_disabled" title="' + jsxc.l.Start_video_call + '"></div>');
+
+         videoIcon.click(function() {
+            self.startCall(data.jid);
+            return false;
+         });
+
+         el.find('.jsxc_options.jsxc_left').append(videoIcon);
+
+         el.on('extra.jsxc', function() {
+            self.updateIcon(bid);
+         });
+      },
+
+      /**
+       * Add "video" button to window menu.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param win jQuery window object
+       */
+      initWindow: function(event, win) {
+         var self = jsxc.webrtc;
+
+         jsxc.debug('webrtc.initWindow');
+
+         if (!self.conn) {
+            $(document).one('connectionReady.jsxc', function() {
+               self.initWindow(null, win);
+            });
+            return;
+         }
+
+         var div = $('<div>').addClass('jsxc_video');
+         win.find('.jsxc_transfer:eq(1)').after(div);
+
+         self.updateIcon(jsxc.jidToBid(win.data('jid')));
+      },
+
+      /**
+       * Enable or disable "video" icon and assign full jid.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param bid CSS conform jid
+       */
+      updateIcon: function(bid) {
+         jsxc.debug('Update icon', bid);
+
+         var self = jsxc.webrtc;
+         var win = jsxc.gui.window.get(bid);
+         var jid = win.data('jid') || jsxc.storage.getUserItem('buddy', bid).jid;
+
+         var el = win.find('.jsxc_video').add(jsxc.gui.roster.getItem(bid).find('.jsxc_video'));
+
+         var capableRes = self.getCapableRes(jid);
+         var targetRes = Strophe.getResourceFromJid(jid);
+
+         if (targetRes === null) {
+            $.each(jsxc.storage.getUserItem('buddy', bid).res, function(index, val) {
+               if (capableRes.indexOf(val) > -1) {
+                  targetRes = val;
+                  return false;
+               }
+            });
+         }
+
+         el.off('click');
+
+         if (capableRes.indexOf(targetRes) > -1) {
+            el.click(function() {
+               self.startCall(jid + '/' + targetRes);
+            });
+
+            el.removeClass('jsxc_disabled');
+
+            el.attr('title', jsxc.translate('%%Start video call%%'));
+         } else {
+            el.addClass('jsxc_disabled');
+
+            el.attr('title', jsxc.translate('%%Video call not possible.%%'));
+         }
+      },
+
+      /**
+       * Check if full jid changed.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param e
+       * @param from full jid
+       */
+      onMessage: function(e, from) {
+         var self = jsxc.webrtc;
+         var bid = jsxc.jidToBid(from);
+
+         jsxc.debug('webrtc.onmessage', from);
+
+         if (self.chatJids[bid] !== from) {
+            self.updateIcon(bid);
+            self.chatJids[bid] = from;
+         }
+      },
+
+      /**
+       * Update icon on presence.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param ev
+       * @param status
+       * @private
+       */
+      onPresence: function(ev, jid) {
+         var self = jsxc.webrtc;
+
+         jsxc.debug('webrtc.onpresence', jid);
+
+         self.updateIcon(jsxc.jidToBid(jid));
+      },
+
+      /**
+       * Display status message to user.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param txt message
+       * @param d duration in ms
+       */
+      setStatus: function(txt, d) {
+         var status = $('.jsxc_webrtc .jsxc_status');
+         var duration = (typeof d === 'undefined' || d === null) ? 4000 : d;
+
+         jsxc.debug('[Webrtc]', txt);
+
+         if (status.html()) {
+            // attach old messages
+            txt = status.html() + '<br />' + txt;
+         }
+
+         status.html(txt);
+
+         status.css({
+            'margin-left': '-' + (status.width() / 2) + 'px',
+            opacity: 0,
+            display: 'block'
+         });
+
+         status.stop().animate({
+            opacity: 1
+         });
+
+         clearTimeout(status.data('timeout'));
+
+         if (duration === 0) {
+            return;
+         }
+
+         var to = setTimeout(function() {
+            status.stop().animate({
+               opacity: 0
+            }, function() {
+               status.html('');
+            });
+         }, duration);
+
+         status.data('timeout', to);
+      },
+
+      /**
+       * Update "video" button if we receive cap information.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param jid
+       */
+      onCaps: function(event, jid) {
+         var self = jsxc.webrtc;
+
+         self.updateIcon(jsxc.jidToBid(jid));
+      },
+
+      /**
+       * Called if video/audio is ready. Open window and display some messages.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param stream
+       */
+      onMediaReady: function(event, stream) {
+         jsxc.debug('media ready');
+
+         var self = jsxc.webrtc;
+
+         self.localStream = stream;
+         self.conn.jingle.localStream = stream;
+
+         jsxc.gui.showVideoWindow(self.last_caller);
+
+         var i;
+         for (i = 0; i < stream.getAudioTracks().length; i++) {
+            self.setStatus((stream.getAudioTracks().length > 0) ? 'Use local audio device.' : 'No local audio device.');
+
+            jsxc.debug('using audio device "' + stream.getAudioTracks()[i].label + '"');
+         }
+         for (i = 0; i < stream.getVideoTracks().length; i++) {
+            self.setStatus((stream.getVideoTracks().length > 0) ? 'Use local video device.' : 'No local video device.');
+
+            jsxc.debug('using video device "' + stream.getVideoTracks()[i].label + '"');
+            $('#jsxc_dialog .jsxc_localvideo').show();
+         }
+
+         $(document).one('cleanup.dialog.jsxc', $.proxy(self.hangUp, self));
+         $(document).trigger('finish.mediaready.jsxc');
+      },
+
+      /**
+       * Called if media failes.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      onMediaFailure: function(ev, err) {
+         this.setStatus('media failure');
+
+         jsxc.gui.window.postMessage(jsxc.jidToBid(jsxc.webrtc.last_caller), 'sys', jsxc.translate('%%Media failure%%: ') + err.name);
+         jsxc.debug('media failure: ' + err.name);
+      },
+
+      /**
+       * Called on incoming call.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param sid Session id
+       */
+      onCallIncoming: function(event, sid) {
+         jsxc.debug('incoming call' + sid);
+
+         var self = this;
+         var sess = this.conn.jingle.sessions[sid];
+         var bid = jsxc.jidToBid(sess.peerjid);
+
+         jsxc.gui.window.postMessage(bid, 'sys', jsxc.translate('%%Incoming call.%%'));
+
+         // display notification
+         jsxc.notification.notify(jsxc.translate('%%Incoming call%%'), jsxc.translate('%%from%% ' + bid));
+
+         // send signal to partner
+         sess.sendRinging();
+
+         jsxc.webrtc.last_caller = sess.peerjid;
+
+         jsxc.switchEvents({
+            'mediaready.jingle': function(event, stream) {
+               self.setStatus('Accept call');
+
+               sess.localStream = stream;
+               sess.peerconnection.addStream(stream);
+
+               sess.sendAnswer();
+               sess.accept();
+            },
+            'mediafailure.jingle': function() {
+               sess.sendTerminate('decline');
+               sess.terminate();
+            }
+         });
+
+         if (jsxc.webrtc.AUTO_ACCEPT) {
+            self.reqUserMedia();
+            return;
+         }
+
+         var dialog = jsxc.gui.dialog.open(jsxc.gui.template.get('incomingCall', bid), {
+            noClose: true
+         });
+
+         dialog.find('.jsxc_accept').click(function() {
+            $(document).trigger('accept.call.jsxc');
+
+            self.reqUserMedia();
+         });
+
+         dialog.find('.jsxc_reject').click(function() {
+            jsxc.gui.dialog.close();
+            $(document).trigger('reject.call.jsxc');
+
+            sess.sendTerminate('decline');
+            sess.terminate();
+         });
+      },
+
+      /**
+       * Called if call is terminated.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param sid Session id
+       * @param reason Reason for termination
+       * @param [text] Optional explanation
+       */
+      onCallTerminated: function(event, sid, reason, text) {
+         this.setStatus('call terminated ' + sid + (reason ? (': ' + reason + ' ' + text) : ''));
+
+         var bid = jsxc.jidToBid(jsxc.webrtc.last_caller);
+
+         if (this.localStream) {
+            this.localStream.stop();
+         }
+
+         if ($('.jsxc_videoContainer').length) {
+            $('.jsxc_remotevideo')[0].src = "";
+            $('.jsxc_localvideo')[0].src = "";
+         }
+
+         this.conn.jingle.localStream = null;
+         this.localStream = null;
+         this.remoteStream = null;
+
+         var win = $('#jsxc_dialog .jsxc_chatarea > ul > li');
+         $('#jsxc_windowList > ul').prepend(win.detach());
+         win.find('.slimScrollDiv').resizable('enable');
+
+         $(document).off('cleanup.dialog.jsxc');
+         $(document).off('error.jingle');
+         jsxc.gui.dialog.close();
+
+         jsxc.gui.window.postMessage(bid, 'sys', jsxc.translate('%%Call terminated%%' + (reason ? (': %%' + reason + '%%') : '') + '.'));
+      },
+
+      /**
+       * Remote station is ringing.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      onCallRinging: function() {
+         this.setStatus('ringing...', 0);
+      },
+
+      /**
+       * Called if we receive a remote stream.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param data
+       * @param sid Session id
+       */
+      onRemoteStreamAdded: function(event, data, sid) {
+         this.setStatus('Remote stream for session ' + sid + ' added.');
+
+         var stream = data.stream;
+         this.remoteStream = stream;
+
+         var sess = this.conn.jingle.sessions[sid];
+
+         var isVideoDevice = stream.getVideoTracks().length > 0;
+         var isAudioDevice = stream.getAudioTracks().length > 0;
+
+         sess.remoteDevices = {
+            video: isVideoDevice,
+            audio: isAudioDevice
+         };
+
+         this.setStatus(isVideoDevice ? 'Use remote video device.' : 'No remote video device');
+         this.setStatus(isAudioDevice ? 'Use remote audio device.' : 'No remote audio device');
+
+         if ($('.jsxc_remotevideo').length) {
+            RTC.attachMediaStream($('#jsxc_dialog .jsxc_remotevideo'), stream);
+
+            $('#jsxc_dialog .jsxc_' + (isVideoDevice ? 'remotevideo' : 'noRemoteVideo')).addClass('jsxc_deviceAvailable');
+         }
+      },
+
+      /**
+       * Called if the remote stream was removed.
+       * 
+       * @private
+       * @meberOf jsxc.webrtc
+       * @param event
+       * @param data
+       * @param sid Session id
+       */
+      onRemoteStreamRemoved: function(event, data, sid) {
+         this.setStatus('Remote stream for session ' + sid + ' removed.');
+      },
+
+      /**
+       * Extracts local and remote ip and display it to the user.
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       * @param event
+       * @param sid session id
+       * @param sess
+       */
+      onIceConnectionStateChanged: function(event, sid, sess) {
+         var sigState = sess.peerconnection.signalingState;
+         var iceCon = sess.peerconnection.iceConnectionState;
+
+         jsxc.debug('iceGat state for ' + sid, sess.peerconnection.iceGatheringState);
+         jsxc.debug('iceCon state for ' + sid, iceCon);
+         jsxc.debug('sig state for ' + sid, sigState);
+
+         if (sigState === 'stable' && (iceCon === 'connected' || iceCon === 'completed')) {
+
+            $('#jsxc_dialog .jsxc_deviceAvailable').show();
+            $('#jsxc_dialog .bubblingG').hide();
+
+            var localSDP = sess.peerconnection.localDescription.sdp;
+            var remoteSDP = sess.peerconnection.remoteDescription.sdp;
+
+            sess.local_fp = SDPUtil.parse_fingerprint(SDPUtil.find_line(localSDP, 'a=fingerprint:')).fingerprint;
+            sess.remote_fp = SDPUtil.parse_fingerprint(SDPUtil.find_line(remoteSDP, 'a=fingerprint:')).fingerprint;
+
+            var ip_regex = "(\\d{1,3}\\.\\d{1,3}.\\d{1,3}\\.\\d{1,3}) \\d+ typ host";
+
+            sess.remote_ip = remoteSDP.match(new RegExp(ip_regex))[1];
+            sess.local_ip = localSDP.match(new RegExp(ip_regex))[1];
+
+            var regex = new RegExp(ip_regex, 'g');
+            var match;
+            while ((match = regex.exec(remoteSDP)) !== null) {
+               if (match[1] !== sess.remote_ip) {
+                  alert('!!! WARNING !!!\n\nPossible Man-in-the-middle attack detected!\n\nYou should close the connection.');
+                  return;
+               }
+            }
+
+            var text = '<p>';
+            text += '<b>' + jsxc.translate('%%Local IP%%: ') + '</b>' + sess.local_ip + '<br />';
+            text += '<b>' + jsxc.translate('%%Remote IP%%: ') + '</b>' + sess.remote_ip + '<br />';
+            text += '<b>' + jsxc.translate('%%Local Fingerprint%%: ') + '</b>' + sess.local_fp + '<br />';
+            text += '<b>' + jsxc.translate('%%Remote Fingerprint%%: ') + '</b>' + sess.remote_fp;
+            text += '</p>';
+
+            $('#jsxc_dialog .jsxc_infobar').html(text);
+         } else if (iceCon === 'failed') {
+            jsxc.gui.window.postMessage(jsxc.jidToBid(sess.peerjid), 'sys', jsxc.translate('%%ICE connection failure%%.'));
+
+            $(document).off('cleanup.dialog.jsxc');
+
+            sess.sendTerminate('failed-transport');
+            sess.terminate();
+
+            $(document).trigger('callterminated.jingle');
+         }
+      },
+
+      /**
+       * No STUN candidates found
+       * 
+       * @private
+       * @memberOf jsxc.webrtc
+       */
+      noStunCandidates: function() {
+
+      },
+
+      /**
+       * Start a call to the specified jid.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param jid full jid
+       * @param um requested user media
+       */
+      startCall: function(jid, um) {
+         var self = this;
+
+         if (Strophe.getResourceFromJid(jid) === null) {
+            jsxc.debug('We need a full jid');
+            return;
+         }
+
+         self.last_caller = jid;
+
+         jsxc.switchEvents({
+            'finish.mediaready.jsxc': function() {
+               self.setStatus('Initiate call');
+
+               jsxc.gui.window.postMessage(jsxc.jidToBid(jid), 'sys', jsxc.translate('%%Call started.%%'));
+
+               $(document).one('error.jingle', function(e, sid, error) {
+                  if (error.source !== 'offer') {
+                     return;
+                  }
+
+                  $(document).off('cleanup.dialog.jsxc');
+                  setTimeout(function() {
+                     jsxc.gui.showAlert("Sorry, we couldn't establish a connection. Maybe your buddy is offline.");
+                  }, 500);
+               });
+
+               self.conn.jingle.initiate(jid, self.conn.jid.toLowerCase());
+            },
+            'mediafailure.jingle': function() {
+               jsxc.gui.dialog.close();
+            }
+         });
+
+         self.reqUserMedia(um);
+      },
+
+      /**
+       * Hang up the current call.
+       * 
+       * @memberOf jsxc.webrtc
+       */
+      hangUp: function(reason, text) {
+         $(document).off('cleanup.dialog.jsxc');
+
+         jsxc.webrtc.conn.jingle.terminate(null, reason, text);
+         $(document).trigger('callterminated.jingle');
+      },
+
+      /**
+       * Request video and audio from local user.
+       * 
+       * @memberOf jsxc.webrtc
+       */
+      reqUserMedia: function(um) {
+         if (this.localStream) {
+            $(document).trigger('mediaready.jingle', [ this.localStream ]);
+            return;
+         }
+
+         um = um || [ 'video', 'audio' ];
+
+         jsxc.gui.dialog.open(jsxc.gui.template.get('allowMediaAccess'), {
+            noClose: true
+         });
+         this.setStatus('please allow access to microphone and camera');
+
+         if (typeof MediaStreamTrack !== 'undefined' && typeof MediaStreamTrack.getSources !== 'undefined') {
+            MediaStreamTrack.getSources(function(sourceInfo) {
+               var availableDevices = sourceInfo.map(function(el) {
+
+                  return el.kind;
+               });
+
+               um = um.filter(function(el) {
+                  return availableDevices.indexOf(el) !== -1;
+               });
+
+               getUserMediaWithConstraints(um);
+            });
+         } else {
+            getUserMediaWithConstraints(um);
+         }
+      },
+
+      /**
+       * Make a snapshot from a video stream and display it.
+       * 
+       * @memberOf jsxc.webrtc
+       * @param video Video stream
+       */
+      snapshot: function(video) {
+         if (!video) {
+            jsxc.debug('Missing video element');
+         }
+
+         $('.jsxc_snapshotbar p').remove();
+
+         var canvas = $('<canvas/>').css('display', 'none').appendTo('body').attr({
+            width: video.width(),
+            height: video.height()
+         }).get(0);
+         var ctx = canvas.getContext('2d');
+
+         ctx.drawImage(video[0], 0, 0);
+         var img = $('<img/>');
+         var url = null;
+
+         try {
+            url = canvas.toDataURL('image/jpeg');
+         } catch (err) {
+            jsxc.warn('Error', err);
+            return;
+         }
+
+         img[0].src = url;
+         var link = $('<a/>').attr({
+            target: '_blank',
+            href: url
+         });
+         link.append(img);
+         $('.jsxc_snapshotbar').append(link);
+
+         canvas.remove();
+      }
+   };
+
+   /**
+    * Display window for video call.
+    * 
+    * @memberOf jsxc.gui
+    */
+   jsxc.gui.showVideoWindow = function(jid) {
+      var self = jsxc.webrtc;
+
+      $(document).one('complete.dialog.jsxc', function() {
+
+         // mute own video element to avoid echoes
+         $('#jsxc_dialog .jsxc_localvideo')[0].muted = true;
+         $('#jsxc_dialog .jsxc_localvideo')[0].volume = 0;
+
+         var rv = $('#jsxc_dialog .jsxc_remotevideo');
+         var lv = $('#jsxc_dialog .jsxc_localvideo');
+
+         lv.draggable({
+            containment: "parent"
+         });
+
+         RTC.attachMediaStream(lv, self.localStream);
+
+         var w_dialog = $('#jsxc_dialog').width();
+         var w_remote = rv.width();
+
+         // fit in video
+         if (w_remote > w_dialog) {
+            var scale = w_dialog / w_remote;
+            var new_h = rv.height() * scale;
+            var new_w = w_dialog;
+            var vc = $('#jsxc_dialog .jsxc_videoContainer');
+
+            rv.height(new_h);
+            rv.width(new_w);
+
+            vc.height(new_h);
+            vc.width(new_w);
+
+            lv.height(lv.height() * scale);
+            lv.width(lv.width() * scale);
+         }
+
+         if (self.remoteStream) {
+            RTC.attachMediaStream(rv, self.remoteStream);
+
+            $('#jsxc_dialog .jsxc_' + (self.remoteStream.getVideoTracks().length > 0 ? 'remotevideo' : 'noRemoteVideo')).addClass('jsxc_deviceAvailable');
+         }
+
+         var toggleMulti = function(elem, open) {
+            $('#jsxc_dialog .jsxc_multi > div').not(elem).slideUp();
+
+            var opt = {
+               complete: jsxc.gui.dialog.resize
+            };
+
+            if (open) {
+               elem.slideDown(opt);
+            } else {
+               elem.slideToggle(opt);
+            }
+         };
+
+         var win = jsxc.gui.window.open(jsxc.jidToBid(jid));
+
+         win.find('.slimScrollDiv').resizable('disable');
+         win.find('.jsxc_textarea').slimScroll({
+            height: 413
+         });
+         win.find('.jsxc_emoticons').css('top', (413 + 6) + 'px');
+
+         $('#jsxc_dialog .jsxc_chatarea ul').append(win.detach());
+
+         $('#jsxc_dialog .jsxc_hangUp').click(function() {
+            jsxc.webrtc.hangUp();
+         });
+
+         $('#jsxc_dialog .jsxc_snapshot').click(function() {
+            jsxc.webrtc.snapshot(rv);
+            toggleMulti($('#jsxc_dialog .jsxc_snapshotbar'), true);
+         });
+
+         $('#jsxc_dialog .jsxc_snapshots').click(function() {
+            toggleMulti($('#jsxc_dialog .jsxc_snapshotbar'));
+         });
+
+         $('#jsxc_dialog .jsxc_showchat').click(function() {
+            var chatarea = $('#jsxc_dialog .jsxc_chatarea');
+
+            if (chatarea.is(':hidden')) {
+               chatarea.show();
+               $('#jsxc_dialog .jsxc_webrtc').width('900');
+               jsxc.gui.dialog.resize({
+                  width: '920px'
+               });
+            } else {
+               chatarea.hide();
+               $('#jsxc_dialog .jsxc_webrtc').width('650');
+               jsxc.gui.dialog.resize({
+                  width: '660px'
+               });
+            }
+         });
+
+         $('#jsxc_dialog .jsxc_info').click(function() {
+            toggleMulti($('#jsxc_dialog .jsxc_infobar'));
+         });
+
+         $('#jsxc_dialog .jsxc_fullscreen').click(function() {
+
+            if ($.support.fullscreen) {
+               // Reset position of localvideo
+               $(document).one('disabled.fullscreen', function() {
+                  lv.removeAttr('style');
+               });
+
+               $('#jsxc_dialog .jsxc_videoContainer').fullscreen();
+            }
+         });
+
+         $('#jsxc_dialog .jsxc_volume').change(function() {
+            rv[0].volume = $(this).val();
+         });
+
+         $('#jsxc_dialog .jsxc_volume').dblclick(function() {
+            $(this).val(0.5);
+         });
+      });
+
+      jsxc.gui.dialog.open(jsxc.gui.template.get('videoWindow'), {
+         noClose: true
+      });
+   };
+
+   $.extend(jsxc.CONST, {
+      KEYCODE_ENTER: 13,
+      KEYCODE_ESC: 27
+   });
+
+   $(document).ready(function() {
+      RTC = setupRTC();
+
+      if (RTC !== null) {
+         RTCPeerconnection = RTC.peerconnection;
+
+         $(document).on('add.roster.jsxc', jsxc.webrtc.onAddRosterItem);
+         $(document).on('init.window.jsxc', jsxc.webrtc.initWindow);
+         $(document).on('attached.jsxc', jsxc.webrtc.init);
+      }
+   });
+
+   $.extend(jsxc.l10n.en, {
+      Please_allow_access_to_microphone_and_camera: 'Please click the "Allow" button at the top, to allow access to microphone and camera.',
+      Incoming_call: 'Incoming call',
+      from: 'from',
+      Do_you_want_to_accept_the_call_from: 'Do you want to accept the call from',
+      Reject: 'Reject',
+      Accept: 'Accept',
+      hang_up: 'hang up',
+      snapshot: 'snapshot',
+      mute_my_audio: 'mute my audio',
+      pause_my_video: 'pause my video',
+      fullscreen: 'fullscreen',
+      Info: 'Info',
+      Local_IP: 'Local IP',
+      Remote_IP: 'Remote IP',
+      Local_Fingerprint: 'Local fingerprint',
+      Remote_Fingerprint: 'Remote fingerprint',
+      Video_call_not_possible: 'Video call not possible. Your buddy does not support video calls.',
+      Start_video_call: 'Start video call'
+   });
+
+   $.extend(jsxc.l10n.de, {
+      Please_allow_access_to_microphone_and_camera: 'Bitte klick auf den "Zulassen" Button oben, um den Zugriff auf Kamera und Mikrofon zu erlauben.',
+      Incoming_call: 'Eingehender Anruf',
+      from: 'von',
+      Do_you_want_to_accept_the_call_from: 'Möchtest Du den Anruf annehmen von',
+      Reject: 'Ablehnen',
+      Accept: 'Annehmen',
+      hang_up: 'Auflegen',
+      snapshot: 'Schnappschuss',
+      mute_my_audio: 'Mein Ton aus',
+      pause_my_video: 'Mein Video pausieren',
+      fullscreen: 'Vollbild',
+      Info: 'Info',
+      Local_IP: 'Lokale IP',
+      Remote_IP: 'Remote IP',
+      Local_Fingerprint: 'Lokaler Fingerprint',
+      Remote_Fingerprint: 'Remote Fingerprint',
+      Video_call_not_possible: 'Videoanruf nicht verfügbar. Dein Gesprächspartner unterstützt keine Videotelefonie.',
+      Start_video_call: 'Starte Videoanruf'
+   });
+
+   $.extend(jsxc.l10n.es, {
+      Please_allow_access_to_microphone_and_camera: 'Por favor, permitir el acceso al micrófono y la cámara.',
+      Incoming_call: 'Llamada entrante',
+      from: 'de',
+      Do_you_want_to_accept_the_call_from: 'Desea aceptar la llamada de',
+      Reject: 'Rechazar',
+      Accept: 'Aceptar',
+      hang_up: 'colgar',
+      snapshot: 'instantánea',
+      mute_my_audio: 'silenciar mi audio',
+      pause_my_video: 'pausar mi vídeo',
+      fullscreen: 'pantalla completa',
+      Info: 'Info',
+      Local_IP: 'IP local',
+      Remote_IP: 'IP remota',
+      Local_Fingerprint: 'Firma digital local',
+      Remote_Fingerprint: 'Firma digital remota',
+      Video_call_not_possible: 'Llamada de vídeo no es posible',
+      Start_video_call: 'Iniciar llamada de vídeo'
+   });
+}(jQuery));
diff --git a/jsxc.min.js b/jsxc.min.js
new file mode 100644
index 0000000..d452788
--- /dev/null
+++ b/jsxc.min.js
@@ -0,0 +1,30 @@
+/*! This file is concatenated for the browser. */
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org> <br>
+ * Released under the MIT license
+ * 
+ * Please see http://www.jsxc.org/
+ * 
+ * @author Klaus Herberth <klaus at jsxc.org>
+ * @version 1.0.0
+ * @license MIT
+ */
+var jsxc;!function($){"use strict";jsxc={version:"1.0.0",master:!1,role_allocation:!1,to:null,toBusy:null,toNotification:null,toNotificationDelay:500,keepalive:null,restore:!1,restoreCompleted:!1,triggeredFromForm:!1,triggeredFromBox:!1,triggeredFromElement:!1,triggeredFromLogout:!1,ls:[],storageNotConform:null,toSNC:null,bid:null,CONST:{NOTIFICATION_DEFAULT:"default",NOTIFICATION_GRANTED:"granted",NOTIFICATION_DENIED:"denied",STATUS:["offline","dnd","xa","away","chat","online"],SOUNDS:{ [...]
+options.hide(),name=name.replaceWith(input),input.val(name.text()),input.keypress(function(ev){13===ev.which&&(options.show(),input.replaceWith(name),jsxc.gui.roster._rename(bid,$(this).val()),$("html").off("click"))}),input.click(function(){return!1}),$("html").one("click",function(){options.show(),input.replaceWith(name),jsxc.gui.roster._rename(bid,input.val())})},_rename:function(bid,newname){if(jsxc.master){var d=jsxc.storage.getUserItem("buddy",bid),iq=$iq({type:"set"}).c("query",{x [...]
+},onRidChange:function(ev,data){jsxc.storage.setItem("rid",data.rid)},resFriendReq:function(from,approve){jsxc.master?(jsxc.xmpp.conn.send($pres({to:from,type:approve?"subscribed":"unsubscribed"})),jsxc.storage.removeUserItem("friendReq"),jsxc.gui.dialog.close()):jsxc.storage.updateUserItem("friendReq","approve",approve)},addBuddy:function(username,alias){var bid=jsxc.jidToBid(username);if(jsxc.master){var iq=$iq({type:"set"}).c("query",{xmlns:"jabber:iq:roster"}).c("item",{jid:username, [...]
+}(jQuery);/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * Copyright (c) 2014 Klaus Herberth <klaus at jsxc.org> <br>
+ * Released under the MIT license
+ * 
+ * Please see http://www.jsxc.org/
+ * 
+ * @author Klaus Herberth <klaus at jsxc.org>
+ * @version 1.0.0
+ * @license MIT
+ */
+var RTC=null,RTCPeerconnection=null;jsxc.gui.template.incomingCall='<h3>%%Incoming_call%%</h3>        <p>%%Do_you_want_to_accept_the_call_from%% {{bid_name}}?</p>        <p class="jsxc_right">            <a href="#" class="button jsxc_reject">%%Reject%%</a> <a href="#" class="button creation jsxc_accept">%%Accept%%</a>         </p>',jsxc.gui.template.allowMediaAccess="<p>%%Please_allow_access_to_microphone_and_camera%%</p>",jsxc.gui.template.videoWindow='<div class="jsxc_webrtc">         [...]
+//# sourceMappingURL=jsxc.min.js.map
\ No newline at end of file
diff --git a/jsxc.min.js.map b/jsxc.min.js.map
new file mode 100644
index 0000000..aa4fdf8
--- /dev/null
+++ b/jsxc.min.js.map
@@ -0,0 +1 @@
+{"version":3,"file":"jsxc.min.js","sources":["jsxc.js"],"names":["jsxc","$","version","master","role_allocation","to","toBusy","toNotification","toNotificationDelay","keepalive","restore","restoreCompleted","triggeredFromForm","triggeredFromBox","triggeredFromElement","triggeredFromLogout","ls","storageNotConform","toSNC","bid","CONST","NOTIFICATION_DEFAULT","NOTIFICATION_GRANTED","NOTIFICATION_DENIED","STATUS","SOUNDS","MSG","CALL","NOTICE","REGEX","JID","RegExp","URL","debug","msg","da [...]
\ No newline at end of file
diff --git a/jsxc.webrtc.css b/jsxc.webrtc.css
new file mode 100644
index 0000000..63e0243
--- /dev/null
+++ b/jsxc.webrtc.css
@@ -0,0 +1,379 @@
+.jsxc_webrtc {
+	width: 900px;
+}
+
+.jsxc_webrtc li .jsxc_name {
+	cursor: auto;
+}
+
+.jsxc_webrtc li .jsxc_name:hover {
+	color: #939393;
+}
+
+.jsxc_videoContainer {
+	width: 640px;
+	height: 480px;
+	position: relative;
+	background-color: #a4a4a4;
+}
+
+.jsxc_videoContainer video, .jsxc_videoContainer .jsxc_noRemoteVideo {
+	display: none;
+}
+
+.jsxc_remotevideo, .jsxc_noRemoteVideo {
+	width: 100%;
+	height: 100%;
+	position: absolute;
+	top: 0px;
+	left: 0px;
+	z-index: 9000;
+	background-color: #a4a4a4;
+}
+
+.jsxc_localvideo {
+	width: 160px;
+	height: 120px;
+	position: absolute;
+	right: 10px;
+	bottom: 10px;
+	z-index: 9990;
+	background-color: black;
+	cursor: pointer;
+}
+
+.jsxc_noRemoteVideo p {
+	position: absolute;
+	bottom: 0px;
+	left:0px;
+	text-align: center;
+	color: #fff;
+	width: 100%;
+	z-index: 100;
+}
+
+.jsxc_noRemoteVideo > div {
+	width: 200px;
+	height: 200px;
+	overflow: hidden;
+	position: absolute;
+	top: 50%;
+	margin-top: -100px;
+	left: 50%;
+	margin-left: -100px;
+}
+
+.jsxc_noRemoteVideo > div > div {
+	background-color: #717171;
+}
+
+.jsxc_noRemoteVideo > div > div:first-child {
+	width: 50%;
+	height: 50%;
+	border-radius: 50%;
+	position: absolute;
+	left: 50%;
+	margin-left: -25%;
+	top: 10%;
+}
+
+.jsxc_noRemoteVideo > div > div:last-child {
+   position: absolute;
+	bottom: -50%;
+	width: 100%;
+	height: 100%;
+	border-radius: 50%;
+}
+
+.jsxc_webrtc .jsxc_status {
+	z-index: 9999;
+	border-radius: 20px;
+	display: none;
+	position: absolute;
+	top: 30px;
+	left: 50%;
+	background-color: lightgrey;
+	background-color: rgba(0, 0, 0, 0.5);
+	color: white;
+	padding: 15px;
+	font-weight: bold;
+	text-align: center;
+}
+
+.jsxc_webrtc .slimScrollDiv {
+	opacity: 1 !important;
+}
+
+div:-moz-full-screen {
+	width: 100%;
+	height: 100%;
+	background-color: black;
+}
+
+div:-ms-full-screen {
+	width: 100%;
+	height: 100%;
+	background-color: black;
+}
+
+div:-webkit-full-screen {
+	width: 100%;
+	height: 100%;
+	background-color: black;
+}
+
+div:fullscreen {
+	width: 100%;
+	height: 100%;
+	background-color: black;
+}
+
+div:-moz-full-screen.jsxc_localvideo {
+	border: 1px solid white;
+}
+
+div:-webkit-full-screen.jsxc_localvideo {
+	border: 1px solid white;
+}
+
+div:-ms-full-screen.jsxc_localvideo {
+	border: 1px solid white;
+}
+
+div:fullscreen.jsxc_localvideo {
+	border: 1px solid white;
+}
+
+.jsxc_fullscreen.jsxc_localvideo {
+	border: 1px solid white;
+}
+
+.jsxc_controlbar {
+	width: 640px;
+}
+
+.jsxc_controlbar .jsxc_showchat {
+	float: right;
+}
+
+.jsxc_multi>div {
+	display: none;
+}
+
+.jsxc_snapshotbar {
+	width: 100%;
+	display: none;
+}
+
+.jsxc_snapshotbar img {
+	height: 50px;
+}
+
+.jsxc_buttongroup {
+	display: inline;
+}
+
+.jsxc_buttongroup button:first-child {
+	border-top-right-radius: 0px;
+	border-bottom-right-radius: 0px;
+	margin-right: 0px;
+}
+
+.jsxc_buttongroup button:last-child {
+	border-top-left-radius: 0px;
+	border-bottom-left-radius: 0px;
+	margin-left: 0px;
+}
+
+.jsxc_chatarea {
+	position: relative;
+	float:right;
+	height: 480px;
+	width: 250px;
+	background-color: #A4A4A4;
+}
+
+.jsxc_chatarea > ul {
+	width: 250px;
+	height: 480px;
+	list-style: none;
+	padding: 0px;
+}
+
+.jsxc_chatarea .jsxc_settings {
+	display: none !important;
+}
+
+.jsxc_chatarea .jsxc_close {
+	display: none !important;
+}
+
+.jsxc_chatarea .jsxc_video {
+   display: none !important;
+}
+
+.jsxc_chatarea .jsxc_bar {
+
+}
+
+.jsxc_chatarea .jsxc_window {
+	bottom: 0px;
+	box-shadow: none;
+}
+
+.jsxc_videoSuitable .jsxc_name {
+	font-style: italic;
+}
+
+div.jsxc_video {
+   background-image: url('img/camera_icon_grey.svg');
+   background-repeat: no-repeat;
+   background-position: center center;
+   background-size: 15px 15px;
+}
+
+div.jsxc_video:not(.jsxc_disabled):hover, #jsxc_buddylist .jsxc_options .jsxc_video {
+    background-image: url('img/camera_icon_white.svg');
+}
+
+#jsxc_buddylist .jsxc_options .jsxc_video.jsxc_disabled {
+	opacity: 0.2;
+}
+
+.bubblingG {
+text-align: center;
+width:129px;
+height:80px;
+position: absolute;
+top: 40%;
+left: 50%;
+margin-left: -64px;
+}
+
+.bubblingG span {
+display: inline-block;
+vertical-align: middle;
+width: 16px;
+height: 16px;
+margin: 40px auto;
+background: #000000;
+-moz-border-radius: 81px;
+-moz-animation: bubblingG 1.3s infinite alternate;
+-webkit-border-radius: 81px;
+-webkit-animation: bubblingG 1.3s infinite alternate;
+-ms-border-radius: 81px;
+-ms-animation: bubblingG 1.3s infinite alternate;
+-o-border-radius: 81px;
+-o-animation: bubblingG 1.3s infinite alternate;
+border-radius: 81px;
+animation: bubblingG 1.3s infinite alternate;
+}
+
+#bubblingG_1 {
+-moz-animation-delay: 0s;
+-webkit-animation-delay: 0s;
+-ms-animation-delay: 0s;
+-o-animation-delay: 0s;
+animation-delay: 0s;
+}
+
+#bubblingG_2 {
+-moz-animation-delay: 0.39s;
+-webkit-animation-delay: 0.39s;
+-ms-animation-delay: 0.39s;
+-o-animation-delay: 0.39s;
+animation-delay: 0.39s;
+}
+
+#bubblingG_3 {
+-moz-animation-delay: 0.78s;
+-webkit-animation-delay: 0.78s;
+-ms-animation-delay: 0.78s;
+-o-animation-delay: 0.78s;
+animation-delay: 0.78s;
+}
+
+ at -moz-keyframes bubblingG {
+0% {
+width: 16px;
+height: 16px;
+background-color:#000000;
+-moz-transform: translateY(0);
+}
+
+100% {
+width: 39px;
+height: 39px;
+background-color:#FFFFFF;
+-moz-transform: translateY(-34px);
+}
+
+}
+
+ at -webkit-keyframes bubblingG {
+0% {
+width: 16px;
+height: 16px;
+background-color:#000000;
+-webkit-transform: translateY(0);
+}
+
+100% {
+width: 39px;
+height: 39px;
+background-color:#FFFFFF;
+-webkit-transform: translateY(-34px);
+}
+
+}
+
+ at -ms-keyframes bubblingG {
+0% {
+width: 16px;
+height: 16px;
+background-color:#000000;
+-ms-transform: translateY(0);
+}
+
+100% {
+width: 39px;
+height: 39px;
+background-color:#FFFFFF;
+-ms-transform: translateY(-34px);
+}
+
+}
+
+ at -o-keyframes bubblingG {
+0% {
+width: 16px;
+height: 16px;
+background-color:#000000;
+-o-transform: translateY(0);
+}
+
+100% {
+width: 39px;
+height: 39px;
+background-color:#FFFFFF;
+-o-transform: translateY(-34px);
+}
+
+}
+
+ at keyframes bubblingG {
+0% {
+width: 16px;
+height: 16px;
+background-color:#000000;
+transform: translateY(0);
+}
+
+100% {
+width: 39px;
+height: 39px;
+background-color:#FFFFFF;
+transform: translateY(-34px);
+}
+
+}
diff --git a/lib/dsa-ww.js b/lib/dsa-ww.js
new file mode 100644
index 0000000..fe9dd51
--- /dev/null
+++ b/lib/dsa-ww.js
@@ -0,0 +1,50 @@
+;
+(function(root) {
+    "use strict";
+
+    root.OTR = {}
+    root.crypto = {}
+    root.DSA = {};
+
+    // default imports
+    var imports = [
+        'build/dep/salsa20.js',
+        'build/dep/bigint.js',
+        'build/dep/crypto.js',
+        'build/dep/eventemitter.js',
+        'lib/const.js',
+        'lib/helpers.js',
+        'lib/dsa.js',
+    ]
+
+    function sendMsg(type, data) {
+        postMessage({
+            type: type,
+            data: data,
+        })
+    }
+
+    self.onmessage = function(e) {
+        var data = e.data;
+
+        root.crypto = {
+            getRandomValues: function() {
+                var buf = data.random;
+            }
+        };
+
+        if (data.imports)
+            imports = data.imports
+
+        importScripts.apply(self, imports);
+
+        sendMsg('debug', 'DSA key creation started')
+
+        var dsa = new DSA()
+
+        sendMsg('debug', 'DSA key creation finished')
+
+        sendMsg('data', {key: dsa.packPrivate()})
+    }
+
+}(this))
\ No newline at end of file
diff --git a/lib/jquery.colorbox-min.js b/lib/jquery.colorbox-min.js
new file mode 100644
index 0000000..48f1eba
--- /dev/null
+++ b/lib/jquery.colorbox-min.js
@@ -0,0 +1,7 @@
+/*!
+   Colorbox v1.4.33 - 2013-10-31
+   jQuery lightbox and modal window plugin
+   (c) 2013 Jack Moore - http://www.jacklmoore.com/colorbox
+   license: http://www.opensource.org/licenses/mit-license.php
+*/
+(function(e,t,i){function o(i,o,n){var r=t.createElement(i);return o&&(r.id=Z+o),n&&(r.style.cssText=n),e(r)}function n(){return i.innerHeight?i.innerHeight:e(i).height()}function r(e){var t=k.length,i=(z+e)%t;return 0>i?t+i:i}function h(e,t){return Math.round((/%/.test(e)?("x"===t?E.width():n())/100:1)*parseInt(e,10))}function l(e,t){return e.photo||e.photoRegex.test(t)}function s(e,t){return e.retinaUrl&&i.devicePixelRatio>1?t.replace(e.photoRegex,e.retinaSuffix):t}function a(e){"conta [...]
\ No newline at end of file
diff --git a/lib/jquery.fullscreen.js b/lib/jquery.fullscreen.js
new file mode 100644
index 0000000..bab35b8
--- /dev/null
+++ b/lib/jquery.fullscreen.js
@@ -0,0 +1,88 @@
+/**
+ * @name        jQuery Fullscreen Plugin
+ * @author      Klaus Herberth, Martin Angelov, Morten Sjøgren
+ * @url         http://tutorialzine.com/2012/02/enhance-your-website-fullscreen-api/
+ * @license     MIT License
+ */
+
+/*jshint browser: true, jquery: true */
+(function($) {
+    "use strict";
+
+    // These helper functions available only to our plugin scope.
+    function supportFullscreen() {
+        var doc = document.documentElement;
+
+        return ('requestFullscreen' in doc) ||
+                ('mozRequestFullScreen' in doc && document.mozFullScreenEnabled) ||
+                ('webkitRequestFullscreen' in doc);
+    }
+
+    function requestFullscreen(elem) {
+        if (elem.requestFullscreen) {
+            elem.requestFullscreen();
+        } else if (elem.mozRequestFullScreen) {
+            elem.mozRequestFullScreen();
+        } else if (elem.webkitRequestFullscreen) {
+            elem.webkitRequestFullscreen(Element.ALLOW_KEYBOARD_INPUT);
+        }
+    }
+
+    function fullscreenStatus() {
+        return document.fullscreen ||
+                document.mozFullScreen ||
+                document.webkitIsFullScreen ||
+                false;
+    }
+
+    function cancelFullscreen() {
+        if (document.exitFullscreen) {
+            document.exitFullscreen();
+        } else if (document.mozCancelFullScreen) {
+            document.mozCancelFullScreen();
+        } else if (document.webkitCancelFullScreen) {
+            document.webkitCancelFullScreen();
+        }
+    }
+
+    // Adding a new test to the jQuery support object
+    $.support.fullscreen = supportFullscreen();
+
+    // Creating the plugin
+    $.fn.fullscreen = function() {
+        if (!$.support.fullscreen || this.length !== 1)
+            return this;
+
+        if (fullscreenStatus()) {
+            // if we are already in fullscreen, exit
+            cancelFullscreen();
+            return this;
+        }
+        
+        var self = this;
+
+        // Chrome trigger event on self, Firefox on document
+        $(self).add(document).on('fullscreenerror mozfullscreenerror webkitfullscreenerror msfullscreenerror', function() {
+            $(document).trigger('error.fullscreen');
+        });
+
+        $(self).add(document).on('fullscreenchange mozfullscreenchange webkitfullscreenchange msfullscreenchange', function() {
+            if (fullscreenStatus()){ 
+                $(document).trigger('enabled.fullscreen');
+            }else{
+                $(document).trigger('disabled.fullscreen');
+                $(self).add(document).off('fullscreenchange mozfullscreenchange webkitfullscreenchange msfullscreenchange');
+            }
+        });
+
+        requestFullscreen($(self).get(0));
+
+        return $(self);
+    };
+
+    $.fn.cancelFullscreen = function( ) {
+        cancelFullscreen();
+
+        return this;
+    };
+}(jQuery));
\ No newline at end of file
diff --git a/lib/jquery.min.js b/lib/jquery.min.js
new file mode 100644
index 0000000..e5ace11
--- /dev/null
+++ b/lib/jquery.min.js
@@ -0,0 +1,4 @@
+/*! jQuery v2.1.1 | (c) 2005, 2014 jQuery Foundation, Inc. | jquery.org/license */
+!function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=c.slice,e=c.concat,f=c.push,g=c.indexOf,h={},i=h.toString,j=h.hasOwnProperty,k={},l=a.document,m="2.1.1",n=function(a,b){return new n.fn.init(a,b)},o=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,p=/^-ms-/,q=/-([\da-z])/gi,r= [...]
+},_data:function(a,b,c){return L.access(a,b,c)},_removeData:function(a,b){L.remove(a,b)}}),n.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=M.get(f),1===f.nodeType&&!L.get(f,"hasDataAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=n.camelCase(d.slice(5)),P(f,d,e[d])));L.set(f,"hasDataAttrs",!0)}return e}return"object"==typeof a?this.each(function(){M.set(this,a)}):J(this,function(b){var c,d=n.camelCase(a); [...]
+},removeAttr:function(a,b){var c,d,e=0,f=b&&b.match(E);if(f&&1===a.nodeType)while(c=f[e++])d=n.propFix[c]||c,n.expr.match.bool.test(c)&&(a[d]=!1),a.removeAttribute(c)},attrHooks:{type:{set:function(a,b){if(!k.radioValue&&"radio"===b&&n.nodeName(a,"input")){var c=a.value;return a.setAttribute("type",b),c&&(a.value=c),b}}}}}),Zb={set:function(a,b,c){return b===!1?n.removeAttr(a,c):a.setAttribute(c,c),c}},n.each(n.expr.match.bool.source.match(/\w+/g),function(a,b){var c=$b[b]||n.find.attr;$ [...]
diff --git a/lib/jquery.slimscroll.js b/lib/jquery.slimscroll.js
new file mode 100644
index 0000000..ccd19cd
--- /dev/null
+++ b/lib/jquery.slimscroll.js
@@ -0,0 +1,474 @@
+/*! Copyright (c) 2011 Piotr Rochala (http://rocha.la)
+ * Dual licensed under the MIT (http://www.opensource.org/licenses/mit-license.php)
+ * and GPL (http://www.opensource.org/licenses/gpl-license.php) licenses.
+ *
+ * Version: 1.3.3 + pr#94
+ *
+ */
+(function($) {
+
+  $.fn.extend({
+    slimScroll: function(options) {
+
+      var defaults = {
+
+        // width in pixels of the visible scroll area
+        width : 'auto',
+
+        // height in pixels of the visible scroll area
+        height : '250px',
+
+        // width in pixels of the scrollbar and rail
+        size : '7px',
+
+        // scrollbar color, accepts any hex/color value
+        color: '#000',
+
+        // scrollbar position - left/right
+        position : 'right',
+
+        // distance in pixels between the side edge and the scrollbar
+        distance : '1px',
+
+        // default scroll position on load - top / bottom / $('selector')
+        start : 'top',
+
+        // sets scrollbar opacity
+        opacity : .4,
+
+        // enables always-on mode for the scrollbar
+        alwaysVisible : false,
+
+        // check if we should hide the scrollbar when user is hovering over
+        disableFadeOut : false,
+
+        // sets visibility of the rail
+        railVisible : false,
+
+        // sets rail color
+        railColor : '#333',
+
+        // sets rail opacity
+        railOpacity : .2,
+
+        // whether  we should use jQuery UI Draggable to enable bar dragging
+        railDraggable : true,
+
+        // defautlt CSS class of the slimscroll rail
+        railClass : 'slimScrollRail',
+
+        // defautlt CSS class of the slimscroll bar
+        barClass : 'slimScrollBar',
+
+        // defautlt CSS class of the slimscroll wrapper
+        wrapperClass : 'slimScrollDiv',
+
+        // check if mousewheel should scroll the window if we reach top/bottom
+        allowPageScroll : false,
+
+        // scroll amount applied to each mouse wheel step
+        wheelStep : 20,
+
+        // scroll amount applied when user is using gestures
+        touchScrollStep : 200,
+
+        // sets border radius
+        borderRadius: '7px',
+
+        // sets border radius of the rail
+        railBorderRadius : '7px'
+      };
+
+      var o = $.extend(defaults, options);
+
+      // do it for every element that matches selector
+      this.each(function(){
+
+      var isOverPanel, isOverBar, isDragg, queueHide, touchDif,
+        barHeight, percentScroll, lastScroll,
+        divS = '<div></div>',
+        minBarHeight = 30,
+        releaseScroll = false;
+
+        // used in event handlers and for better minification
+        var me = $(this);
+
+        // ensure we are not binding it again
+        if (me.parent().hasClass(o.wrapperClass))
+        {
+            // start from last bar position
+            var offset = me.scrollTop();
+
+            // find bar and rail
+            bar = me.parent().find('.' + o.barClass);
+            rail = me.parent().find('.' + o.railClass);
+
+            getBarHeight();
+
+            // check if we should scroll existing instance
+            if ($.isPlainObject(options))
+            {
+              // Pass height: auto to an existing slimscroll object to force a resize after contents have changed
+              if ( 'height' in options && options.height == 'auto' ) {
+                me.parent().css('height', 'auto');
+                me.css('height', 'auto');
+                var height = me.parent().parent().height();
+                me.parent().css('height', height);
+                me.css('height', height);
+              } else if ('height' in options) {
+                var h = options.height;
+                me.parent().css('height', h);
+                me.css('height', h);
+              }
+
+              if ('scrollTo' in options)
+              {
+                // jump to a static point
+                offset = parseInt(o.scrollTo);
+              }
+              else if ('scrollBy' in options)
+              {
+                // jump by value pixels
+                offset += parseInt(o.scrollBy);
+              }
+              else if ('destroy' in options)
+              {
+                // remove slimscroll elements
+                bar.remove();
+                rail.remove();
+                me.unwrap();
+                return;
+              }
+
+              // scroll content by the given offset
+              scrollContent(offset, false, true);
+            }
+
+            return;
+        }
+        else if ($.isPlainObject(options))
+        {
+            if ('destroy' in options)
+            {
+               return;
+            }
+        }
+
+        // optionally set height to the parent's height
+        o.height = (o.height == 'auto') ? me.parent().height() : o.height;
+
+        // wrap content
+        var wrapper = $(divS)
+          .addClass(o.wrapperClass)
+          .css({
+            position: 'relative',
+            overflow: 'hidden',
+            width: o.width,
+            height: o.height
+          });
+
+        // update style for the div
+        me.css({
+          overflow: 'hidden',
+          width: o.width,
+          height: o.height
+        });
+
+        // create scrollbar rail
+        var rail = $(divS)
+          .addClass(o.railClass)
+          .css({
+            width: o.size,
+            height: '100%',
+            position: 'absolute',
+            top: 0,
+            display: (o.alwaysVisible && o.railVisible) ? 'block' : 'none',
+            'border-radius': o.railBorderRadius,
+            background: o.railColor,
+            opacity: o.railOpacity,
+            zIndex: 90
+          });
+
+        // create scrollbar
+        var bar = $(divS)
+          .addClass(o.barClass)
+          .css({
+            background: o.color,
+            width: o.size,
+            position: 'absolute',
+            top: 0,
+            opacity: o.opacity,
+            display: o.alwaysVisible ? 'block' : 'none',
+            'border-radius' : o.borderRadius,
+            BorderRadius: o.borderRadius,
+            MozBorderRadius: o.borderRadius,
+            WebkitBorderRadius: o.borderRadius,
+            zIndex: 99
+          });
+
+        // set position
+        var posCss = (o.position == 'right') ? { right: o.distance } : { left: o.distance };
+        rail.css(posCss);
+        bar.css(posCss);
+
+        // wrap it
+        me.wrap(wrapper);
+
+        // append to parent div
+        me.parent().append(bar);
+        me.parent().append(rail);
+
+        // make it draggable and no longer dependent on the jqueryUI
+        if (o.railDraggable){
+          bar.bind("mousedown", function(e) {
+            var $doc = $(document);
+            isDragg = true;
+            t = parseFloat(bar.css('top'));
+            pageY = e.pageY;
+
+            $doc.bind("mousemove.slimscroll", function(e){
+              currTop = t + e.pageY - pageY;
+              bar.css('top', currTop);
+              scrollContent(0, bar.position().top, false);// scroll content
+            });
+
+            $doc.bind("mouseup.slimscroll", function(e) {
+              isDragg = false;hideBar();
+              $doc.unbind('.slimscroll');
+            });
+            return false;
+          }).bind("selectstart.slimscroll", function(e){
+            e.stopPropagation();
+            e.preventDefault();
+            return false;
+          });
+        }
+
+        // on rail over
+        rail.hover(function(){
+          showBar();
+        }, function(){
+          hideBar();
+        });
+
+        // on bar over
+        bar.hover(function(){
+          isOverBar = true;
+        }, function(){
+          isOverBar = false;
+        });
+
+        // show on parent mouseover
+        me.hover(function(){
+          isOverPanel = true;
+          showBar();
+          hideBar();
+        }, function(){
+          isOverPanel = false;
+          hideBar();
+        });
+
+        // support for mobile
+        me.bind('touchstart', function(e,b){
+          if (e.originalEvent.touches.length)
+          {
+            // record where touch started
+            touchDif = e.originalEvent.touches[0].pageY;
+          }
+        });
+
+        me.bind('touchmove', function(e){
+          // prevent scrolling the page if necessary
+          if(!releaseScroll)
+          {
+            e.originalEvent.preventDefault();
+            }
+          if (e.originalEvent.touches.length)
+          {
+            // see how far user swiped
+            var diff = (touchDif - e.originalEvent.touches[0].pageY) / o.touchScrollStep;
+            // scroll content
+            scrollContent(diff, true);
+            touchDif = e.originalEvent.touches[0].pageY;
+          }
+        });
+
+        // set up initial height
+        getBarHeight();
+
+        // check start position
+        if (o.start === 'bottom')
+        {
+          // scroll content to bottom
+          bar.css({ top: me.outerHeight() - bar.outerHeight() });
+          scrollContent(0, true);
+        }
+        else if (o.start !== 'top')
+        {
+          // assume jQuery selector
+          scrollContent($(o.start).position().top, null, true);
+
+          // make sure bar stays hidden
+          if (!o.alwaysVisible) { bar.hide(); }
+        }
+
+        // attach scroll events
+        attachWheel();
+
+        function _onWheel(e)
+        {
+          // use mouse wheel only when mouse is over
+          if (!isOverPanel) { return; }
+
+          var e = e || window.event;
+
+          var delta = 0;
+          if (e.wheelDelta) { delta = -e.wheelDelta/120; }
+          if (e.detail) { delta = e.detail / 3; }
+
+          var target = e.target || e.srcTarget || e.srcElement;
+          if ($(target).closest('.' + o.wrapperClass).is(me.parent())) {
+            // scroll content
+            scrollContent(delta, true);
+          }
+
+          // stop window scroll
+          if (e.preventDefault && !releaseScroll) { e.preventDefault(); }
+          if (!releaseScroll) { e.returnValue = false; }
+        }
+
+        function scrollContent(y, isWheel, isJump)
+        {
+          releaseScroll = false;
+          var delta = y;
+          var maxTop = me.outerHeight() - bar.outerHeight();
+
+          if (isWheel)
+          {
+            // move bar with mouse wheel
+            delta = parseInt(bar.css('top')) + y * parseInt(o.wheelStep) / 100 * bar.outerHeight();
+
+            // move bar, make sure it doesn't go out
+            delta = Math.min(Math.max(delta, 0), maxTop);
+
+            // if scrolling down, make sure a fractional change to the
+            // scroll position isn't rounded away when the scrollbar's CSS is set
+            // this flooring of delta would happened automatically when
+            // bar.css is set below, but we floor here for clarity
+            delta = (y > 0) ? Math.ceil(delta) : Math.floor(delta);
+
+            // scroll the scrollbar
+            bar.css({ top: delta + 'px' });
+          }
+
+          // calculate actual scroll amount
+          percentScroll = parseInt(bar.css('top')) / (me.outerHeight() - bar.outerHeight());
+          delta = percentScroll * (me[0].scrollHeight - me.outerHeight());
+
+          if (isJump)
+          {
+            delta = y;
+            var offsetTop = delta / me[0].scrollHeight * me.outerHeight();
+            offsetTop = Math.min(Math.max(offsetTop, 0), maxTop);
+            bar.css({ top: offsetTop + 'px' });
+          }
+
+          // scroll content
+          me.scrollTop(delta);
+
+          // fire scrolling event
+          me.trigger('slimscrolling', ~~delta);
+
+          // ensure bar is visible
+          showBar();
+
+          // trigger hide when scroll is stopped
+          hideBar();
+        }
+
+        function attachWheel()
+        {
+          if (window.addEventListener)
+          {
+            this.addEventListener('DOMMouseScroll', _onWheel, false );
+            this.addEventListener('mousewheel', _onWheel, false );
+          }
+          else
+          {
+            document.attachEvent("onmousewheel", _onWheel)
+          }
+        }
+
+        function getBarHeight()
+        {
+          // calculate scrollbar height and make sure it is not too small
+          barHeight = Math.max((me.outerHeight() / me[0].scrollHeight) * me.outerHeight(), minBarHeight);
+          bar.css({ height: barHeight + 'px' });
+
+          // hide scrollbar if content is not long enough
+          var display = barHeight == me.outerHeight() ? 'none' : 'block';
+          bar.css({ display: display });
+        }
+
+        function showBar()
+        {
+          // recalculate bar height
+          getBarHeight();
+          clearTimeout(queueHide);
+
+          // when bar reached top or bottom
+          if (percentScroll == ~~percentScroll)
+          {
+            //release wheel
+            releaseScroll = o.allowPageScroll;
+
+            // publish approporiate event
+            if (lastScroll != percentScroll)
+            {
+                var msg = (~~percentScroll == 0) ? 'top' : 'bottom';
+                me.trigger('slimscroll', msg);
+            }
+          }
+          else
+          {
+            releaseScroll = false;
+          }
+          lastScroll = percentScroll;
+
+          // show only when required
+          if(barHeight >= me.outerHeight()) {
+            //allow window scroll
+            releaseScroll = true;
+            return;
+          }
+          bar.stop(true,true).fadeIn('fast');
+          if (o.railVisible) { rail.stop(true,true).fadeIn('fast'); }
+        }
+
+        function hideBar()
+        {
+          // only hide when options allow it
+          if (!o.alwaysVisible)
+          {
+            queueHide = setTimeout(function(){
+              if (!(o.disableFadeOut && isOverPanel) && !isOverBar && !isDragg)
+              {
+                bar.fadeOut('slow');
+                rail.fadeOut('slow');
+              }
+            }, 1000);
+          }
+        }
+
+      });
+
+      // maintain chainability
+      return this;
+    }
+  });
+
+  $.fn.extend({
+    slimscroll: $.fn.slimScroll
+  });
+
+})(jQuery);
\ No newline at end of file
diff --git a/lib/jquery.ui.min.js b/lib/jquery.ui.min.js
new file mode 100644
index 0000000..f26f027
--- /dev/null
+++ b/lib/jquery.ui.min.js
@@ -0,0 +1,7 @@
+/*! jQuery UI - v1.10.3 - 2013-10-20
+* http://jqueryui.com
+* Includes: jquery.ui.core.js, jquery.ui.widget.js, jquery.ui.mouse.js, jquery.ui.position.js, jquery.ui.draggable.js, jquery.ui.droppable.js, jquery.ui.resizable.js, jquery.ui.selectable.js, jquery.ui.sortable.js, jquery.ui.accordion.js, jquery.ui.autocomplete.js, jquery.ui.button.js, jquery.ui.datepicker.js, jquery.ui.dialog.js, jquery.ui.menu.js, jquery.ui.progressbar.js, jquery.ui.slider.js, jquery.ui.spinner.js, jquery.ui.tabs.js, jquery.ui.tooltip.js, jquery.ui.effect.js, jquery.ui [...]
+* Copyright 2013 jQuery Foundation and other contributors; Licensed MIT */
+
+(function(e,t){function i(t,i){var s,n,r,o=t.nodeName.toLowerCase();return"area"===o?(s=t.parentNode,n=s.name,t.href&&n&&"map"===s.nodeName.toLowerCase()?(r=e("img[usemap=#"+n+"]")[0],!!r&&a(r)):!1):(/input|select|textarea|button|object/.test(o)?!t.disabled:"a"===o?t.href||i:i)&&a(t)}function a(t){return e.expr.filters.visible(t)&&!e(t).parents().addBack().filter(function(){return"hidden"===e.css(this,"visibility")}).length}var s=0,n=/^ui-id-\d+$/;e.ui=e.ui||{},e.extend(e.ui,{version:"1. [...]
+return isNaN(t)?c:t},m=p(d[0]),f=Math.max(m,p(d[1]||"")),m=a?Math.max(m,a.getFullYear()):m,f=s?Math.min(f,s.getFullYear()):f,e.yearshtml+="<select class='ui-datepicker-year' data-handler='selectYear' data-event='change'>";f>=m;m++)e.yearshtml+="<option value='"+m+"'"+(m===i?" selected='selected'":"")+">"+m+"</option>";e.yearshtml+="</select>",b+=e.yearshtml,e.yearshtml=null}return b+=this._get(e,"yearSuffix"),y&&(b+=(!n&&g&&v?"":"&#xa0;")+_),b+="</div>"},_adjustInstDate:function(e,t,i){v [...]
\ No newline at end of file
diff --git a/lib/jsxc.dep.js b/lib/jsxc.dep.js
new file mode 100644
index 0000000..c81e8c4
--- /dev/null
+++ b/lib/jsxc.dep.js
@@ -0,0 +1,16644 @@
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * This file concatenates all dependencies of jsxc.
+ * 
+ */
+
+/*!
+ * Source: build/lib/strophe.js, license: multiple, url: http://strophe.im/strophejs/ */
+/**
+ * Modified by
+ * Klaus Herberth, 2014
+ */
+
+/*! This code was written by Tyler Akins and has been placed in the
+   public domain.  It would be nice if you left this header intact.
+   Base64 code from Tyler Akins -- http://rumkin.com
+*/
+
+var Base64 = (function () {
+    var keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
+
+    var obj = {
+        /**
+         * Encodes a string in base64
+         * @param {String} input The string to encode in base64.
+         */
+        encode: function (input) {
+            var output = "";
+            var chr1, chr2, chr3;
+            var enc1, enc2, enc3, enc4;
+            var i = 0;
+
+            do {
+                chr1 = input.charCodeAt(i++);
+                chr2 = input.charCodeAt(i++);
+                chr3 = input.charCodeAt(i++);
+
+                enc1 = chr1 >> 2;
+                enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
+                enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
+                enc4 = chr3 & 63;
+
+                if (isNaN(chr2)) {
+                    enc3 = enc4 = 64;
+                } else if (isNaN(chr3)) {
+                    enc4 = 64;
+                }
+
+                output = output + keyStr.charAt(enc1) + keyStr.charAt(enc2) +
+                    keyStr.charAt(enc3) + keyStr.charAt(enc4);
+            } while (i < input.length);
+
+            return output;
+        },
+
+        /**
+         * Decodes a base64 string.
+         * @param {String} input The string to decode.
+         */
+        decode: function (input) {
+            var output = "";
+            var chr1, chr2, chr3;
+            var enc1, enc2, enc3, enc4;
+            var i = 0;
+
+            // remove all characters that are not A-Z, a-z, 0-9, +, /, or =
+            input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
+
+            do {
+                enc1 = keyStr.indexOf(input.charAt(i++));
+                enc2 = keyStr.indexOf(input.charAt(i++));
+                enc3 = keyStr.indexOf(input.charAt(i++));
+                enc4 = keyStr.indexOf(input.charAt(i++));
+
+                chr1 = (enc1 << 2) | (enc2 >> 4);
+                chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
+                chr3 = ((enc3 & 3) << 6) | enc4;
+
+                output = output + String.fromCharCode(chr1);
+
+                if (enc3 != 64) {
+                    output = output + String.fromCharCode(chr2);
+                }
+                if (enc4 != 64) {
+                    output = output + String.fromCharCode(chr3);
+                }
+            } while (i < input.length);
+
+            return output;
+        }
+    };
+
+    return obj;
+})();
+
+/*!
+ * A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined
+ * in FIPS PUB 180-1
+ * Version 2.1a Copyright Paul Johnston 2000 - 2002.
+ * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+ * Distributed under the BSD License
+ * See http://pajhome.org.uk/crypt/md5 for details.
+ */
+
+/* Some functions and variables have been stripped for use with Strophe */
+
+/*
+ * These are the functions you'll usually want to call
+ * They take string arguments and return either hex or base-64 encoded strings
+ */
+function b64_sha1(s){return binb2b64(core_sha1(str2binb(s),s.length * 8));}
+function str_sha1(s){return binb2str(core_sha1(str2binb(s),s.length * 8));}
+function b64_hmac_sha1(key, data){ return binb2b64(core_hmac_sha1(key, data));}
+function str_hmac_sha1(key, data){ return binb2str(core_hmac_sha1(key, data));}
+
+/*
+ * Calculate the SHA-1 of an array of big-endian words, and a bit length
+ */
+function core_sha1(x, len)
+{
+  /* append padding */
+  x[len >> 5] |= 0x80 << (24 - len % 32);
+  x[((len + 64 >> 9) << 4) + 15] = len;
+
+  var w = new Array(80);
+  var a =  1732584193;
+  var b = -271733879;
+  var c = -1732584194;
+  var d =  271733878;
+  var e = -1009589776;
+
+  var i, j, t, olda, oldb, oldc, oldd, olde;
+  for (i = 0; i < x.length; i += 16)
+  {
+    olda = a;
+    oldb = b;
+    oldc = c;
+    oldd = d;
+    olde = e;
+
+    for (j = 0; j < 80; j++)
+    {
+      if (j < 16) { w[j] = x[i + j]; }
+      else { w[j] = rol(w[j-3] ^ w[j-8] ^ w[j-14] ^ w[j-16], 1); }
+      t = safe_add(safe_add(rol(a, 5), sha1_ft(j, b, c, d)),
+                       safe_add(safe_add(e, w[j]), sha1_kt(j)));
+      e = d;
+      d = c;
+      c = rol(b, 30);
+      b = a;
+      a = t;
+    }
+
+    a = safe_add(a, olda);
+    b = safe_add(b, oldb);
+    c = safe_add(c, oldc);
+    d = safe_add(d, oldd);
+    e = safe_add(e, olde);
+  }
+  return [a, b, c, d, e];
+}
+
+/*
+ * Perform the appropriate triplet combination function for the current
+ * iteration
+ */
+function sha1_ft(t, b, c, d)
+{
+  if (t < 20) { return (b & c) | ((~b) & d); }
+  if (t < 40) { return b ^ c ^ d; }
+  if (t < 60) { return (b & c) | (b & d) | (c & d); }
+  return b ^ c ^ d;
+}
+
+/*
+ * Determine the appropriate additive constant for the current iteration
+ */
+function sha1_kt(t)
+{
+  return (t < 20) ?  1518500249 : (t < 40) ?  1859775393 :
+         (t < 60) ? -1894007588 : -899497514;
+}
+
+/*
+ * Calculate the HMAC-SHA1 of a key and some data
+ */
+function core_hmac_sha1(key, data)
+{
+  var bkey = str2binb(key);
+  if (bkey.length > 16) { bkey = core_sha1(bkey, key.length * 8); }
+
+  var ipad = new Array(16), opad = new Array(16);
+  for (var i = 0; i < 16; i++)
+  {
+    ipad[i] = bkey[i] ^ 0x36363636;
+    opad[i] = bkey[i] ^ 0x5C5C5C5C;
+  }
+
+  var hash = core_sha1(ipad.concat(str2binb(data)), 512 + data.length * 8);
+  return core_sha1(opad.concat(hash), 512 + 160);
+}
+
+/*
+ * Add integers, wrapping at 2^32. This uses 16-bit operations internally
+ * to work around bugs in some JS interpreters.
+ */
+function safe_add(x, y)
+{
+  var lsw = (x & 0xFFFF) + (y & 0xFFFF);
+  var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
+  return (msw << 16) | (lsw & 0xFFFF);
+}
+
+/*
+ * Bitwise rotate a 32-bit number to the left.
+ */
+function rol(num, cnt)
+{
+  return (num << cnt) | (num >>> (32 - cnt));
+}
+
+/*
+ * Convert an 8-bit or 16-bit string to an array of big-endian words
+ * In 8-bit function, characters >255 have their hi-byte silently ignored.
+ */
+function str2binb(str)
+{
+  var bin = [];
+  var mask = 255;
+  for (var i = 0; i < str.length * 8; i += 8)
+  {
+    bin[i>>5] |= (str.charCodeAt(i / 8) & mask) << (24 - i%32);
+  }
+  return bin;
+}
+
+/*
+ * Convert an array of big-endian words to a string
+ */
+function binb2str(bin)
+{
+  var str = "";
+  var mask = 255;
+  for (var i = 0; i < bin.length * 32; i += 8)
+  {
+    str += String.fromCharCode((bin[i>>5] >>> (24 - i%32)) & mask);
+  }
+  return str;
+}
+
+/*
+ * Convert an array of big-endian words to a base-64 string
+ */
+function binb2b64(binarray)
+{
+  var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
+  var str = "";
+  var triplet, j;
+  for (var i = 0; i < binarray.length * 4; i += 3)
+  {
+    triplet = (((binarray[i   >> 2] >> 8 * (3 -  i   %4)) & 0xFF) << 16) |
+              (((binarray[i+1 >> 2] >> 8 * (3 - (i+1)%4)) & 0xFF) << 8 ) |
+               ((binarray[i+2 >> 2] >> 8 * (3 - (i+2)%4)) & 0xFF);
+    for (j = 0; j < 4; j++)
+    {
+      if (i * 8 + j * 6 > binarray.length * 32) { str += "="; }
+      else { str += tab.charAt((triplet >> 6*(3-j)) & 0x3F); }
+    }
+  }
+  return str;
+}
+
+/*!
+ * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
+ * Digest Algorithm, as defined in RFC 1321.
+ * Version 2.1 Copyright (C) Paul Johnston 1999 - 2002.
+ * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+ * Distributed under the BSD License
+ * See http://pajhome.org.uk/crypt/md5 for more info.
+ */
+
+/*
+ * Everything that isn't used by Strophe has been stripped here!
+ */
+
+var MD5 = (function () {
+    /*
+     * Add integers, wrapping at 2^32. This uses 16-bit operations internally
+     * to work around bugs in some JS interpreters.
+     */
+    var safe_add = function (x, y) {
+        var lsw = (x & 0xFFFF) + (y & 0xFFFF);
+        var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
+        return (msw << 16) | (lsw & 0xFFFF);
+    };
+
+    /*
+     * Bitwise rotate a 32-bit number to the left.
+     */
+    var bit_rol = function (num, cnt) {
+        return (num << cnt) | (num >>> (32 - cnt));
+    };
+
+    /*
+     * Convert a string to an array of little-endian words
+     */
+    var str2binl = function (str) {
+        var bin = [];
+        for(var i = 0; i < str.length * 8; i += 8)
+        {
+            bin[i>>5] |= (str.charCodeAt(i / 8) & 255) << (i%32);
+        }
+        return bin;
+    };
+
+    /*
+     * Convert an array of little-endian words to a string
+     */
+    var binl2str = function (bin) {
+        var str = "";
+        for(var i = 0; i < bin.length * 32; i += 8)
+        {
+            str += String.fromCharCode((bin[i>>5] >>> (i % 32)) & 255);
+        }
+        return str;
+    };
+
+    /*
+     * Convert an array of little-endian words to a hex string.
+     */
+    var binl2hex = function (binarray) {
+        var hex_tab = "0123456789abcdef";
+        var str = "";
+        for(var i = 0; i < binarray.length * 4; i++)
+        {
+            str += hex_tab.charAt((binarray[i>>2] >> ((i%4)*8+4)) & 0xF) +
+                hex_tab.charAt((binarray[i>>2] >> ((i%4)*8  )) & 0xF);
+        }
+        return str;
+    };
+
+    /*
+     * These functions implement the four basic operations the algorithm uses.
+     */
+    var md5_cmn = function (q, a, b, x, s, t) {
+        return safe_add(bit_rol(safe_add(safe_add(a, q),safe_add(x, t)), s),b);
+    };
+
+    var md5_ff = function (a, b, c, d, x, s, t) {
+        return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
+    };
+
+    var md5_gg = function (a, b, c, d, x, s, t) {
+        return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
+    };
+
+    var md5_hh = function (a, b, c, d, x, s, t) {
+        return md5_cmn(b ^ c ^ d, a, b, x, s, t);
+    };
+
+    var md5_ii = function (a, b, c, d, x, s, t) {
+        return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
+    };
+
+    /*
+     * Calculate the MD5 of an array of little-endian words, and a bit length
+     */
+    var core_md5 = function (x, len) {
+        /* append padding */
+        x[len >> 5] |= 0x80 << ((len) % 32);
+        x[(((len + 64) >>> 9) << 4) + 14] = len;
+
+        var a =  1732584193;
+        var b = -271733879;
+        var c = -1732584194;
+        var d =  271733878;
+
+        var olda, oldb, oldc, oldd;
+        for (var i = 0; i < x.length; i += 16)
+        {
+            olda = a;
+            oldb = b;
+            oldc = c;
+            oldd = d;
+
+            a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
+            d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
+            c = md5_ff(c, d, a, b, x[i+ 2], 17,  606105819);
+            b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
+            a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
+            d = md5_ff(d, a, b, c, x[i+ 5], 12,  1200080426);
+            c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
+            b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
+            a = md5_ff(a, b, c, d, x[i+ 8], 7 ,  1770035416);
+            d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
+            c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
+            b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
+            a = md5_ff(a, b, c, d, x[i+12], 7 ,  1804603682);
+            d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
+            c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
+            b = md5_ff(b, c, d, a, x[i+15], 22,  1236535329);
+
+            a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
+            d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
+            c = md5_gg(c, d, a, b, x[i+11], 14,  643717713);
+            b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
+            a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
+            d = md5_gg(d, a, b, c, x[i+10], 9 ,  38016083);
+            c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
+            b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
+            a = md5_gg(a, b, c, d, x[i+ 9], 5 ,  568446438);
+            d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
+            c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
+            b = md5_gg(b, c, d, a, x[i+ 8], 20,  1163531501);
+            a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
+            d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
+            c = md5_gg(c, d, a, b, x[i+ 7], 14,  1735328473);
+            b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);
+
+            a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
+            d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
+            c = md5_hh(c, d, a, b, x[i+11], 16,  1839030562);
+            b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
+            a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
+            d = md5_hh(d, a, b, c, x[i+ 4], 11,  1272893353);
+            c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
+            b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
+            a = md5_hh(a, b, c, d, x[i+13], 4 ,  681279174);
+            d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
+            c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
+            b = md5_hh(b, c, d, a, x[i+ 6], 23,  76029189);
+            a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
+            d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
+            c = md5_hh(c, d, a, b, x[i+15], 16,  530742520);
+            b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);
+
+            a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
+            d = md5_ii(d, a, b, c, x[i+ 7], 10,  1126891415);
+            c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
+            b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
+            a = md5_ii(a, b, c, d, x[i+12], 6 ,  1700485571);
+            d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
+            c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
+            b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
+            a = md5_ii(a, b, c, d, x[i+ 8], 6 ,  1873313359);
+            d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
+            c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
+            b = md5_ii(b, c, d, a, x[i+13], 21,  1309151649);
+            a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
+            d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
+            c = md5_ii(c, d, a, b, x[i+ 2], 15,  718787259);
+            b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);
+
+            a = safe_add(a, olda);
+            b = safe_add(b, oldb);
+            c = safe_add(c, oldc);
+            d = safe_add(d, oldd);
+        }
+        return [a, b, c, d];
+    };
+
+
+    var obj = {
+        /*
+         * These are the functions you'll usually want to call.
+         * They take string arguments and return either hex or base-64 encoded
+         * strings.
+         */
+        hexdigest: function (s) {
+            return binl2hex(core_md5(str2binl(s), s.length * 8));
+        },
+
+        hash: function (s) {
+            return binl2str(core_md5(str2binl(s), s.length * 8));
+        }
+    };
+
+    return obj;
+})();
+
+/*!
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+
+/* jshint undef: true, unused: true:, noarg: true, latedef: true */
+/*global document, window, setTimeout, clearTimeout, console,
+    ActiveXObject, Base64, MD5, DOMParser */
+// from sha1.js
+/*global core_hmac_sha1, binb2str, str_hmac_sha1, str_sha1, b64_hmac_sha1*/
+
+/** File: strophe.js
+ *  A JavaScript library for XMPP BOSH/XMPP over Websocket.
+ *
+ *  This is the JavaScript version of the Strophe library.  Since JavaScript
+ *  had no facilities for persistent TCP connections, this library uses
+ *  Bidirectional-streams Over Synchronous HTTP (BOSH) to emulate
+ *  a persistent, stateful, two-way connection to an XMPP server.  More
+ *  information on BOSH can be found in XEP 124.
+ *
+ *  This version of Strophe also works with WebSockets.
+ *  For more information on XMPP-over WebSocket see this RFC draft:
+ *  http://tools.ietf.org/html/draft-ietf-xmpp-websocket-00
+ */
+
+/** PrivateFunction: Function.prototype.bind
+ *  Bind a function to an instance.
+ *
+ *  This Function object extension method creates a bound method similar
+ *  to those in Python.  This means that the 'this' object will point
+ *  to the instance you want.  See
+ *  <a href='https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Function/bind'>MDC's bind() documentation</a> and
+ *  <a href='http://benjamin.smedbergs.us/blog/2007-01-03/bound-functions-and-function-imports-in-javascript/'>Bound Functions and Function Imports in JavaScript</a>
+ *  for a complete explanation.
+ *
+ *  This extension already exists in some browsers (namely, Firefox 3), but
+ *  we provide it to support those that don't.
+ *
+ *  Parameters:
+ *    (Object) obj - The object that will become 'this' in the bound function.
+ *    (Object) argN - An option argument that will be prepended to the
+ *      arguments given for the function call
+ *
+ *  Returns:
+ *    The bound function.
+ */
+if (!Function.prototype.bind) {
+    Function.prototype.bind = function (obj /*, arg1, arg2, ... */)
+    {
+        var func = this;
+        var _slice = Array.prototype.slice;
+        var _concat = Array.prototype.concat;
+        var _args = _slice.call(arguments, 1);
+
+        return function () {
+            return func.apply(obj ? obj : this,
+                              _concat.call(_args,
+                                           _slice.call(arguments, 0)));
+        };
+    };
+}
+
+/** PrivateFunction: Array.prototype.indexOf
+ *  Return the index of an object in an array.
+ *
+ *  This function is not supplied by some JavaScript implementations, so
+ *  we provide it if it is missing.  This code is from:
+ *  http://developer.mozilla.org/En/Core_JavaScript_1.5_Reference:Objects:Array:indexOf
+ *
+ *  Parameters:
+ *    (Object) elt - The object to look for.
+ *    (Integer) from - The index from which to start looking. (optional).
+ *
+ *  Returns:
+ *    The index of elt in the array or -1 if not found.
+ */
+if (!Array.prototype.indexOf)
+{
+    Array.prototype.indexOf = function(elt /*, from*/)
+    {
+        var len = this.length;
+
+        var from = Number(arguments[1]) || 0;
+        from = (from < 0) ? Math.ceil(from) : Math.floor(from);
+        if (from < 0) {
+            from += len;
+        }
+
+        for (; from < len; from++) {
+            if (from in this && this[from] === elt) {
+                return from;
+            }
+        }
+
+        return -1;
+    };
+}
+
+/* All of the Strophe globals are defined in this special function below so
+ * that references to the globals become closures.  This will ensure that
+ * on page reload, these references will still be available to callbacks
+ * that are still executing.
+ */
+
+(function (callback) {
+var Strophe;
+
+/** Function: $build
+ *  Create a Strophe.Builder.
+ *  This is an alias for 'new Strophe.Builder(name, attrs)'.
+ *
+ *  Parameters:
+ *    (String) name - The root element name.
+ *    (Object) attrs - The attributes for the root element in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $build(name, attrs) { return new Strophe.Builder(name, attrs); }
+/** Function: $msg
+ *  Create a Strophe.Builder with a <message/> element as the root.
+ *
+ *  Parmaeters:
+ *    (Object) attrs - The <message/> element attributes in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $msg(attrs) { return new Strophe.Builder("message", attrs); }
+/** Function: $iq
+ *  Create a Strophe.Builder with an <iq/> element as the root.
+ *
+ *  Parameters:
+ *    (Object) attrs - The <iq/> element attributes in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $iq(attrs) { return new Strophe.Builder("iq", attrs); }
+/** Function: $pres
+ *  Create a Strophe.Builder with a <presence/> element as the root.
+ *
+ *  Parameters:
+ *    (Object) attrs - The <presence/> element attributes in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $pres(attrs) { return new Strophe.Builder("presence", attrs); }
+
+/** Class: Strophe
+ *  An object container for all Strophe library functions.
+ *
+ *  This class is just a container for all the objects and constants
+ *  used in the library.  It is not meant to be instantiated, but to
+ *  provide a namespace for library objects, constants, and functions.
+ */
+Strophe = {
+    /** Constant: VERSION
+     *  The version of the Strophe library. Unreleased builds will have
+     *  a version of head-HASH where HASH is a partial revision.
+     */
+    VERSION: "1.1.3",
+
+    /** Constants: XMPP Namespace Constants
+     *  Common namespace constants from the XMPP RFCs and XEPs.
+     *
+     *  NS.HTTPBIND - HTTP BIND namespace from XEP 124.
+     *  NS.BOSH - BOSH namespace from XEP 206.
+     *  NS.CLIENT - Main XMPP client namespace.
+     *  NS.AUTH - Legacy authentication namespace.
+     *  NS.ROSTER - Roster operations namespace.
+     *  NS.PROFILE - Profile namespace.
+     *  NS.DISCO_INFO - Service discovery info namespace from XEP 30.
+     *  NS.DISCO_ITEMS - Service discovery items namespace from XEP 30.
+     *  NS.MUC - Multi-User Chat namespace from XEP 45.
+     *  NS.SASL - XMPP SASL namespace from RFC 3920.
+     *  NS.STREAM - XMPP Streams namespace from RFC 3920.
+     *  NS.BIND - XMPP Binding namespace from RFC 3920.
+     *  NS.SESSION - XMPP Session namespace from RFC 3920.
+     *  NS.XHTML_IM - XHTML-IM namespace from XEP 71.
+     *  NS.XHTML - XHTML body namespace from XEP 71.
+     */
+    NS: {
+        HTTPBIND: "http://jabber.org/protocol/httpbind",
+        BOSH: "urn:xmpp:xbosh",
+        CLIENT: "jabber:client",
+        AUTH: "jabber:iq:auth",
+        ROSTER: "jabber:iq:roster",
+        PROFILE: "jabber:iq:profile",
+        DISCO_INFO: "http://jabber.org/protocol/disco#info",
+        DISCO_ITEMS: "http://jabber.org/protocol/disco#items",
+        MUC: "http://jabber.org/protocol/muc",
+        SASL: "urn:ietf:params:xml:ns:xmpp-sasl",
+        STREAM: "http://etherx.jabber.org/streams",
+        BIND: "urn:ietf:params:xml:ns:xmpp-bind",
+        SESSION: "urn:ietf:params:xml:ns:xmpp-session",
+        VERSION: "jabber:iq:version",
+        STANZAS: "urn:ietf:params:xml:ns:xmpp-stanzas",
+        XHTML_IM: "http://jabber.org/protocol/xhtml-im",
+        XHTML: "http://www.w3.org/1999/xhtml"
+    },
+
+
+    /** Constants: XHTML_IM Namespace
+     *  contains allowed tags, tag attributes, and css properties.
+     *  Used in the createHtml function to filter incoming html into the allowed XHTML-IM subset.
+     *  See http://xmpp.org/extensions/xep-0071.html#profile-summary for the list of recommended
+     *  allowed tags and their attributes.
+     */
+    XHTML: {
+                tags: ['a','blockquote','br','cite','em','img','li','ol','p','span','strong','ul','body'],
+                attributes: {
+                        'a':          ['href'],
+                        'blockquote': ['style'],
+                        'br':         [],
+                        'cite':       ['style'],
+                        'em':         [],
+                        'img':        ['src', 'alt', 'style', 'height', 'width'],
+                        'li':         ['style'],
+                        'ol':         ['style'],
+                        'p':          ['style'],
+                        'span':       ['style'],
+                        'strong':     [],
+                        'ul':         ['style'],
+                        'body':       []
+                },
+                css: ['background-color','color','font-family','font-size','font-style','font-weight','margin-left','margin-right','text-align','text-decoration'],
+                validTag: function(tag)
+                {
+                        for(var i = 0; i < Strophe.XHTML.tags.length; i++) {
+                                if(tag == Strophe.XHTML.tags[i]) {
+                                        return true;
+                                }
+                        }
+                        return false;
+                },
+                validAttribute: function(tag, attribute)
+                {
+                        if(typeof Strophe.XHTML.attributes[tag] !== 'undefined' && Strophe.XHTML.attributes[tag].length > 0) {
+                                for(var i = 0; i < Strophe.XHTML.attributes[tag].length; i++) {
+                                        if(attribute == Strophe.XHTML.attributes[tag][i]) {
+                                                return true;
+                                        }
+                                }
+                        }
+                        return false;
+                },
+                validCSS: function(style)
+                {
+                        for(var i = 0; i < Strophe.XHTML.css.length; i++) {
+                                if(style == Strophe.XHTML.css[i]) {
+                                        return true;
+                                }
+                        }
+                        return false;
+                }
+    },
+
+    /** Constants: Connection Status Constants
+     *  Connection status constants for use by the connection handler
+     *  callback.
+     *
+     *  Status.ERROR - An error has occurred
+     *  Status.CONNECTING - The connection is currently being made
+     *  Status.CONNFAIL - The connection attempt failed
+     *  Status.AUTHENTICATING - The connection is authenticating
+     *  Status.AUTHFAIL - The authentication attempt failed
+     *  Status.CONNECTED - The connection has succeeded
+     *  Status.DISCONNECTED - The connection has been terminated
+     *  Status.DISCONNECTING - The connection is currently being terminated
+     *  Status.ATTACHED - The connection has been attached
+     */
+    Status: {
+        ERROR: 0,
+        CONNECTING: 1,
+        CONNFAIL: 2,
+        AUTHENTICATING: 3,
+        AUTHFAIL: 4,
+        CONNECTED: 5,
+        DISCONNECTED: 6,
+        DISCONNECTING: 7,
+        ATTACHED: 8
+    },
+
+    /** Constants: Log Level Constants
+     *  Logging level indicators.
+     *
+     *  LogLevel.DEBUG - Debug output
+     *  LogLevel.INFO - Informational output
+     *  LogLevel.WARN - Warnings
+     *  LogLevel.ERROR - Errors
+     *  LogLevel.FATAL - Fatal errors
+     */
+    LogLevel: {
+        DEBUG: 0,
+        INFO: 1,
+        WARN: 2,
+        ERROR: 3,
+        FATAL: 4
+    },
+
+    /** PrivateConstants: DOM Element Type Constants
+     *  DOM element types.
+     *
+     *  ElementType.NORMAL - Normal element.
+     *  ElementType.TEXT - Text data element.
+     *  ElementType.FRAGMENT - XHTML fragment element.
+     */
+    ElementType: {
+        NORMAL: 1,
+        TEXT: 3,
+        CDATA: 4,
+        FRAGMENT: 11
+    },
+
+    /** PrivateConstants: Timeout Values
+     *  Timeout values for error states.  These values are in seconds.
+     *  These should not be changed unless you know exactly what you are
+     *  doing.
+     *
+     *  TIMEOUT - Timeout multiplier. A waiting request will be considered
+     *      failed after Math.floor(TIMEOUT * wait) seconds have elapsed.
+     *      This defaults to 1.1, and with default wait, 66 seconds.
+     *  SECONDARY_TIMEOUT - Secondary timeout multiplier. In cases where
+     *      Strophe can detect early failure, it will consider the request
+     *      failed if it doesn't return after
+     *      Math.floor(SECONDARY_TIMEOUT * wait) seconds have elapsed.
+     *      This defaults to 0.1, and with default wait, 6 seconds.
+     */
+    TIMEOUT: 1.1,
+    SECONDARY_TIMEOUT: 0.1,
+
+    /** Function: addNamespace
+     *  This function is used to extend the current namespaces in
+     *  Strophe.NS.  It takes a key and a value with the key being the
+     *  name of the new namespace, with its actual value.
+     *  For example:
+     *  Strophe.addNamespace('PUBSUB', "http://jabber.org/protocol/pubsub");
+     *
+     *  Parameters:
+     *    (String) name - The name under which the namespace will be
+     *      referenced under Strophe.NS
+     *    (String) value - The actual namespace.
+     */
+    addNamespace: function (name, value)
+    {
+      Strophe.NS[name] = value;
+    },
+
+    /** Function: forEachChild
+     *  Map a function over some or all child elements of a given element.
+     *
+     *  This is a small convenience function for mapping a function over
+     *  some or all of the children of an element.  If elemName is null, all
+     *  children will be passed to the function, otherwise only children
+     *  whose tag names match elemName will be passed.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The element to operate on.
+     *    (String) elemName - The child element tag name filter.
+     *    (Function) func - The function to apply to each child.  This
+     *      function should take a single argument, a DOM element.
+     */
+    forEachChild: function (elem, elemName, func)
+    {
+        var i, childNode;
+
+        for (i = 0; i < elem.childNodes.length; i++) {
+            childNode = elem.childNodes[i];
+            if (childNode.nodeType == Strophe.ElementType.NORMAL &&
+                (!elemName || this.isTagEqual(childNode, elemName))) {
+                func(childNode);
+            }
+        }
+    },
+
+    /** Function: isTagEqual
+     *  Compare an element's tag name with a string.
+     *
+     *  This function is case insensitive.
+     *
+     *  Parameters:
+     *    (XMLElement) el - A DOM element.
+     *    (String) name - The element name.
+     *
+     *  Returns:
+     *    true if the element's tag name matches _el_, and false
+     *    otherwise.
+     */
+    isTagEqual: function (el, name)
+    {
+        return el.tagName.toLowerCase() == name.toLowerCase();
+    },
+
+    /** PrivateVariable: _xmlGenerator
+     *  _Private_ variable that caches a DOM document to
+     *  generate elements.
+     */
+    _xmlGenerator: null,
+
+    /** PrivateFunction: _makeGenerator
+     *  _Private_ function that creates a dummy XML DOM document to serve as
+     *  an element and text node generator.
+     */
+    _makeGenerator: function () {
+        var doc;
+
+        // IE9 does implement createDocument(); however, using it will cause the browser to leak memory on page unload.
+        // Here, we test for presence of createDocument() plus IE's proprietary documentMode attribute, which would be
+                // less than 10 in the case of IE9 and below.
+        if (document.implementation.createDocument === undefined ||
+                        document.implementation.createDocument && document.documentMode && document.documentMode < 10) {
+            doc = this._getIEXmlDom();
+            doc.appendChild(doc.createElement('strophe'));
+        } else {
+            doc = document.implementation
+                .createDocument('jabber:client', 'strophe', null);
+        }
+
+        return doc;
+    },
+
+    /** Function: xmlGenerator
+     *  Get the DOM document to generate elements.
+     *
+     *  Returns:
+     *    The currently used DOM document.
+     */
+    xmlGenerator: function () {
+        if (!Strophe._xmlGenerator) {
+            Strophe._xmlGenerator = Strophe._makeGenerator();
+        }
+        return Strophe._xmlGenerator;
+    },
+
+    /** PrivateFunction: _getIEXmlDom
+     *  Gets IE xml doc object
+     *
+     *  Returns:
+     *    A Microsoft XML DOM Object
+     *  See Also:
+     *    http://msdn.microsoft.com/en-us/library/ms757837%28VS.85%29.aspx
+     */
+    _getIEXmlDom : function() {
+        var doc = null;
+        var docStrings = [
+            "Msxml2.DOMDocument.6.0",
+            "Msxml2.DOMDocument.5.0",
+            "Msxml2.DOMDocument.4.0",
+            "MSXML2.DOMDocument.3.0",
+            "MSXML2.DOMDocument",
+            "MSXML.DOMDocument",
+            "Microsoft.XMLDOM"
+        ];
+
+        for (var d = 0; d < docStrings.length; d++) {
+            if (doc === null) {
+                try {
+                    doc = new ActiveXObject(docStrings[d]);
+                } catch (e) {
+                    doc = null;
+                }
+            } else {
+                break;
+            }
+        }
+
+        return doc;
+    },
+
+    /** Function: xmlElement
+     *  Create an XML DOM element.
+     *
+     *  This function creates an XML DOM element correctly across all
+     *  implementations. Note that these are not HTML DOM elements, which
+     *  aren't appropriate for XMPP stanzas.
+     *
+     *  Parameters:
+     *    (String) name - The name for the element.
+     *    (Array|Object) attrs - An optional array or object containing
+     *      key/value pairs to use as element attributes. The object should
+     *      be in the format {'key': 'value'} or {key: 'value'}. The array
+     *      should have the format [['key1', 'value1'], ['key2', 'value2']].
+     *    (String) text - The text child data for the element.
+     *
+     *  Returns:
+     *    A new XML DOM element.
+     */
+    xmlElement: function (name)
+    {
+        if (!name) { return null; }
+
+        var node = Strophe.xmlGenerator().createElement(name);
+
+        // FIXME: this should throw errors if args are the wrong type or
+        // there are more than two optional args
+        var a, i, k;
+        for (a = 1; a < arguments.length; a++) {
+            if (!arguments[a]) { continue; }
+            if (typeof(arguments[a]) == "string" ||
+                typeof(arguments[a]) == "number") {
+                node.appendChild(Strophe.xmlTextNode(arguments[a]));
+            } else if (typeof(arguments[a]) == "object" &&
+                       typeof(arguments[a].sort) == "function") {
+                for (i = 0; i < arguments[a].length; i++) {
+                    if (typeof(arguments[a][i]) == "object" &&
+                        typeof(arguments[a][i].sort) == "function") {
+                        node.setAttribute(arguments[a][i][0],
+                                          arguments[a][i][1]);
+                    }
+                }
+            } else if (typeof(arguments[a]) == "object") {
+                for (k in arguments[a]) {
+                    if (arguments[a].hasOwnProperty(k)) {
+                        node.setAttribute(k, arguments[a][k]);
+                    }
+                }
+            }
+        }
+
+        return node;
+    },
+
+    /*  Function: xmlescape
+     *  Excapes invalid xml characters.
+     *
+     *  Parameters:
+     *     (String) text - text to escape.
+     *
+     *  Returns:
+     *      Escaped text.
+     */
+    xmlescape: function(text)
+    {
+        text = text.replace(/\&/g, "&");
+        text = text.replace(/</g,  "<");
+        text = text.replace(/>/g,  ">");
+        text = text.replace(/'/g,  "'");
+        text = text.replace(/"/g,  """);
+        return text;
+    },
+
+    /** Function: xmlTextNode
+     *  Creates an XML DOM text node.
+     *
+     *  Provides a cross implementation version of document.createTextNode.
+     *
+     *  Parameters:
+     *    (String) text - The content of the text node.
+     *
+     *  Returns:
+     *    A new XML DOM text node.
+     */
+    xmlTextNode: function (text)
+    {
+        return Strophe.xmlGenerator().createTextNode(text);
+    },
+
+    /** Function: xmlHtmlNode
+     *  Creates an XML DOM html node.
+     *
+     *  Parameters:
+     *    (String) html - The content of the html node.
+     *
+     *  Returns:
+     *    A new XML DOM text node.
+     */
+    xmlHtmlNode: function (html)
+    {
+        var node;
+        //ensure text is escaped
+        if (window.DOMParser) {
+            var parser = new DOMParser();
+            node = parser.parseFromString(html, "text/xml");
+        } else {
+            node = new ActiveXObject("Microsoft.XMLDOM");
+            node.async="false";
+            node.loadXML(html);
+        }
+        return node;
+    },
+
+    /** Function: getText
+     *  Get the concatenation of all text children of an element.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    A String with the concatenated text of all text element children.
+     */
+    getText: function (elem)
+    {
+        if (!elem) { return null; }
+
+        var str = "";
+        if (elem.childNodes.length === 0 && elem.nodeType ==
+            Strophe.ElementType.TEXT) {
+            str += elem.nodeValue;
+        }
+
+        for (var i = 0; i < elem.childNodes.length; i++) {
+            if (elem.childNodes[i].nodeType == Strophe.ElementType.TEXT) {
+                str += elem.childNodes[i].nodeValue;
+            }
+        }
+
+        return Strophe.xmlescape(str);
+    },
+
+    /** Function: copyElement
+     *  Copy an XML DOM element.
+     *
+     *  This function copies a DOM element and all its descendants and returns
+     *  the new copy.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    A new, copied DOM element tree.
+     */
+    copyElement: function (elem)
+    {
+        var i, el;
+        if (elem.nodeType == Strophe.ElementType.NORMAL) {
+            el = Strophe.xmlElement(elem.tagName);
+
+            for (i = 0; i < elem.attributes.length; i++) {
+                el.setAttribute(elem.attributes[i].nodeName.toLowerCase(),
+                                elem.attributes[i].value);
+            }
+
+            for (i = 0; i < elem.childNodes.length; i++) {
+                el.appendChild(Strophe.copyElement(elem.childNodes[i]));
+            }
+        } else if (elem.nodeType == Strophe.ElementType.TEXT) {
+            el = Strophe.xmlGenerator().createTextNode(elem.nodeValue);
+        }
+
+        return el;
+    },
+
+
+    /** Function: createHtml
+     *  Copy an HTML DOM element into an XML DOM.
+     *
+     *  This function copies a DOM element and all its descendants and returns
+     *  the new copy.
+     *
+     *  Parameters:
+     *    (HTMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    A new, copied DOM element tree.
+     */
+    createHtml: function (elem)
+    {
+        var i, el, j, tag, attribute, value, css, cssAttrs, attr, cssName, cssValue;
+        if (elem.nodeType == Strophe.ElementType.NORMAL) {
+            tag = elem.nodeName.toLowerCase();
+            if(Strophe.XHTML.validTag(tag)) {
+                try {
+                    el = Strophe.xmlElement(tag);
+                    for(i = 0; i < Strophe.XHTML.attributes[tag].length; i++) {
+                        attribute = Strophe.XHTML.attributes[tag][i];
+                        value = elem.getAttribute(attribute);
+                        if(typeof value == 'undefined' || value === null || value === '' || value === false || value === 0) {
+                            continue;
+                        }
+                        if(attribute == 'style' && typeof value == 'object') {
+                            if(typeof value.cssText != 'undefined') {
+                                value = value.cssText; // we're dealing with IE, need to get CSS out
+                            }
+                        }
+                        // filter out invalid css styles
+                        if(attribute == 'style') {
+                            css = [];
+                            cssAttrs = value.split(';');
+                            for(j = 0; j < cssAttrs.length; j++) {
+                                attr = cssAttrs[j].split(':');
+                                cssName = attr[0].replace(/^\s*/, "").replace(/\s*$/, "").toLowerCase();
+                                if(Strophe.XHTML.validCSS(cssName)) {
+                                    cssValue = attr[1].replace(/^\s*/, "").replace(/\s*$/, "");
+                                    css.push(cssName + ': ' + cssValue);
+                                }
+                            }
+                            if(css.length > 0) {
+                                value = css.join('; ');
+                                el.setAttribute(attribute, value);
+                            }
+                        } else {
+                            el.setAttribute(attribute, value);
+                        }
+                    }
+
+                    for (i = 0; i < elem.childNodes.length; i++) {
+                        el.appendChild(Strophe.createHtml(elem.childNodes[i]));
+                    }
+                } catch(e) { // invalid elements
+                  el = Strophe.xmlTextNode('');
+                }
+            } else {
+                el = Strophe.xmlGenerator().createDocumentFragment();
+                for (i = 0; i < elem.childNodes.length; i++) {
+                    el.appendChild(Strophe.createHtml(elem.childNodes[i]));
+                }
+            }
+        } else if (elem.nodeType == Strophe.ElementType.FRAGMENT) {
+            el = Strophe.xmlGenerator().createDocumentFragment();
+            for (i = 0; i < elem.childNodes.length; i++) {
+                el.appendChild(Strophe.createHtml(elem.childNodes[i]));
+            }
+        } else if (elem.nodeType == Strophe.ElementType.TEXT) {
+            el = Strophe.xmlTextNode(elem.nodeValue);
+        }
+
+        return el;
+    },
+
+    /** Function: escapeNode
+     *  Escape the node part (also called local part) of a JID.
+     *
+     *  Parameters:
+     *    (String) node - A node (or local part).
+     *
+     *  Returns:
+     *    An escaped node (or local part).
+     */
+    escapeNode: function (node)
+    {
+        return node.replace(/^\s+|\s+$/g, '')
+            .replace(/\\/g,  "\\5c")
+            .replace(/ /g,   "\\20")
+            .replace(/\"/g,  "\\22")
+            .replace(/\&/g,  "\\26")
+            .replace(/\'/g,  "\\27")
+            .replace(/\//g,  "\\2f")
+            .replace(/:/g,   "\\3a")
+            .replace(/</g,   "\\3c")
+            .replace(/>/g,   "\\3e")
+            .replace(/@/g,   "\\40");
+    },
+
+    /** Function: unescapeNode
+     *  Unescape a node part (also called local part) of a JID.
+     *
+     *  Parameters:
+     *    (String) node - A node (or local part).
+     *
+     *  Returns:
+     *    An unescaped node (or local part).
+     */
+    unescapeNode: function (node)
+    {
+        return node.replace(/\\20/g, " ")
+            .replace(/\\22/g, '"')
+            .replace(/\\26/g, "&")
+            .replace(/\\27/g, "'")
+            .replace(/\\2f/g, "/")
+            .replace(/\\3a/g, ":")
+            .replace(/\\3c/g, "<")
+            .replace(/\\3e/g, ">")
+            .replace(/\\40/g, "@")
+            .replace(/\\5c/g, "\\");
+    },
+
+    /** Function: getNodeFromJid
+     *  Get the node portion of a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the node.
+     */
+    getNodeFromJid: function (jid)
+    {
+        if (jid.indexOf("@") < 0) { return null; }
+        return jid.split("@")[0];
+    },
+
+    /** Function: getDomainFromJid
+     *  Get the domain portion of a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the domain.
+     */
+    getDomainFromJid: function (jid)
+    {
+        var bare = Strophe.getBareJidFromJid(jid);
+        if (bare.indexOf("@") < 0) {
+            return bare;
+        } else {
+            var parts = bare.split("@");
+            parts.splice(0, 1);
+            return parts.join('@');
+        }
+    },
+
+    /** Function: getResourceFromJid
+     *  Get the resource portion of a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the resource.
+     */
+    getResourceFromJid: function (jid)
+    {
+        var s = jid.split("/");
+        if (s.length < 2) { return null; }
+        s.splice(0, 1);
+        return s.join('/');
+    },
+
+    /** Function: getBareJidFromJid
+     *  Get the bare JID from a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the bare JID.
+     */
+    getBareJidFromJid: function (jid)
+    {
+        return jid ? jid.split("/")[0] : null;
+    },
+
+    /** Function: log
+     *  User overrideable logging function.
+     *
+     *  This function is called whenever the Strophe library calls any
+     *  of the logging functions.  The default implementation of this
+     *  function does nothing.  If client code wishes to handle the logging
+     *  messages, it should override this with
+     *  > Strophe.log = function (level, msg) {
+     *  >   (user code here)
+     *  > };
+     *
+     *  Please note that data sent and received over the wire is logged
+     *  via Strophe.Connection.rawInput() and Strophe.Connection.rawOutput().
+     *
+     *  The different levels and their meanings are
+     *
+     *    DEBUG - Messages useful for debugging purposes.
+     *    INFO - Informational messages.  This is mostly information like
+     *      'disconnect was called' or 'SASL auth succeeded'.
+     *    WARN - Warnings about potential problems.  This is mostly used
+     *      to report transient connection errors like request timeouts.
+     *    ERROR - Some error occurred.
+     *    FATAL - A non-recoverable fatal error occurred.
+     *
+     *  Parameters:
+     *    (Integer) level - The log level of the log message.  This will
+     *      be one of the values in Strophe.LogLevel.
+     *    (String) msg - The log message.
+     */
+    /* jshint ignore:start */
+    log: function (level, msg)
+    {
+        return;
+    },
+    /* jshint ignore:end */
+
+    /** Function: debug
+     *  Log a message at the Strophe.LogLevel.DEBUG level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    debug: function(msg)
+    {
+        this.log(this.LogLevel.DEBUG, msg);
+    },
+
+    /** Function: info
+     *  Log a message at the Strophe.LogLevel.INFO level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    info: function (msg)
+    {
+        this.log(this.LogLevel.INFO, msg);
+    },
+
+    /** Function: warn
+     *  Log a message at the Strophe.LogLevel.WARN level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    warn: function (msg)
+    {
+        this.log(this.LogLevel.WARN, msg);
+    },
+
+    /** Function: error
+     *  Log a message at the Strophe.LogLevel.ERROR level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    error: function (msg)
+    {
+        this.log(this.LogLevel.ERROR, msg);
+    },
+
+    /** Function: fatal
+     *  Log a message at the Strophe.LogLevel.FATAL level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    fatal: function (msg)
+    {
+        this.log(this.LogLevel.FATAL, msg);
+    },
+
+    /** Function: serialize
+     *  Render a DOM element and all descendants to a String.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    The serialized element tree as a String.
+     */
+    serialize: function (elem)
+    {
+        var result;
+
+        if (!elem) { return null; }
+
+        if (typeof(elem.tree) === "function") {
+            elem = elem.tree();
+        }
+
+        var nodeName = elem.nodeName;
+        var i, child;
+
+        if (elem.getAttribute("_realname")) {
+            nodeName = elem.getAttribute("_realname");
+        }
+
+        result = "<" + nodeName;
+        for (i = 0; i < elem.attributes.length; i++) {
+               if(elem.attributes[i].nodeName != "_realname") {
+                 result += " " + elem.attributes[i].nodeName.toLowerCase() +
+                "='" + elem.attributes[i].value
+                    .replace(/&/g, "&")
+                       .replace(/\'/g, "'")
+                       .replace(/>/g, ">")
+                       .replace(/</g, "<") + "'";
+               }
+        }
+
+        if (elem.childNodes.length > 0) {
+            result += ">";
+            for (i = 0; i < elem.childNodes.length; i++) {
+                child = elem.childNodes[i];
+                switch( child.nodeType ){
+                  case Strophe.ElementType.NORMAL:
+                    // normal element, so recurse
+                    result += Strophe.serialize(child);
+                    break;
+                  case Strophe.ElementType.TEXT:
+                    // text element to escape values
+                    result += Strophe.xmlescape(child.nodeValue);
+                    break;
+                  case Strophe.ElementType.CDATA:
+                    // cdata section so don't escape values
+                    result += "<![CDATA["+child.nodeValue+"]]>";
+                }
+            }
+            result += "</" + nodeName + ">";
+        } else {
+            result += "/>";
+        }
+
+        return result;
+    },
+
+    /** PrivateVariable: _requestId
+     *  _Private_ variable that keeps track of the request ids for
+     *  connections.
+     */
+    _requestId: 0,
+
+    /** PrivateVariable: Strophe.connectionPlugins
+     *  _Private_ variable Used to store plugin names that need
+     *  initialization on Strophe.Connection construction.
+     */
+    _connectionPlugins: {},
+
+    /** Function: addConnectionPlugin
+     *  Extends the Strophe.Connection object with the given plugin.
+     *
+     *  Parameters:
+     *    (String) name - The name of the extension.
+     *    (Object) ptype - The plugin's prototype.
+     */
+    addConnectionPlugin: function (name, ptype)
+    {
+        Strophe._connectionPlugins[name] = ptype;
+    }
+};
+
+/** Class: Strophe.Builder
+ *  XML DOM builder.
+ *
+ *  This object provides an interface similar to JQuery but for building
+ *  DOM element easily and rapidly.  All the functions except for toString()
+ *  and tree() return the object, so calls can be chained.  Here's an
+ *  example using the $iq() builder helper.
+ *  > $iq({to: 'you', from: 'me', type: 'get', id: '1'})
+ *  >     .c('query', {xmlns: 'strophe:example'})
+ *  >     .c('example')
+ *  >     .toString()
+ *  The above generates this XML fragment
+ *  > <iq to='you' from='me' type='get' id='1'>
+ *  >   <query xmlns='strophe:example'>
+ *  >     <example/>
+ *  >   </query>
+ *  > </iq>
+ *  The corresponding DOM manipulations to get a similar fragment would be
+ *  a lot more tedious and probably involve several helper variables.
+ *
+ *  Since adding children makes new operations operate on the child, up()
+ *  is provided to traverse up the tree.  To add two children, do
+ *  > builder.c('child1', ...).up().c('child2', ...)
+ *  The next operation on the Builder will be relative to the second child.
+ */
+
+/** Constructor: Strophe.Builder
+ *  Create a Strophe.Builder object.
+ *
+ *  The attributes should be passed in object notation.  For example
+ *  > var b = new Builder('message', {to: 'you', from: 'me'});
+ *  or
+ *  > var b = new Builder('messsage', {'xml:lang': 'en'});
+ *
+ *  Parameters:
+ *    (String) name - The name of the root element.
+ *    (Object) attrs - The attributes for the root element in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder.
+ */
+Strophe.Builder = function (name, attrs)
+{
+    // Set correct namespace for jabber:client elements
+    if (name == "presence" || name == "message" || name == "iq") {
+        if (attrs && !attrs.xmlns) {
+            attrs.xmlns = Strophe.NS.CLIENT;
+        } else if (!attrs) {
+            attrs = {xmlns: Strophe.NS.CLIENT};
+        }
+    }
+
+    // Holds the tree being built.
+    this.nodeTree = Strophe.xmlElement(name, attrs);
+
+    // Points to the current operation node.
+    this.node = this.nodeTree;
+};
+
+Strophe.Builder.prototype = {
+    /** Function: tree
+     *  Return the DOM tree.
+     *
+     *  This function returns the current DOM tree as an element object.  This
+     *  is suitable for passing to functions like Strophe.Connection.send().
+     *
+     *  Returns:
+     *    The DOM tree as a element object.
+     */
+    tree: function ()
+    {
+        return this.nodeTree;
+    },
+
+    /** Function: toString
+     *  Serialize the DOM tree to a String.
+     *
+     *  This function returns a string serialization of the current DOM
+     *  tree.  It is often used internally to pass data to a
+     *  Strophe.Request object.
+     *
+     *  Returns:
+     *    The serialized DOM tree in a String.
+     */
+    toString: function ()
+    {
+        return Strophe.serialize(this.nodeTree);
+    },
+
+    /** Function: up
+     *  Make the current parent element the new current element.
+     *
+     *  This function is often used after c() to traverse back up the tree.
+     *  For example, to add two children to the same element
+     *  > builder.c('child1', {}).up().c('child2', {});
+     *
+     *  Returns:
+     *    The Stophe.Builder object.
+     */
+    up: function ()
+    {
+        this.node = this.node.parentNode;
+        return this;
+    },
+
+    /** Function: attrs
+     *  Add or modify attributes of the current element.
+     *
+     *  The attributes should be passed in object notation.  This function
+     *  does not move the current element pointer.
+     *
+     *  Parameters:
+     *    (Object) moreattrs - The attributes to add/modify in object notation.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    attrs: function (moreattrs)
+    {
+        for (var k in moreattrs) {
+            if (moreattrs.hasOwnProperty(k)) {
+                this.node.setAttribute(k, moreattrs[k]);
+            }
+        }
+        return this;
+    },
+
+    /** Function: c
+     *  Add a child to the current element and make it the new current
+     *  element.
+     *
+     *  This function moves the current element pointer to the child,
+     *  unless text is provided.  If you need to add another child, it
+     *  is necessary to use up() to go back to the parent in the tree.
+     *
+     *  Parameters:
+     *    (String) name - The name of the child.
+     *    (Object) attrs - The attributes of the child in object notation.
+     *    (String) text - The text to add to the child.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    c: function (name, attrs, text)
+    {
+        var child = Strophe.xmlElement(name, attrs, text);
+        this.node.appendChild(child);
+        if (!text) {
+            this.node = child;
+        }
+        return this;
+    },
+
+    /** Function: cnode
+     *  Add a child to the current element and make it the new current
+     *  element.
+     *
+     *  This function is the same as c() except that instead of using a
+     *  name and an attributes object to create the child it uses an
+     *  existing DOM element object.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    cnode: function (elem)
+    {
+        var impNode;
+        var xmlGen = Strophe.xmlGenerator();
+        try {
+            impNode = (xmlGen.importNode !== undefined);
+        }
+        catch (e) {
+            impNode = false;
+        }
+        var newElem = impNode ?
+                      xmlGen.importNode(elem, true) :
+                      Strophe.copyElement(elem);
+        this.node.appendChild(newElem);
+        this.node = newElem;
+        return this;
+    },
+
+    /** Function: t
+     *  Add a child text element.
+     *
+     *  This *does not* make the child the new current element since there
+     *  are no children of text elements.
+     *
+     *  Parameters:
+     *    (String) text - The text data to append to the current element.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    t: function (text)
+    {
+        var child = Strophe.xmlTextNode(text);
+        this.node.appendChild(child);
+        return this;
+    },
+
+    /** Function: h
+     *  Replace current element contents with the HTML passed in.
+     *
+     *  This *does not* make the child the new current element
+     *
+     *  Parameters:
+     *    (String) html - The html to insert as contents of current element.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    h: function (html)
+    {
+        var fragment = document.createElement('body');
+
+        // force the browser to try and fix any invalid HTML tags
+        fragment.innerHTML = html;
+
+        // copy cleaned html into an xml dom
+        var xhtml = Strophe.createHtml(fragment);
+
+        while(xhtml.childNodes.length > 0) {
+            this.node.appendChild(xhtml.childNodes[0]);
+        }
+        return this;
+    }
+};
+
+/** PrivateClass: Strophe.Handler
+ *  _Private_ helper class for managing stanza handlers.
+ *
+ *  A Strophe.Handler encapsulates a user provided callback function to be
+ *  executed when matching stanzas are received by the connection.
+ *  Handlers can be either one-off or persistant depending on their
+ *  return value. Returning true will cause a Handler to remain active, and
+ *  returning false will remove the Handler.
+ *
+ *  Users will not use Strophe.Handler objects directly, but instead they
+ *  will use Strophe.Connection.addHandler() and
+ *  Strophe.Connection.deleteHandler().
+ */
+
+/** PrivateConstructor: Strophe.Handler
+ *  Create and initialize a new Strophe.Handler.
+ *
+ *  Parameters:
+ *    (Function) handler - A function to be executed when the handler is run.
+ *    (String) ns - The namespace to match.
+ *    (String) name - The element name to match.
+ *    (String) type - The element type to match.
+ *    (String) id - The element id attribute to match.
+ *    (String) from - The element from attribute to match.
+ *    (Object) options - Handler options
+ *
+ *  Returns:
+ *    A new Strophe.Handler object.
+ */
+Strophe.Handler = function (handler, ns, name, type, id, from, options)
+{
+    this.handler = handler;
+    this.ns = ns;
+    this.name = name;
+    this.type = type;
+    this.id = id;
+    this.options = options || {matchBare: false};
+
+    // default matchBare to false if undefined
+    if (!this.options.matchBare) {
+        this.options.matchBare = false;
+    }
+
+    if (this.options.matchBare) {
+        this.from = from ? Strophe.getBareJidFromJid(from) : null;
+    } else {
+        this.from = from;
+    }
+
+    // whether the handler is a user handler or a system handler
+    this.user = true;
+};
+
+Strophe.Handler.prototype = {
+    /** PrivateFunction: isMatch
+     *  Tests if a stanza matches the Strophe.Handler.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The XML element to test.
+     *
+     *  Returns:
+     *    true if the stanza matches and false otherwise.
+     */
+    isMatch: function (elem)
+    {
+        var nsMatch;
+        var from = null;
+
+        if (this.options.matchBare) {
+            from = Strophe.getBareJidFromJid(elem.getAttribute('from'));
+        } else {
+            from = elem.getAttribute('from');
+        }
+
+        nsMatch = false;
+        if (!this.ns) {
+            nsMatch = true;
+        } else {
+            var that = this;
+            Strophe.forEachChild(elem, null, function (elem) {
+                if (elem.getAttribute("xmlns") == that.ns) {
+                    nsMatch = true;
+                }
+            });
+
+            nsMatch = nsMatch || elem.getAttribute("xmlns") == this.ns;
+        }
+
+        if (nsMatch &&
+            (!this.name || Strophe.isTagEqual(elem, this.name)) &&
+            (!this.type || elem.getAttribute("type") == this.type) &&
+            (!this.id || elem.getAttribute("id") == this.id) &&
+            (!this.from || from == this.from)) {
+                return true;
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: run
+     *  Run the callback on a matching stanza.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The DOM element that triggered the
+     *      Strophe.Handler.
+     *
+     *  Returns:
+     *    A boolean indicating if the handler should remain active.
+     */
+    run: function (elem)
+    {
+        var result = null;
+        try {
+            result = this.handler(elem);
+        } catch (e) {
+            if (e.sourceURL) {
+                Strophe.fatal("error: " + this.handler +
+                              " " + e.sourceURL + ":" +
+                              e.line + " - " + e.name + ": " + e.message);
+            } else if (e.fileName) {
+                if (typeof(console) != "undefined") {
+                    console.trace();
+                    console.error(this.handler, " - error - ", e, e.message);
+                }
+                Strophe.fatal("error: " + this.handler + " " +
+                              e.fileName + ":" + e.lineNumber + " - " +
+                              e.name + ": " + e.message);
+            } else {
+                Strophe.fatal("error: " + e.message + "\n" + e.stack);
+            }
+
+            throw e;
+        }
+
+        return result;
+    },
+
+    /** PrivateFunction: toString
+     *  Get a String representation of the Strophe.Handler object.
+     *
+     *  Returns:
+     *    A String.
+     */
+    toString: function ()
+    {
+        return "{Handler: " + this.handler + "(" + this.name + "," +
+            this.id + "," + this.ns + ")}";
+    }
+};
+
+/** PrivateClass: Strophe.TimedHandler
+ *  _Private_ helper class for managing timed handlers.
+ *
+ *  A Strophe.TimedHandler encapsulates a user provided callback that
+ *  should be called after a certain period of time or at regular
+ *  intervals.  The return value of the callback determines whether the
+ *  Strophe.TimedHandler will continue to fire.
+ *
+ *  Users will not use Strophe.TimedHandler objects directly, but instead
+ *  they will use Strophe.Connection.addTimedHandler() and
+ *  Strophe.Connection.deleteTimedHandler().
+ */
+
+/** PrivateConstructor: Strophe.TimedHandler
+ *  Create and initialize a new Strophe.TimedHandler object.
+ *
+ *  Parameters:
+ *    (Integer) period - The number of milliseconds to wait before the
+ *      handler is called.
+ *    (Function) handler - The callback to run when the handler fires.  This
+ *      function should take no arguments.
+ *
+ *  Returns:
+ *    A new Strophe.TimedHandler object.
+ */
+Strophe.TimedHandler = function (period, handler)
+{
+    this.period = period;
+    this.handler = handler;
+
+    this.lastCalled = new Date().getTime();
+    this.user = true;
+};
+
+Strophe.TimedHandler.prototype = {
+    /** PrivateFunction: run
+     *  Run the callback for the Strophe.TimedHandler.
+     *
+     *  Returns:
+     *    true if the Strophe.TimedHandler should be called again, and false
+     *      otherwise.
+     */
+    run: function ()
+    {
+        this.lastCalled = new Date().getTime();
+        return this.handler();
+    },
+
+    /** PrivateFunction: reset
+     *  Reset the last called time for the Strophe.TimedHandler.
+     */
+    reset: function ()
+    {
+        this.lastCalled = new Date().getTime();
+    },
+
+    /** PrivateFunction: toString
+     *  Get a string representation of the Strophe.TimedHandler object.
+     *
+     *  Returns:
+     *    The string representation.
+     */
+    toString: function ()
+    {
+        return "{TimedHandler: " + this.handler + "(" + this.period +")}";
+    }
+};
+
+/** Class: Strophe.Connection
+ *  XMPP Connection manager.
+ *
+ *  This class is the main part of Strophe.  It manages a BOSH connection
+ *  to an XMPP server and dispatches events to the user callbacks as
+ *  data arrives.  It supports SASL PLAIN, SASL DIGEST-MD5, SASL SCRAM-SHA1
+ *  and legacy authentication.
+ *
+ *  After creating a Strophe.Connection object, the user will typically
+ *  call connect() with a user supplied callback to handle connection level
+ *  events like authentication failure, disconnection, or connection
+ *  complete.
+ *
+ *  The user will also have several event handlers defined by using
+ *  addHandler() and addTimedHandler().  These will allow the user code to
+ *  respond to interesting stanzas or do something periodically with the
+ *  connection.  These handlers will be active once authentication is
+ *  finished.
+ *
+ *  To send data to the connection, use send().
+ */
+
+/** Constructor: Strophe.Connection
+ *  Create and initialize a Strophe.Connection object.
+ *
+ *  The transport-protocol for this connection will be chosen automatically
+ *  based on the given service parameter. URLs starting with "ws://" or
+ *  "wss://" will use WebSockets, URLs starting with "http://", "https://"
+ *  or without a protocol will use BOSH.
+ *
+ *  To make Strophe connect to the current host you can leave out the protocol
+ *  and host part and just pass the path, e.g.
+ *
+ *  > var conn = new Strophe.Connection("/http-bind/");
+ *
+ *  WebSocket options:
+ *
+ *  If you want to connect to the current host with a WebSocket connection you
+ *  can tell Strophe to use WebSockets through a "protocol" attribute in the
+ *  optional options parameter. Valid values are "ws" for WebSocket and "wss"
+ *  for Secure WebSocket.
+ *  So to connect to "wss://CURRENT_HOSTNAME/xmpp-websocket" you would call
+ *
+ *  > var conn = new Strophe.Connection("/xmpp-websocket/", {protocol: "wss"});
+ *
+ *  Note that relative URLs _NOT_ starting with a "/" will also include the path
+ *  of the current site.
+ *
+ *  Also because downgrading security is not permitted by browsers, when using
+ *  relative URLs both BOSH and WebSocket connections will use their secure
+ *  variants if the current connection to the site is also secure (https).
+ *
+ *  BOSH options:
+ *
+ *  by adding "sync" to the options, you can control if requests will
+ *  be made synchronously or not. The default behaviour is asynchronous.
+ *  If you want to make requests synchronous, make "sync" evaluate to true:
+ *  > var conn = new Strophe.Connection("/http-bind/", {sync: true});
+ *  You can also toggle this on an already established connection:
+ *  > conn.options.sync = true;
+ *
+ *
+ *  Parameters:
+ *    (String) service - The BOSH or WebSocket service URL.
+ *    (Object) options - A hash of configuration options
+ *
+ *  Returns:
+ *    A new Strophe.Connection object.
+ */
+Strophe.Connection = function (service, options)
+{
+    // The service URL
+    this.service = service;
+
+    // Configuration options
+    this.options = options || {};
+    var proto = this.options.protocol || "";
+
+    // Select protocal based on service or options
+    if (service.indexOf("ws:") === 0 || service.indexOf("wss:") === 0 ||
+            proto.indexOf("ws") === 0) {
+        this._proto = new Strophe.Websocket(this);
+    } else {
+        this._proto = new Strophe.Bosh(this);
+    }
+    /* The connected JID. */
+    this.jid = "";
+    /* the JIDs domain */
+    this.domain = null;
+    /* stream:features */
+    this.features = null;
+
+    // SASL
+    this._sasl_data = {};
+    this.do_session = false;
+    this.do_bind = false;
+
+    // handler lists
+    this.timedHandlers = [];
+    this.handlers = [];
+    this.removeTimeds = [];
+    this.removeHandlers = [];
+    this.addTimeds = [];
+    this.addHandlers = [];
+
+    this._authentication = {};
+    this._idleTimeout = null;
+    this._disconnectTimeout = null;
+
+    this.do_authentication = true;
+    this.authenticated = false;
+    this.disconnecting = false;
+    this.connected = false;
+
+    this.errors = 0;
+
+    this.paused = false;
+
+    this._data = [];
+    this._uniqueId = 0;
+
+    this._sasl_success_handler = null;
+    this._sasl_failure_handler = null;
+    this._sasl_challenge_handler = null;
+
+    // Max retries before disconnecting
+    this.maxRetries = 5;
+
+    // setup onIdle callback every 1/10th of a second
+    this._idleTimeout = setTimeout(this._onIdle.bind(this), 100);
+
+    // initialize plugins
+    for (var k in Strophe._connectionPlugins) {
+        if (Strophe._connectionPlugins.hasOwnProperty(k)) {
+            var ptype = Strophe._connectionPlugins[k];
+            // jslint complaints about the below line, but this is fine
+            var F = function () {}; // jshint ignore:line
+            F.prototype = ptype;
+            this[k] = new F();
+            this[k].init(this);
+        }
+    }
+};
+
+Strophe.Connection.prototype = {
+    /** Function: reset
+     *  Reset the connection.
+     *
+     *  This function should be called after a connection is disconnected
+     *  before that connection is reused.
+     */
+    reset: function ()
+    {
+        this._proto._reset();
+
+        // SASL
+        this.do_session = false;
+        this.do_bind = false;
+
+        // handler lists
+        this.timedHandlers = [];
+        this.handlers = [];
+        this.removeTimeds = [];
+        this.removeHandlers = [];
+        this.addTimeds = [];
+        this.addHandlers = [];
+        this._authentication = {};
+
+        this.authenticated = false;
+        this.disconnecting = false;
+        this.connected = false;
+
+        this.errors = 0;
+
+        this._requests = [];
+        this._uniqueId = 0;
+    },
+
+    /** Function: pause
+     *  Pause the request manager.
+     *
+     *  This will prevent Strophe from sending any more requests to the
+     *  server.  This is very useful for temporarily pausing
+     *  BOSH-Connections while a lot of send() calls are happening quickly.
+     *  This causes Strophe to send the data in a single request, saving
+     *  many request trips.
+     */
+    pause: function ()
+    {
+        this.paused = true;
+    },
+
+    /** Function: resume
+     *  Resume the request manager.
+     *
+     *  This resumes after pause() has been called.
+     */
+    resume: function ()
+    {
+        this.paused = false;
+    },
+
+    /** Function: getUniqueId
+     *  Generate a unique ID for use in <iq/> elements.
+     *
+     *  All <iq/> stanzas are required to have unique id attributes.  This
+     *  function makes creating these easy.  Each connection instance has
+     *  a counter which starts from zero, and the value of this counter
+     *  plus a colon followed by the suffix becomes the unique id. If no
+     *  suffix is supplied, the counter is used as the unique id.
+     *
+     *  Suffixes are used to make debugging easier when reading the stream
+     *  data, and their use is recommended.  The counter resets to 0 for
+     *  every new connection for the same reason.  For connections to the
+     *  same server that authenticate the same way, all the ids should be
+     *  the same, which makes it easy to see changes.  This is useful for
+     *  automated testing as well.
+     *
+     *  Parameters:
+     *    (String) suffix - A optional suffix to append to the id.
+     *
+     *  Returns:
+     *    A unique string to be used for the id attribute.
+     */
+    getUniqueId: function (suffix)
+    {
+        if (typeof(suffix) == "string" || typeof(suffix) == "number") {
+            return ++this._uniqueId + ":" + suffix;
+        } else {
+            return ++this._uniqueId + "";
+        }
+    },
+
+    /** Function: connect
+     *  Starts the connection process.
+     *
+     *  As the connection process proceeds, the user supplied callback will
+     *  be triggered multiple times with status updates.  The callback
+     *  should take two arguments - the status code and the error condition.
+     *
+     *  The status code will be one of the values in the Strophe.Status
+     *  constants.  The error condition will be one of the conditions
+     *  defined in RFC 3920 or the condition 'strophe-parsererror'.
+     *
+     *  The Parameters _wait_, _hold_ and _route_ are optional and only relevant
+     *  for BOSH connections. Please see XEP 124 for a more detailed explanation
+     *  of the optional parameters.
+     *
+     *  Parameters:
+     *    (String) jid - The user's JID.  This may be a bare JID,
+     *      or a full JID.  If a node is not supplied, SASL ANONYMOUS
+     *      authentication will be attempted.
+     *    (String) pass - The user's password.
+     *    (Function) callback - The connect callback function.
+     *    (Integer) wait - The optional HTTPBIND wait value.  This is the
+     *      time the server will wait before returning an empty result for
+     *      a request.  The default setting of 60 seconds is recommended.
+     *    (Integer) hold - The optional HTTPBIND hold value.  This is the
+     *      number of connections the server will hold at one time.  This
+     *      should almost always be set to 1 (the default).
+     *    (String) route - The optional route value.
+     */
+    connect: function (jid, pass, callback, wait, hold, route)
+    {
+        this.jid = jid;
+        /** Variable: authzid
+         *  Authorization identity.
+         */
+        this.authzid = Strophe.getBareJidFromJid(this.jid);
+        /** Variable: authcid
+         *  Authentication identity (User name).
+         */
+        this.authcid = Strophe.getNodeFromJid(this.jid);
+        /** Variable: pass
+         *  Authentication identity (User password).
+         */
+        this.pass = pass;
+        /** Variable: servtype
+         *  Digest MD5 compatibility.
+         */
+        this.servtype = "xmpp";
+        this.connect_callback = callback;
+        this.disconnecting = false;
+        this.connected = false;
+        this.authenticated = false;
+        this.errors = 0;
+
+        // parse jid for domain
+        this.domain = Strophe.getDomainFromJid(this.jid);
+
+        this._changeConnectStatus(Strophe.Status.CONNECTING, null);
+
+        this._proto._connect(wait, hold, route);
+    },
+
+    /** Function: attach
+     *  Attach to an already created and authenticated BOSH session.
+     *
+     *  This function is provided to allow Strophe to attach to BOSH
+     *  sessions which have been created externally, perhaps by a Web
+     *  application.  This is often used to support auto-login type features
+     *  without putting user credentials into the page.
+     *
+     *  Parameters:
+     *    (String) jid - The full JID that is bound by the session.
+     *    (String) sid - The SID of the BOSH session.
+     *    (String) rid - The current RID of the BOSH session.  This RID
+     *      will be used by the next request.
+     *    (Function) callback The connect callback function.
+     *    (Integer) wait - The optional HTTPBIND wait value.  This is the
+     *      time the server will wait before returning an empty result for
+     *      a request.  The default setting of 60 seconds is recommended.
+     *      Other settings will require tweaks to the Strophe.TIMEOUT value.
+     *    (Integer) hold - The optional HTTPBIND hold value.  This is the
+     *      number of connections the server will hold at one time.  This
+     *      should almost always be set to 1 (the default).
+     *    (Integer) wind - The optional HTTBIND window value.  This is the
+     *      allowed range of request ids that are valid.  The default is 5.
+     */
+    attach: function (jid, sid, rid, callback, wait, hold, wind)
+    {
+        this._proto._attach(jid, sid, rid, callback, wait, hold, wind);
+    },
+
+    /** Function: xmlInput
+     *  User overrideable function that receives XML data coming into the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.xmlInput = function (elem) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Due to limitations of current Browsers' XML-Parsers the opening and closing
+     *  <stream> tag for WebSocket-Connoctions will be passed as selfclosing here.
+     *
+     *  BOSH-Connections will have all stanzas wrapped in a <body> tag. See
+     *  <Strophe.Bosh.strip> if you want to strip this tag.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The XML data received by the connection.
+     */
+    /* jshint unused:false */
+    xmlInput: function (elem)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: xmlOutput
+     *  User overrideable function that receives XML data sent to the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.xmlOutput = function (elem) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Due to limitations of current Browsers' XML-Parsers the opening and closing
+     *  <stream> tag for WebSocket-Connoctions will be passed as selfclosing here.
+     *
+     *  BOSH-Connections will have all stanzas wrapped in a <body> tag. See
+     *  <Strophe.Bosh.strip> if you want to strip this tag.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The XMLdata sent by the connection.
+     */
+    /* jshint unused:false */
+    xmlOutput: function (elem)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: rawInput
+     *  User overrideable function that receives raw data coming into the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.rawInput = function (data) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Parameters:
+     *    (String) data - The data received by the connection.
+     */
+    /* jshint unused:false */
+    rawInput: function (data)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: rawOutput
+     *  User overrideable function that receives raw data sent to the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.rawOutput = function (data) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Parameters:
+     *    (String) data - The data sent by the connection.
+     */
+    /* jshint unused:false */
+    rawOutput: function (data)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: send
+     *  Send a stanza.
+     *
+     *  This function is called to push data onto the send queue to
+     *  go out over the wire.  Whenever a request is sent to the BOSH
+     *  server, all pending data is sent and the queue is flushed.
+     *
+     *  Parameters:
+     *    (XMLElement |
+     *     [XMLElement] |
+     *     Strophe.Builder) elem - The stanza to send.
+     */
+    send: function (elem)
+    {
+        if (elem === null) { return ; }
+        if (typeof(elem.sort) === "function") {
+            for (var i = 0; i < elem.length; i++) {
+                this._queueData(elem[i]);
+            }
+        } else if (typeof(elem.tree) === "function") {
+            this._queueData(elem.tree());
+        } else {
+            this._queueData(elem);
+        }
+
+        this._proto._send();
+    },
+
+    /** Function: flush
+     *  Immediately send any pending outgoing data.
+     *
+     *  Normally send() queues outgoing data until the next idle period
+     *  (100ms), which optimizes network use in the common cases when
+     *  several send()s are called in succession. flush() can be used to
+     *  immediately send all pending data.
+     */
+    flush: function ()
+    {
+        // cancel the pending idle period and run the idle function
+        // immediately
+        clearTimeout(this._idleTimeout);
+        this._onIdle();
+    },
+
+    /** Function: sendIQ
+     *  Helper function to send IQ stanzas.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The stanza to send.
+     *    (Function) callback - The callback function for a successful request.
+     *    (Function) errback - The callback function for a failed or timed
+     *      out request.  On timeout, the stanza will be null.
+     *    (Integer) timeout - The time specified in milliseconds for a
+     *      timeout to occur.
+     *
+     *  Returns:
+     *    The id used to send the IQ.
+    */
+    sendIQ: function(elem, callback, errback, timeout) {
+        var timeoutHandler = null;
+        var that = this;
+
+        if (typeof(elem.tree) === "function") {
+            elem = elem.tree();
+        }
+        var id = elem.getAttribute('id');
+
+        // inject id if not found
+        if (!id) {
+            id = this.getUniqueId("sendIQ");
+            elem.setAttribute("id", id);
+        }
+
+        var handler = this.addHandler(function (stanza) {
+            // remove timeout handler if there is one
+            if (timeoutHandler) {
+                that.deleteTimedHandler(timeoutHandler);
+            }
+
+            var iqtype = stanza.getAttribute('type');
+            if (iqtype == 'result') {
+                if (callback) {
+                    callback(stanza);
+                }
+            } else if (iqtype == 'error') {
+                if (errback) {
+                    errback(stanza);
+                }
+            } else {
+                throw {
+                    name: "StropheError",
+            message: "Got bad IQ type of " + iqtype
+                };
+            }
+        }, null, 'iq', null, id);
+
+        // if timeout specified, setup timeout handler.
+        if (timeout) {
+            timeoutHandler = this.addTimedHandler(timeout, function () {
+                // get rid of normal handler
+                that.deleteHandler(handler);
+
+                // call errback on timeout with null stanza
+                if (errback) {
+                    errback(null);
+                }
+                return false;
+            });
+        }
+
+        this.send(elem);
+
+        return id;
+    },
+
+    /** PrivateFunction: _queueData
+     *  Queue outgoing data for later sending.  Also ensures that the data
+     *  is a DOMElement.
+     */
+    _queueData: function (element) {
+        if (element === null ||
+            !element.tagName ||
+            !element.childNodes) {
+            throw {
+                name: "StropheError",
+                message: "Cannot queue non-DOMElement."
+            };
+        }
+
+        this._data.push(element);
+    },
+
+    /** PrivateFunction: _sendRestart
+     *  Send an xmpp:restart stanza.
+     */
+    _sendRestart: function ()
+    {
+        this._data.push("restart");
+
+        this._proto._sendRestart();
+
+        this._idleTimeout = setTimeout(this._onIdle.bind(this), 100);
+    },
+
+    /** Function: addTimedHandler
+     *  Add a timed handler to the connection.
+     *
+     *  This function adds a timed handler.  The provided handler will
+     *  be called every period milliseconds until it returns false,
+     *  the connection is terminated, or the handler is removed.  Handlers
+     *  that wish to continue being invoked should return true.
+     *
+     *  Because of method binding it is necessary to save the result of
+     *  this function if you wish to remove a handler with
+     *  deleteTimedHandler().
+     *
+     *  Note that user handlers are not active until authentication is
+     *  successful.
+     *
+     *  Parameters:
+     *    (Integer) period - The period of the handler.
+     *    (Function) handler - The callback function.
+     *
+     *  Returns:
+     *    A reference to the handler that can be used to remove it.
+     */
+    addTimedHandler: function (period, handler)
+    {
+        var thand = new Strophe.TimedHandler(period, handler);
+        this.addTimeds.push(thand);
+        return thand;
+    },
+
+    /** Function: deleteTimedHandler
+     *  Delete a timed handler for a connection.
+     *
+     *  This function removes a timed handler from the connection.  The
+     *  handRef parameter is *not* the function passed to addTimedHandler(),
+     *  but is the reference returned from addTimedHandler().
+     *
+     *  Parameters:
+     *    (Strophe.TimedHandler) handRef - The handler reference.
+     */
+    deleteTimedHandler: function (handRef)
+    {
+        // this must be done in the Idle loop so that we don't change
+        // the handlers during iteration
+        this.removeTimeds.push(handRef);
+    },
+
+    /** Function: addHandler
+     *  Add a stanza handler for the connection.
+     *
+     *  This function adds a stanza handler to the connection.  The
+     *  handler callback will be called for any stanza that matches
+     *  the parameters.  Note that if multiple parameters are supplied,
+     *  they must all match for the handler to be invoked.
+     *
+     *  The handler will receive the stanza that triggered it as its argument.
+     *  The handler should return true if it is to be invoked again;
+     *  returning false will remove the handler after it returns.
+     *
+     *  As a convenience, the ns parameters applies to the top level element
+     *  and also any of its immediate children.  This is primarily to make
+     *  matching /iq/query elements easy.
+     *
+     *  The options argument contains handler matching flags that affect how
+     *  matches are determined. Currently the only flag is matchBare (a
+     *  boolean). When matchBare is true, the from parameter and the from
+     *  attribute on the stanza will be matched as bare JIDs instead of
+     *  full JIDs. To use this, pass {matchBare: true} as the value of
+     *  options. The default value for matchBare is false.
+     *
+     *  The return value should be saved if you wish to remove the handler
+     *  with deleteHandler().
+     *
+     *  Parameters:
+     *    (Function) handler - The user callback.
+     *    (String) ns - The namespace to match.
+     *    (String) name - The stanza name to match.
+     *    (String) type - The stanza type attribute to match.
+     *    (String) id - The stanza id attribute to match.
+     *    (String) from - The stanza from attribute to match.
+     *    (String) options - The handler options
+     *
+     *  Returns:
+     *    A reference to the handler that can be used to remove it.
+     */
+    addHandler: function (handler, ns, name, type, id, from, options)
+    {
+        var hand = new Strophe.Handler(handler, ns, name, type, id, from, options);
+        this.addHandlers.push(hand);
+        return hand;
+    },
+
+    /** Function: deleteHandler
+     *  Delete a stanza handler for a connection.
+     *
+     *  This function removes a stanza handler from the connection.  The
+     *  handRef parameter is *not* the function passed to addHandler(),
+     *  but is the reference returned from addHandler().
+     *
+     *  Parameters:
+     *    (Strophe.Handler) handRef - The handler reference.
+     */
+    deleteHandler: function (handRef)
+    {
+        // this must be done in the Idle loop so that we don't change
+        // the handlers during iteration
+        this.removeHandlers.push(handRef);
+    },
+
+    /** Function: disconnect
+     *  Start the graceful disconnection process.
+     *
+     *  This function starts the disconnection process.  This process starts
+     *  by sending unavailable presence and sending BOSH body of type
+     *  terminate.  A timeout handler makes sure that disconnection happens
+     *  even if the BOSH server does not respond.
+     *
+     *  The user supplied connection callback will be notified of the
+     *  progress as this process happens.
+     *
+     *  Parameters:
+     *    (String) reason - The reason the disconnect is occuring.
+     */
+    disconnect: function (reason)
+    {
+        this._changeConnectStatus(Strophe.Status.DISCONNECTING, reason);
+
+        Strophe.info("Disconnect was called because: " + reason);
+        if (this.connected) {
+            var pres = false;
+            this.disconnecting = true;
+            if (this.authenticated) {
+                pres = $pres({
+                    xmlns: Strophe.NS.CLIENT,
+                    type: 'unavailable'
+                });
+            }
+            // setup timeout handler
+            this._disconnectTimeout = this._addSysTimedHandler(
+                3000, this._onDisconnectTimeout.bind(this));
+            this._proto._disconnect(pres);
+        }
+    },
+
+    /** PrivateFunction: _changeConnectStatus
+     *  _Private_ helper function that makes sure plugins and the user's
+     *  callback are notified of connection status changes.
+     *
+     *  Parameters:
+     *    (Integer) status - the new connection status, one of the values
+     *      in Strophe.Status
+     *    (String) condition - the error condition or null
+     */
+    _changeConnectStatus: function (status, condition)
+    {
+        // notify all plugins listening for status changes
+        for (var k in Strophe._connectionPlugins) {
+            if (Strophe._connectionPlugins.hasOwnProperty(k)) {
+                var plugin = this[k];
+                if (plugin.statusChanged) {
+                    try {
+                        plugin.statusChanged(status, condition);
+                    } catch (err) {
+                        Strophe.error("" + k + " plugin caused an exception " +
+                                      "changing status: " + err);
+                    }
+                }
+            }
+        }
+
+        // notify the user's callback
+        if (this.connect_callback) {
+            try {
+                this.connect_callback(status, condition);
+            } catch (e) {
+                Strophe.error("User connection callback caused an " +
+                              "exception: " + e);
+            }
+        }
+    },
+
+    /** PrivateFunction: _doDisconnect
+     *  _Private_ function to disconnect.
+     *
+     *  This is the last piece of the disconnection logic.  This resets the
+     *  connection and alerts the user's connection callback.
+     */
+    _doDisconnect: function ()
+    {
+        // Cancel Disconnect Timeout
+        if (this._disconnectTimeout !== null) {
+            this.deleteTimedHandler(this._disconnectTimeout);
+            this._disconnectTimeout = null;
+        }
+
+        Strophe.info("_doDisconnect was called");
+        this._proto._doDisconnect();
+
+        this.authenticated = false;
+        this.disconnecting = false;
+
+        // delete handlers
+        this.handlers = [];
+        this.timedHandlers = [];
+        this.removeTimeds = [];
+        this.removeHandlers = [];
+        this.addTimeds = [];
+        this.addHandlers = [];
+
+        // tell the parent we disconnected
+        this._changeConnectStatus(Strophe.Status.DISCONNECTED, null);
+        this.connected = false;
+    },
+
+    /** PrivateFunction: _dataRecv
+     *  _Private_ handler to processes incoming data from the the connection.
+     *
+     *  Except for _connect_cb handling the initial connection request,
+     *  this function handles the incoming data for all requests.  This
+     *  function also fires stanza handlers that match each incoming
+     *  stanza.
+     *
+     *  Parameters:
+     *    (Strophe.Request) req - The request that has data ready.
+     *    (string) req - The stanza a raw string (optiona).
+     */
+    _dataRecv: function (req, raw)
+    {
+        Strophe.info("_dataRecv called");
+        var elem = this._proto._reqToData(req);
+        if (elem === null) { return; }
+
+        if (this.xmlInput !== Strophe.Connection.prototype.xmlInput) {
+            if (elem.nodeName === this._proto.strip && elem.childNodes.length) {
+                this.xmlInput(elem.childNodes[0]);
+            } else {
+                this.xmlInput(elem);
+            }
+        }
+        if (this.rawInput !== Strophe.Connection.prototype.rawInput) {
+            if (raw) {
+                this.rawInput(raw);
+            } else {
+                this.rawInput(Strophe.serialize(elem));
+            }
+        }
+
+        // remove handlers scheduled for deletion
+        var i, hand;
+        while (this.removeHandlers.length > 0) {
+            hand = this.removeHandlers.pop();
+            i = this.handlers.indexOf(hand);
+            if (i >= 0) {
+                this.handlers.splice(i, 1);
+            }
+        }
+
+        // add handlers scheduled for addition
+        while (this.addHandlers.length > 0) {
+            this.handlers.push(this.addHandlers.pop());
+        }
+
+        // handle graceful disconnect
+        if (this.disconnecting && this._proto._emptyQueue()) {
+            this._doDisconnect();
+            return;
+        }
+
+        var typ = elem.getAttribute("type");
+        var cond, conflict;
+        if (typ !== null && typ == "terminate") {
+            // Don't process stanzas that come in after disconnect
+            if (this.disconnecting) {
+                return;
+            }
+
+            // an error occurred
+            cond = elem.getAttribute("condition");
+            conflict = elem.getElementsByTagName("conflict");
+            if (cond !== null) {
+                if (cond == "remote-stream-error" && conflict.length > 0) {
+                    cond = "conflict";
+                }
+                this._changeConnectStatus(Strophe.Status.CONNFAIL, cond);
+            } else {
+                this._changeConnectStatus(Strophe.Status.CONNFAIL, "unknown");
+            }
+            this.disconnect('unknown stream-error');
+            return;
+        }
+
+        // send each incoming stanza through the handler chain
+        var that = this;
+        Strophe.forEachChild(elem, null, function (child) {
+            var i, newList;
+            // process handlers
+            newList = that.handlers;
+            that.handlers = [];
+            for (i = 0; i < newList.length; i++) {
+                var hand = newList[i];
+                // encapsulate 'handler.run' not to lose the whole handler list if
+                // one of the handlers throws an exception
+                try {
+                    if (hand.isMatch(child) &&
+                        (that.authenticated || !hand.user)) {
+                        if (hand.run(child)) {
+                            that.handlers.push(hand);
+                        }
+                    } else {
+                        that.handlers.push(hand);
+                    }
+                } catch(e) {
+                    // if the handler throws an exception, we consider it as false
+                    Strophe.warn('Removing Strophe handlers due to uncaught exception: ' + e.message);
+                }
+            }
+        });
+    },
+
+
+    /** Attribute: mechanisms
+     *  SASL Mechanisms available for Conncection.
+     */
+    mechanisms: {},
+
+    /** PrivateFunction: _connect_cb
+     *  _Private_ handler for initial connection request.
+     *
+     *  This handler is used to process the initial connection request
+     *  response from the BOSH server. It is used to set up authentication
+     *  handlers and start the authentication process.
+     *
+     *  SASL authentication will be attempted if available, otherwise
+     *  the code will fall back to legacy authentication.
+     *
+     *  Parameters:
+     *    (Strophe.Request) req - The current request.
+     *    (Function) _callback - low level (xmpp) connect callback function.
+     *      Useful for plugins with their own xmpp connect callback (when their)
+     *      want to do something special).
+     */
+    _connect_cb: function (req, _callback, raw)
+    {
+        Strophe.info("_connect_cb was called");
+
+        this.connected = true;
+
+        var bodyWrap = this._proto._reqToData(req);
+        if (!bodyWrap) { return; }
+
+        if (this.xmlInput !== Strophe.Connection.prototype.xmlInput) {
+            if (bodyWrap.nodeName === this._proto.strip && bodyWrap.childNodes.length) {
+                this.xmlInput(bodyWrap.childNodes[0]);
+            } else {
+                this.xmlInput(bodyWrap);
+            }
+        }
+        if (this.rawInput !== Strophe.Connection.prototype.rawInput) {
+            if (raw) {
+                this.rawInput(raw);
+            } else {
+                this.rawInput(Strophe.serialize(bodyWrap));
+            }
+        }
+
+        var conncheck = this._proto._connect_cb(bodyWrap);
+        if (conncheck === Strophe.Status.CONNFAIL) {
+            return;
+        }
+
+        this._authentication.sasl_scram_sha1 = false;
+        this._authentication.sasl_plain = false;
+        this._authentication.sasl_digest_md5 = false;
+        this._authentication.sasl_anonymous = false;
+
+        this._authentication.legacy_auth = false;
+
+        // Check for the stream:features tag
+        var hasFeatures = bodyWrap.getElementsByTagName("stream:features").length > 0;
+        if (!hasFeatures) {
+            hasFeatures = bodyWrap.getElementsByTagName("features").length > 0;
+        }
+        var mechanisms = bodyWrap.getElementsByTagName("mechanism");
+        var matched = [];
+        var i, mech, found_authentication = false;
+        if (!hasFeatures) {
+            this._proto._no_auth_received(_callback);
+            return;
+        }
+        if (mechanisms.length > 0) {
+            for (i = 0; i < mechanisms.length; i++) {
+                mech = Strophe.getText(mechanisms[i]);
+                if (this.mechanisms[mech]) matched.push(this.mechanisms[mech]);
+            }
+        }
+        this._authentication.legacy_auth =
+            bodyWrap.getElementsByTagName("auth").length > 0;
+        found_authentication = this._authentication.legacy_auth ||
+            matched.length > 0;
+        if (!found_authentication) {
+            this._proto._no_auth_received(_callback);
+            return;
+        }
+        if (this.do_authentication !== false)
+            this.authenticate(matched);
+    },
+
+    /** Function: authenticate
+     * Set up authentication
+     *
+     *  Contiunues the initial connection request by setting up authentication
+     *  handlers and start the authentication process.
+     *
+     *  SASL authentication will be attempted if available, otherwise
+     *  the code will fall back to legacy authentication.
+     *
+     */
+    authenticate: function (matched)
+    {
+      var i;
+      // Sorting matched mechanisms according to priority.
+      for (i = 0; i < matched.length - 1; ++i) {
+        var higher = i;
+        for (var j = i + 1; j < matched.length; ++j) {
+          if (matched[j].prototype.priority > matched[higher].prototype.priority) {
+            higher = j;
+          }
+        }
+        if (higher != i) {
+          var swap = matched[i];
+          matched[i] = matched[higher];
+          matched[higher] = swap;
+        }
+      }
+
+      // run each mechanism
+      var mechanism_found = false;
+      for (i = 0; i < matched.length; ++i) {
+        if (!matched[i].test(this)) continue;
+
+        this._sasl_success_handler = this._addSysHandler(
+          this._sasl_success_cb.bind(this), null,
+          "success", null, null);
+        this._sasl_failure_handler = this._addSysHandler(
+          this._sasl_failure_cb.bind(this), null,
+          "failure", null, null);
+        this._sasl_challenge_handler = this._addSysHandler(
+          this._sasl_challenge_cb.bind(this), null,
+          "challenge", null, null);
+
+        this._sasl_mechanism = new matched[i]();
+        this._sasl_mechanism.onStart(this);
+
+        var request_auth_exchange = $build("auth", {
+          xmlns: Strophe.NS.SASL,
+          mechanism: this._sasl_mechanism.name
+        });
+
+        if (this._sasl_mechanism.isClientFirst) {
+          var response = this._sasl_mechanism.onChallenge(this, null);
+          request_auth_exchange.t(Base64.encode(response));
+        }
+
+        this.send(request_auth_exchange.tree());
+
+        mechanism_found = true;
+        break;
+      }
+
+      if (!mechanism_found) {
+        // if none of the mechanism worked
+        if (Strophe.getNodeFromJid(this.jid) === null) {
+            // we don't have a node, which is required for non-anonymous
+            // client connections
+            this._changeConnectStatus(Strophe.Status.CONNFAIL,
+                                      'x-strophe-bad-non-anon-jid');
+            this.disconnect('x-strophe-bad-non-anon-jid');
+        } else {
+          // fall back to legacy authentication
+          this._changeConnectStatus(Strophe.Status.AUTHENTICATING, null);
+          this._addSysHandler(this._auth1_cb.bind(this), null, null,
+                              null, "_auth_1");
+
+          this.send($iq({
+            type: "get",
+            to: this.domain,
+            id: "_auth_1"
+          }).c("query", {
+            xmlns: Strophe.NS.AUTH
+          }).c("username", {}).t(Strophe.getNodeFromJid(this.jid)).tree());
+        }
+      }
+
+    },
+
+    _sasl_challenge_cb: function(elem) {
+      var challenge = Base64.decode(Strophe.getText(elem));
+      var response = this._sasl_mechanism.onChallenge(this, challenge);
+
+      var stanza = $build('response', {
+          xmlns: Strophe.NS.SASL
+      });
+      if (response !== "") {
+        stanza.t(Base64.encode(response));
+      }
+      this.send(stanza.tree());
+
+      return true;
+    },
+
+    /** PrivateFunction: _auth1_cb
+     *  _Private_ handler for legacy authentication.
+     *
+     *  This handler is called in response to the initial <iq type='get'/>
+     *  for legacy authentication.  It builds an authentication <iq/> and
+     *  sends it, creating a handler (calling back to _auth2_cb()) to
+     *  handle the result
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The stanza that triggered the callback.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    /* jshint unused:false */
+    _auth1_cb: function (elem)
+    {
+        // build plaintext auth iq
+        var iq = $iq({type: "set", id: "_auth_2"})
+            .c('query', {xmlns: Strophe.NS.AUTH})
+            .c('username', {}).t(Strophe.getNodeFromJid(this.jid))
+            .up()
+            .c('password').t(this.pass);
+
+        if (!Strophe.getResourceFromJid(this.jid)) {
+            // since the user has not supplied a resource, we pick
+            // a default one here.  unlike other auth methods, the server
+            // cannot do this for us.
+            this.jid = Strophe.getBareJidFromJid(this.jid) + '/strophe';
+        }
+        iq.up().c('resource', {}).t(Strophe.getResourceFromJid(this.jid));
+
+        this._addSysHandler(this._auth2_cb.bind(this), null,
+                            null, null, "_auth_2");
+
+        this.send(iq.tree());
+
+        return false;
+    },
+    /* jshint unused:true */
+
+    /** PrivateFunction: _sasl_success_cb
+     *  _Private_ handler for succesful SASL authentication.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_success_cb: function (elem)
+    {
+        if (this._sasl_data["server-signature"]) {
+            var serverSignature;
+            var success = Base64.decode(Strophe.getText(elem));
+            var attribMatch = /([a-z]+)=([^,]+)(,|$)/;
+            var matches = success.match(attribMatch);
+            if (matches[1] == "v") {
+                serverSignature = matches[2];
+            }
+
+            if (serverSignature != this._sasl_data["server-signature"]) {
+              // remove old handlers
+              this.deleteHandler(this._sasl_failure_handler);
+              this._sasl_failure_handler = null;
+              if (this._sasl_challenge_handler) {
+                this.deleteHandler(this._sasl_challenge_handler);
+                this._sasl_challenge_handler = null;
+              }
+
+              this._sasl_data = {};
+              return this._sasl_failure_cb(null);
+            }
+        }
+
+        Strophe.info("SASL authentication succeeded.");
+
+        if(this._sasl_mechanism)
+          this._sasl_mechanism.onSuccess();
+
+        // remove old handlers
+        this.deleteHandler(this._sasl_failure_handler);
+        this._sasl_failure_handler = null;
+        if (this._sasl_challenge_handler) {
+            this.deleteHandler(this._sasl_challenge_handler);
+            this._sasl_challenge_handler = null;
+        }
+
+        this._addSysHandler(this._sasl_auth1_cb.bind(this), null,
+                            "stream:features", null, null);
+
+        // we must send an xmpp:restart now
+        this._sendRestart();
+
+        return false;
+    },
+
+    /** PrivateFunction: _sasl_auth1_cb
+     *  _Private_ handler to start stream binding.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_auth1_cb: function (elem)
+    {
+        // save stream:features for future usage
+        this.features = elem;
+
+        var i, child;
+
+        for (i = 0; i < elem.childNodes.length; i++) {
+            child = elem.childNodes[i];
+            if (child.nodeName == 'bind') {
+                this.do_bind = true;
+            }
+
+            if (child.nodeName == 'session') {
+                this.do_session = true;
+            }
+        }
+
+        if (!this.do_bind) {
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            return false;
+        } else {
+            this._addSysHandler(this._sasl_bind_cb.bind(this), null, null,
+                                null, "_bind_auth_2");
+
+            var resource = Strophe.getResourceFromJid(this.jid);
+            if (resource) {
+                this.send($iq({type: "set", id: "_bind_auth_2"})
+                          .c('bind', {xmlns: Strophe.NS.BIND})
+                          .c('resource', {}).t(resource).tree());
+            } else {
+                this.send($iq({type: "set", id: "_bind_auth_2"})
+                          .c('bind', {xmlns: Strophe.NS.BIND})
+                          .tree());
+            }
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: _sasl_bind_cb
+     *  _Private_ handler for binding result and session start.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_bind_cb: function (elem)
+    {
+        if (elem.getAttribute("type") == "error") {
+            Strophe.info("SASL binding failed.");
+            var conflict = elem.getElementsByTagName("conflict"), condition;
+            if (conflict.length > 0) {
+                condition = 'conflict';
+            }
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, condition);
+            return false;
+        }
+
+        // TODO - need to grab errors
+        var bind = elem.getElementsByTagName("bind");
+        var jidNode;
+        if (bind.length > 0) {
+            // Grab jid
+            jidNode = bind[0].getElementsByTagName("jid");
+            if (jidNode.length > 0) {
+                this.jid = Strophe.getText(jidNode[0]);
+
+                if (this.do_session) {
+                    this._addSysHandler(this._sasl_session_cb.bind(this),
+                                        null, null, null, "_session_auth_2");
+
+                    this.send($iq({type: "set", id: "_session_auth_2"})
+                                  .c('session', {xmlns: Strophe.NS.SESSION})
+                                  .tree());
+                } else {
+                    this.authenticated = true;
+                    this._changeConnectStatus(Strophe.Status.CONNECTED, null);
+                }
+            }
+        } else {
+            Strophe.info("SASL binding failed.");
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            return false;
+        }
+    },
+
+    /** PrivateFunction: _sasl_session_cb
+     *  _Private_ handler to finish successful SASL connection.
+     *
+     *  This sets Connection.authenticated to true on success, which
+     *  starts the processing of user handlers.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_session_cb: function (elem)
+    {
+        if (elem.getAttribute("type") == "result") {
+            this.authenticated = true;
+            this._changeConnectStatus(Strophe.Status.CONNECTED, null);
+        } else if (elem.getAttribute("type") == "error") {
+            Strophe.info("Session creation failed.");
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            return false;
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: _sasl_failure_cb
+     *  _Private_ handler for SASL authentication failure.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    /* jshint unused:false */
+    _sasl_failure_cb: function (elem)
+    {
+        // delete unneeded handlers
+        if (this._sasl_success_handler) {
+            this.deleteHandler(this._sasl_success_handler);
+            this._sasl_success_handler = null;
+        }
+        if (this._sasl_challenge_handler) {
+            this.deleteHandler(this._sasl_challenge_handler);
+            this._sasl_challenge_handler = null;
+        }
+
+        if(this._sasl_mechanism)
+          this._sasl_mechanism.onFailure();
+        this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+        return false;
+    },
+    /* jshint unused:true */
+
+    /** PrivateFunction: _auth2_cb
+     *  _Private_ handler to finish legacy authentication.
+     *
+     *  This handler is called when the result from the jabber:iq:auth
+     *  <iq/> stanza is returned.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The stanza that triggered the callback.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _auth2_cb: function (elem)
+    {
+        if (elem.getAttribute("type") == "result") {
+            this.authenticated = true;
+            this._changeConnectStatus(Strophe.Status.CONNECTED, null);
+        } else if (elem.getAttribute("type") == "error") {
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            this.disconnect('authentication failed');
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: _addSysTimedHandler
+     *  _Private_ function to add a system level timed handler.
+     *
+     *  This function is used to add a Strophe.TimedHandler for the
+     *  library code.  System timed handlers are allowed to run before
+     *  authentication is complete.
+     *
+     *  Parameters:
+     *    (Integer) period - The period of the handler.
+     *    (Function) handler - The callback function.
+     */
+    _addSysTimedHandler: function (period, handler)
+    {
+        var thand = new Strophe.TimedHandler(period, handler);
+        thand.user = false;
+        this.addTimeds.push(thand);
+        return thand;
+    },
+
+    /** PrivateFunction: _addSysHandler
+     *  _Private_ function to add a system level stanza handler.
+     *
+     *  This function is used to add a Strophe.Handler for the
+     *  library code.  System stanza handlers are allowed to run before
+     *  authentication is complete.
+     *
+     *  Parameters:
+     *    (Function) handler - The callback function.
+     *    (String) ns - The namespace to match.
+     *    (String) name - The stanza name to match.
+     *    (String) type - The stanza type attribute to match.
+     *    (String) id - The stanza id attribute to match.
+     */
+    _addSysHandler: function (handler, ns, name, type, id)
+    {
+        var hand = new Strophe.Handler(handler, ns, name, type, id);
+        hand.user = false;
+        this.addHandlers.push(hand);
+        return hand;
+    },
+
+    /** PrivateFunction: _onDisconnectTimeout
+     *  _Private_ timeout handler for handling non-graceful disconnection.
+     *
+     *  If the graceful disconnect process does not complete within the
+     *  time allotted, this handler finishes the disconnect anyway.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _onDisconnectTimeout: function ()
+    {
+        Strophe.info("_onDisconnectTimeout was called");
+
+        this._proto._onDisconnectTimeout();
+
+        // actually disconnect
+        this._doDisconnect();
+
+        return false;
+    },
+
+    /** PrivateFunction: _onIdle
+     *  _Private_ handler to process events during idle cycle.
+     *
+     *  This handler is called every 100ms to fire timed handlers that
+     *  are ready and keep poll requests going.
+     */
+    _onIdle: function ()
+    {
+        var i, thand, since, newList;
+
+        // add timed handlers scheduled for addition
+        // NOTE: we add before remove in the case a timed handler is
+        // added and then deleted before the next _onIdle() call.
+        while (this.addTimeds.length > 0) {
+            this.timedHandlers.push(this.addTimeds.pop());
+        }
+
+        // remove timed handlers that have been scheduled for deletion
+        while (this.removeTimeds.length > 0) {
+            thand = this.removeTimeds.pop();
+            i = this.timedHandlers.indexOf(thand);
+            if (i >= 0) {
+                this.timedHandlers.splice(i, 1);
+            }
+        }
+
+        // call ready timed handlers
+        var now = new Date().getTime();
+        newList = [];
+        for (i = 0; i < this.timedHandlers.length; i++) {
+            thand = this.timedHandlers[i];
+            if (this.authenticated || !thand.user) {
+                since = thand.lastCalled + thand.period;
+                if (since - now <= 0) {
+                    if (thand.run()) {
+                        newList.push(thand);
+                    }
+                } else {
+                    newList.push(thand);
+                }
+            }
+        }
+        this.timedHandlers = newList;
+
+        clearTimeout(this._idleTimeout);
+
+        this._proto._onIdle();
+
+        // reactivate the timer only if connected
+        if (this.connected) {
+            this._idleTimeout = setTimeout(this._onIdle.bind(this), 100);
+        }
+    }
+};
+
+if (callback) {
+    callback(Strophe, $build, $msg, $iq, $pres);
+}
+
+/** Class: Strophe.SASLMechanism
+ *
+ *  encapsulates SASL authentication mechanisms.
+ *
+ *  User code may override the priority for each mechanism or disable it completely.
+ *  See <priority> for information about changing priority and <test> for informatian on
+ *  how to disable a mechanism.
+ *
+ *  By default, all mechanisms are enabled and the priorities are
+ *
+ *  SCRAM-SHA1 - 40
+ *  DIGEST-MD5 - 30
+ *  Plain - 20
+ */
+
+/**
+ * PrivateConstructor: Strophe.SASLMechanism
+ * SASL auth mechanism abstraction.
+ *
+ *  Parameters:
+ *    (String) name - SASL Mechanism name.
+ *    (Boolean) isClientFirst - If client should send response first without challenge.
+ *    (Number) priority - Priority.
+ *
+ *  Returns:
+ *    A new Strophe.SASLMechanism object.
+ */
+Strophe.SASLMechanism = function(name, isClientFirst, priority) {
+  /** PrivateVariable: name
+   *  Mechanism name.
+   */
+  this.name = name;
+  /** PrivateVariable: isClientFirst
+   *  If client sends response without initial server challenge.
+   */
+  this.isClientFirst = isClientFirst;
+  /** Variable: priority
+   *  Determines which <SASLMechanism> is chosen for authentication (Higher is better).
+   *  Users may override this to prioritize mechanisms differently.
+   *
+   *  In the default configuration the priorities are
+   *
+   *  SCRAM-SHA1 - 40
+   *  DIGEST-MD5 - 30
+   *  Plain - 20
+   *
+   *  Example: (This will cause Strophe to choose the mechanism that the server sent first)
+   *
+   *  > Strophe.SASLMD5.priority = Strophe.SASLSHA1.priority;
+   *
+   *  See <SASL mechanisms> for a list of available mechanisms.
+   *
+   */
+  this.priority = priority;
+};
+
+Strophe.SASLMechanism.prototype = {
+  /**
+   *  Function: test
+   *  Checks if mechanism able to run.
+   *  To disable a mechanism, make this return false;
+   *
+   *  To disable plain authentication run
+   *  > Strophe.SASLPlain.test = function() {
+   *  >   return false;
+   *  > }
+   *
+   *  See <SASL mechanisms> for a list of available mechanisms.
+   *
+   *  Parameters:
+   *    (Strophe.Connection) connection - Target Connection.
+   *
+   *  Returns:
+   *    (Boolean) If mechanism was able to run.
+   */
+  /* jshint unused:false */
+  test: function(connection) {
+    return true;
+  },
+  /* jshint unused:true */
+
+  /** PrivateFunction: onStart
+   *  Called before starting mechanism on some connection.
+   *
+   *  Parameters:
+   *    (Strophe.Connection) connection - Target Connection.
+   */
+  onStart: function(connection)
+  {
+    this._connection = connection;
+  },
+
+  /** PrivateFunction: onChallenge
+   *  Called by protocol implementation on incoming challenge. If client is
+   *  first (isClientFirst == true) challenge will be null on the first call.
+   *
+   *  Parameters:
+   *    (Strophe.Connection) connection - Target Connection.
+   *    (String) challenge - current challenge to handle.
+   *
+   *  Returns:
+   *    (String) Mechanism response.
+   */
+  /* jshint unused:false */
+  onChallenge: function(connection, challenge) {
+    throw new Error("You should implement challenge handling!");
+  },
+  /* jshint unused:true */
+
+  /** PrivateFunction: onFailure
+   *  Protocol informs mechanism implementation about SASL failure.
+   */
+  onFailure: function() {
+    this._connection = null;
+  },
+
+  /** PrivateFunction: onSuccess
+   *  Protocol informs mechanism implementation about SASL success.
+   */
+  onSuccess: function() {
+    this._connection = null;
+  }
+};
+
+  /** Constants: SASL mechanisms
+   *  Available authentication mechanisms
+   *
+   *  Strophe.SASLAnonymous - SASL Anonymous authentication.
+   *  Strophe.SASLPlain - SASL Plain authentication.
+   *  Strophe.SASLMD5 - SASL Digest-MD5 authentication
+   *  Strophe.SASLSHA1 - SASL SCRAM-SHA1 authentication
+   */
+
+// Building SASL callbacks
+
+/** PrivateConstructor: SASLAnonymous
+ *  SASL Anonymous authentication.
+ */
+Strophe.SASLAnonymous = function() {};
+
+Strophe.SASLAnonymous.prototype = new Strophe.SASLMechanism("ANONYMOUS", false, 10);
+
+Strophe.SASLAnonymous.test = function(connection) {
+  return connection.authcid === null;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLAnonymous.prototype.name] = Strophe.SASLAnonymous;
+
+/** PrivateConstructor: SASLPlain
+ *  SASL Plain authentication.
+ */
+Strophe.SASLPlain = function() {};
+
+Strophe.SASLPlain.prototype = new Strophe.SASLMechanism("PLAIN", true, 20);
+
+Strophe.SASLPlain.test = function(connection) {
+  return connection.authcid !== null;
+};
+
+Strophe.SASLPlain.prototype.onChallenge = function(connection) {
+  var auth_str = connection.authzid;
+  auth_str = auth_str + "\u0000";
+  auth_str = auth_str + connection.authcid;
+  auth_str = auth_str + "\u0000";
+  auth_str = auth_str + connection.pass;
+  return auth_str;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLPlain.prototype.name] = Strophe.SASLPlain;
+
+/** PrivateConstructor: SASLSHA1
+ *  SASL SCRAM SHA 1 authentication.
+ */
+Strophe.SASLSHA1 = function() {};
+
+/* TEST:
+ * This is a simple example of a SCRAM-SHA-1 authentication exchange
+ * when the client doesn't support channel bindings (username 'user' and
+ * password 'pencil' are used):
+ *
+ * C: n,,n=user,r=fyko+d2lbbFgONRv9qkxdawL
+ * S: r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j,s=QSXCR+Q6sek8bf92,
+ * i=4096
+ * C: c=biws,r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j,
+ * p=v0X8v3Bz2T0CJGbJQyF0X+HI4Ts=
+ * S: v=rmF9pqV8S7suAoZWja4dJRkFsKQ=
+ *
+ */
+
+Strophe.SASLSHA1.prototype = new Strophe.SASLMechanism("SCRAM-SHA-1", true, 40);
+
+Strophe.SASLSHA1.test = function(connection) {
+  return connection.authcid !== null;
+};
+
+Strophe.SASLSHA1.prototype.onChallenge = function(connection, challenge, test_cnonce) {
+  var cnonce = test_cnonce || MD5.hexdigest(Math.random() * 1234567890);
+
+  var auth_str = "n=" + connection.authcid;
+  auth_str += ",r=";
+  auth_str += cnonce;
+
+  connection._sasl_data.cnonce = cnonce;
+  connection._sasl_data["client-first-message-bare"] = auth_str;
+
+  auth_str = "n,," + auth_str;
+
+  this.onChallenge = function (connection, challenge)
+  {
+    var nonce, salt, iter, Hi, U, U_old, i, k;
+    var clientKey, serverKey, clientSignature;
+    var responseText = "c=biws,";
+    var authMessage = connection._sasl_data["client-first-message-bare"] + "," +
+      challenge + ",";
+    var cnonce = connection._sasl_data.cnonce;
+    var attribMatch = /([a-z]+)=([^,]+)(,|$)/;
+
+    while (challenge.match(attribMatch)) {
+      var matches = challenge.match(attribMatch);
+      challenge = challenge.replace(matches[0], "");
+      switch (matches[1]) {
+      case "r":
+        nonce = matches[2];
+        break;
+      case "s":
+        salt = matches[2];
+        break;
+      case "i":
+        iter = matches[2];
+        break;
+      }
+    }
+
+    if (nonce.substr(0, cnonce.length) !== cnonce) {
+      connection._sasl_data = {};
+      return connection._sasl_failure_cb();
+    }
+
+    responseText += "r=" + nonce;
+    authMessage += responseText;
+
+    salt = Base64.decode(salt);
+    salt += "\x00\x00\x00\x01";
+
+    Hi = U_old = core_hmac_sha1(connection.pass, salt);
+    for (i = 1; i < iter; i++) {
+      U = core_hmac_sha1(connection.pass, binb2str(U_old));
+      for (k = 0; k < 5; k++) {
+        Hi[k] ^= U[k];
+      }
+      U_old = U;
+    }
+    Hi = binb2str(Hi);
+
+    clientKey = core_hmac_sha1(Hi, "Client Key");
+    serverKey = str_hmac_sha1(Hi, "Server Key");
+    clientSignature = core_hmac_sha1(str_sha1(binb2str(clientKey)), authMessage);
+    connection._sasl_data["server-signature"] = b64_hmac_sha1(serverKey, authMessage);
+
+    for (k = 0; k < 5; k++) {
+      clientKey[k] ^= clientSignature[k];
+    }
+
+    responseText += ",p=" + Base64.encode(binb2str(clientKey));
+
+    return responseText;
+  }.bind(this);
+
+  return auth_str;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLSHA1.prototype.name] = Strophe.SASLSHA1;
+
+/** PrivateConstructor: SASLMD5
+ *  SASL DIGEST MD5 authentication.
+ */
+Strophe.SASLMD5 = function() {};
+
+Strophe.SASLMD5.prototype = new Strophe.SASLMechanism("DIGEST-MD5", false, 30);
+
+Strophe.SASLMD5.test = function(connection) {
+  return connection.authcid !== null;
+};
+
+/** PrivateFunction: _quote
+ *  _Private_ utility function to backslash escape and quote strings.
+ *
+ *  Parameters:
+ *    (String) str - The string to be quoted.
+ *
+ *  Returns:
+ *    quoted string
+ */
+Strophe.SASLMD5.prototype._quote = function (str)
+  {
+    return '"' + str.replace(/\\/g, "\\\\").replace(/"/g, '\\"') + '"';
+    //" end string workaround for emacs
+  };
+
+
+Strophe.SASLMD5.prototype.onChallenge = function(connection, challenge, test_cnonce) {
+  var attribMatch = /([a-z]+)=("[^"]+"|[^,"]+)(?:,|$)/;
+  var cnonce = test_cnonce || MD5.hexdigest("" + (Math.random() * 1234567890));
+  var realm = "";
+  var host = null;
+  var nonce = "";
+  var qop = "";
+  var matches;
+
+  while (challenge.match(attribMatch)) {
+    matches = challenge.match(attribMatch);
+    challenge = challenge.replace(matches[0], "");
+    matches[2] = matches[2].replace(/^"(.+)"$/, "$1");
+    switch (matches[1]) {
+    case "realm":
+      realm = matches[2];
+      break;
+    case "nonce":
+      nonce = matches[2];
+      break;
+    case "qop":
+      qop = matches[2];
+      break;
+    case "host":
+      host = matches[2];
+      break;
+    }
+  }
+
+  var digest_uri = connection.servtype + "/" + connection.domain;
+  if (host !== null) {
+    digest_uri = digest_uri + "/" + host;
+  }
+
+  var A1 = MD5.hash(connection.authcid +
+                    ":" + realm + ":" + this._connection.pass) +
+    ":" + nonce + ":" + cnonce;
+  var A2 = 'AUTHENTICATE:' + digest_uri;
+
+  var responseText = "";
+  responseText += 'charset=utf-8,';
+  responseText += 'username=' +
+    this._quote(connection.authcid) + ',';
+  responseText += 'realm=' + this._quote(realm) + ',';
+  responseText += 'nonce=' + this._quote(nonce) + ',';
+  responseText += 'nc=00000001,';
+  responseText += 'cnonce=' + this._quote(cnonce) + ',';
+  responseText += 'digest-uri=' + this._quote(digest_uri) + ',';
+  responseText += 'response=' + MD5.hexdigest(MD5.hexdigest(A1) + ":" +
+                                              nonce + ":00000001:" +
+                                              cnonce + ":auth:" +
+                                              MD5.hexdigest(A2)) + ",";
+  responseText += 'qop=auth';
+
+  this.onChallenge = function ()
+  {
+    return "";
+  }.bind(this);
+
+  return responseText;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLMD5.prototype.name] = Strophe.SASLMD5;
+
+})(function () {
+    window.Strophe = arguments[0];
+    window.$build = arguments[1];
+    window.$msg = arguments[2];
+    window.$iq = arguments[3];
+    window.$pres = arguments[4];
+});
+
+/*
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+
+/* jshint undef: true, unused: true:, noarg: true, latedef: true */
+/*global window, setTimeout, clearTimeout,
+    XMLHttpRequest, ActiveXObject,
+    Strophe, $build */
+
+
+/** PrivateClass: Strophe.Request
+ *  _Private_ helper class that provides a cross implementation abstraction
+ *  for a BOSH related XMLHttpRequest.
+ *
+ *  The Strophe.Request class is used internally to encapsulate BOSH request
+ *  information.  It is not meant to be used from user's code.
+ */
+
+/** PrivateConstructor: Strophe.Request
+ *  Create and initialize a new Strophe.Request object.
+ *
+ *  Parameters:
+ *    (XMLElement) elem - The XML data to be sent in the request.
+ *    (Function) func - The function that will be called when the
+ *      XMLHttpRequest readyState changes.
+ *    (Integer) rid - The BOSH rid attribute associated with this request.
+ *    (Integer) sends - The number of times this same request has been
+ *      sent.
+ */
+Strophe.Request = function (elem, func, rid, sends)
+{
+    this.id = ++Strophe._requestId;
+    this.xmlData = elem;
+    this.data = Strophe.serialize(elem);
+    // save original function in case we need to make a new request
+    // from this one.
+    this.origFunc = func;
+    this.func = func;
+    this.rid = rid;
+    this.date = NaN;
+    this.sends = sends || 0;
+    this.abort = false;
+    this.dead = null;
+
+    this.age = function () {
+        if (!this.date) { return 0; }
+        var now = new Date();
+        return (now - this.date) / 1000;
+    };
+    this.timeDead = function () {
+        if (!this.dead) { return 0; }
+        var now = new Date();
+        return (now - this.dead) / 1000;
+    };
+    this.xhr = this._newXHR();
+};
+
+Strophe.Request.prototype = {
+    /** PrivateFunction: getResponse
+     *  Get a response from the underlying XMLHttpRequest.
+     *
+     *  This function attempts to get a response from the request and checks
+     *  for errors.
+     *
+     *  Throws:
+     *    "parsererror" - A parser error occured.
+     *
+     *  Returns:
+     *    The DOM element tree of the response.
+     */
+    getResponse: function ()
+    {
+        var node = null;
+        if (this.xhr.responseXML && this.xhr.responseXML.documentElement) {
+            node = this.xhr.responseXML.documentElement;
+            if (node.tagName == "parsererror") {
+                Strophe.error("invalid response received");
+                Strophe.error("responseText: " + this.xhr.responseText);
+                Strophe.error("responseXML: " +
+                              Strophe.serialize(this.xhr.responseXML));
+                throw "parsererror";
+            }
+        } else if (this.xhr.responseText) {
+            Strophe.error("invalid response received");
+            Strophe.error("responseText: " + this.xhr.responseText);
+            Strophe.error("responseXML: " +
+                          Strophe.serialize(this.xhr.responseXML));
+        }
+
+        return node;
+    },
+
+    /** PrivateFunction: _newXHR
+     *  _Private_ helper function to create XMLHttpRequests.
+     *
+     *  This function creates XMLHttpRequests across all implementations.
+     *
+     *  Returns:
+     *    A new XMLHttpRequest.
+     */
+    _newXHR: function ()
+    {
+        var xhr = null;
+        if (window.XMLHttpRequest) {
+            xhr = new XMLHttpRequest();
+            if (xhr.overrideMimeType) {
+                xhr.overrideMimeType("text/xml");
+            }
+        } else if (window.ActiveXObject) {
+            xhr = new ActiveXObject("Microsoft.XMLHTTP");
+        }
+
+        // use Function.bind() to prepend ourselves as an argument
+        xhr.onreadystatechange = this.func.bind(null, this);
+
+        return xhr;
+    }
+};
+
+/** Class: Strophe.Bosh
+ *  _Private_ helper class that handles BOSH Connections
+ *
+ *  The Strophe.Bosh class is used internally by Strophe.Connection
+ *  to encapsulate BOSH sessions. It is not meant to be used from user's code.
+ */
+
+/** File: bosh.js
+ *  A JavaScript library to enable BOSH in Strophejs.
+ *
+ *  this library uses Bidirectional-streams Over Synchronous HTTP (BOSH)
+ *  to emulate a persistent, stateful, two-way connection to an XMPP server.
+ *  More information on BOSH can be found in XEP 124.
+ */
+
+/** PrivateConstructor: Strophe.Bosh
+ *  Create and initialize a Strophe.Bosh object.
+ *
+ *  Parameters:
+ *    (Strophe.Connection) connection - The Strophe.Connection that will use BOSH.
+ *
+ *  Returns:
+ *    A new Strophe.Bosh object.
+ */
+Strophe.Bosh = function(connection) {
+    this._conn = connection;
+    /* request id for body tags */
+    this.rid = Math.floor(Math.random() * 4294967295);
+    /* The current session ID. */
+    this.sid = null;
+
+    // default BOSH values
+    this.hold = 1;
+    this.wait = 60;
+    this.window = 5;
+
+    this._requests = [];
+};
+
+Strophe.Bosh.prototype = {
+    /** Variable: strip
+     *
+     *  BOSH-Connections will have all stanzas wrapped in a <body> tag when
+     *  passed to <Strophe.Connection.xmlInput> or <Strophe.Connection.xmlOutput>.
+     *  To strip this tag, User code can set <Strophe.Bosh.strip> to "body":
+     *
+     *  > Strophe.Bosh.prototype.strip = "body";
+     *
+     *  This will enable stripping of the body tag in both
+     *  <Strophe.Connection.xmlInput> and <Strophe.Connection.xmlOutput>.
+     */
+    strip: null,
+
+    /** PrivateFunction: _buildBody
+     *  _Private_ helper function to generate the <body/> wrapper for BOSH.
+     *
+     *  Returns:
+     *    A Strophe.Builder with a <body/> element.
+     */
+    _buildBody: function ()
+    {
+        var bodyWrap = $build('body', {
+            rid: this.rid++,
+            xmlns: Strophe.NS.HTTPBIND
+        });
+
+        if (this.sid !== null) {
+            bodyWrap.attrs({sid: this.sid});
+        }
+
+        return bodyWrap;
+    },
+
+    /** PrivateFunction: _reset
+     *  Reset the connection.
+     *
+     *  This function is called by the reset function of the Strophe Connection
+     */
+    _reset: function ()
+    {
+        this.rid = Math.floor(Math.random() * 4294967295);
+        this.sid = null;
+        
+        jQuery(document).trigger('ridChange', {rid: this.rid});
+    },
+
+    /** PrivateFunction: _connect
+     *  _Private_ function that initializes the BOSH connection.
+     *
+     *  Creates and sends the Request that initializes the BOSH connection.
+     */
+    _connect: function (wait, hold, route)
+    {
+        this.wait = wait || this.wait;
+        this.hold = hold || this.hold;
+
+        // build the body tag
+        var body = this._buildBody().attrs({
+            to: this._conn.domain,
+            "xml:lang": "en",
+            wait: this.wait,
+            hold: this.hold,
+            content: "text/xml; charset=utf-8",
+            ver: "1.6",
+            "xmpp:version": "1.0",
+            "xmlns:xmpp": Strophe.NS.BOSH
+        });
+
+        if(route){
+            body.attrs({
+                route: route
+            });
+        }
+
+        var _connect_cb = this._conn._connect_cb;
+
+        this._requests.push(
+            new Strophe.Request(body.tree(),
+                                this._onRequestStateChange.bind(
+                                    this, _connect_cb.bind(this._conn)),
+                                body.tree().getAttribute("rid")));
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _attach
+     *  Attach to an already created and authenticated BOSH session.
+     *
+     *  This function is provided to allow Strophe to attach to BOSH
+     *  sessions which have been created externally, perhaps by a Web
+     *  application.  This is often used to support auto-login type features
+     *  without putting user credentials into the page.
+     *
+     *  Parameters:
+     *    (String) jid - The full JID that is bound by the session.
+     *    (String) sid - The SID of the BOSH session.
+     *    (String) rid - The current RID of the BOSH session.  This RID
+     *      will be used by the next request.
+     *    (Function) callback The connect callback function.
+     *    (Integer) wait - The optional HTTPBIND wait value.  This is the
+     *      time the server will wait before returning an empty result for
+     *      a request.  The default setting of 60 seconds is recommended.
+     *      Other settings will require tweaks to the Strophe.TIMEOUT value.
+     *    (Integer) hold - The optional HTTPBIND hold value.  This is the
+     *      number of connections the server will hold at one time.  This
+     *      should almost always be set to 1 (the default).
+     *    (Integer) wind - The optional HTTBIND window value.  This is the
+     *      allowed range of request ids that are valid.  The default is 5.
+     */
+    _attach: function (jid, sid, rid, callback, wait, hold, wind)
+    {
+        this._conn.jid = jid;
+        this.sid = sid;
+        this.rid = rid;
+
+        this._conn.connect_callback = callback;
+
+        this._conn.domain = Strophe.getDomainFromJid(this._conn.jid);
+
+        this._conn.authenticated = true;
+        this._conn.connected = true;
+
+        this.wait = wait || this.wait;
+        this.hold = hold || this.hold;
+        this.window = wind || this.window;
+
+        this._conn._changeConnectStatus(Strophe.Status.ATTACHED, null);
+    },
+
+    /** PrivateFunction: _connect_cb
+     *  _Private_ handler for initial connection request.
+     *
+     *  This handler is used to process the Bosh-part of the initial request.
+     *  Parameters:
+     *    (Strophe.Request) bodyWrap - The received stanza.
+     */
+    _connect_cb: function (bodyWrap)
+    {
+        var typ = bodyWrap.getAttribute("type");
+        var cond, conflict;
+        if (typ !== null && typ == "terminate") {
+            // an error occurred
+            Strophe.error("BOSH-Connection failed: " + cond);
+            cond = bodyWrap.getAttribute("condition");
+            conflict = bodyWrap.getElementsByTagName("conflict");
+            if (cond !== null) {
+                if (cond == "remote-stream-error" && conflict.length > 0) {
+                    cond = "conflict";
+                }
+                this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, cond);
+            } else {
+                this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "unknown");
+            }
+            this._conn._doDisconnect();
+            return Strophe.Status.CONNFAIL;
+        }
+
+        // check to make sure we don't overwrite these if _connect_cb is
+        // called multiple times in the case of missing stream:features
+        if (!this.sid) {
+            this.sid = bodyWrap.getAttribute("sid");
+        }
+        var wind = bodyWrap.getAttribute('requests');
+        if (wind) { this.window = parseInt(wind, 10); }
+        var hold = bodyWrap.getAttribute('hold');
+        if (hold) { this.hold = parseInt(hold, 10); }
+        var wait = bodyWrap.getAttribute('wait');
+        if (wait) { this.wait = parseInt(wait, 10); }
+    },
+
+    /** PrivateFunction: _disconnect
+     *  _Private_ part of Connection.disconnect for Bosh
+     *
+     *  Parameters:
+     *    (Request) pres - This stanza will be sent before disconnecting.
+     */
+    _disconnect: function (pres)
+    {
+        this._sendTerminate(pres);
+    },
+
+    /** PrivateFunction: _doDisconnect
+     *  _Private_ function to disconnect.
+     *
+     *  Resets the SID and RID.
+     */
+    _doDisconnect: function ()
+    {
+        this.sid = null;
+        this.rid = Math.floor(Math.random() * 4294967295);
+        
+        jQuery(document).trigger('ridChange', {rid: this.rid});
+    },
+
+    /** PrivateFunction: _emptyQueue
+     * _Private_ function to check if the Request queue is empty.
+     *
+     *  Returns:
+     *    True, if there are no Requests queued, False otherwise.
+     */
+    _emptyQueue: function ()
+    {
+        return this._requests.length === 0;
+    },
+
+    /** PrivateFunction: _hitError
+     *  _Private_ function to handle the error count.
+     *
+     *  Requests are resent automatically until their error count reaches
+     *  5.  Each time an error is encountered, this function is called to
+     *  increment the count and disconnect if the count is too high.
+     *
+     *  Parameters:
+     *    (Integer) reqStatus - The request status.
+     */
+    _hitError: function (reqStatus)
+    {
+        this.errors++;
+        Strophe.warn("request errored, status: " + reqStatus +
+                     ", number of errors: " + this.errors);
+        if (this.errors > 4) {
+            this._onDisconnectTimeout();
+        }
+    },
+
+    /** PrivateFunction: _no_auth_received
+     *
+     * Called on stream start/restart when no stream:features
+     * has been received and sends a blank poll request.
+     */
+    _no_auth_received: function (_callback)
+    {
+        if (_callback) {
+            _callback = _callback.bind(this._conn);
+        } else {
+            _callback = this._conn._connect_cb.bind(this._conn);
+        }
+        var body = this._buildBody();
+        this._requests.push(
+                new Strophe.Request(body.tree(),
+                    this._onRequestStateChange.bind(
+                        this, _callback.bind(this._conn)),
+                    body.tree().getAttribute("rid")));
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _onDisconnectTimeout
+     *  _Private_ timeout handler for handling non-graceful disconnection.
+     *
+     *  Cancels all remaining Requests and clears the queue.
+     */
+    _onDisconnectTimeout: function ()
+    {
+        var req;
+        while (this._requests.length > 0) {
+            req = this._requests.pop();
+            req.abort = true;
+            req.xhr.abort();
+            // jslint complains, but this is fine. setting to empty func
+            // is necessary for IE6
+            req.xhr.onreadystatechange = function () {}; // jshint ignore:line
+        }
+    },
+
+    /** PrivateFunction: _onIdle
+     *  _Private_ handler called by Strophe.Connection._onIdle
+     *
+     *  Sends all queued Requests or polls with empty Request if there are none.
+     */
+    _onIdle: function () {
+        var data = this._conn._data;
+
+        // if no requests are in progress, poll
+        if (this._conn.authenticated && this._requests.length === 0 &&
+            data.length === 0 && !this._conn.disconnecting) {
+            Strophe.info("no requests during idle cycle, sending " +
+                         "blank request");
+            data.push(null);
+        }
+
+        if (this._requests.length < 2 && data.length > 0 &&
+            !this._conn.paused) {
+            var body = this._buildBody();
+            for (var i = 0; i < data.length; i++) {
+                if (data[i] !== null) {
+                    if (data[i] === "restart") {
+                        body.attrs({
+                            to: this._conn.domain,
+                            "xml:lang": "en",
+                            "xmpp:restart": "true",
+                            "xmlns:xmpp": Strophe.NS.BOSH
+                        });
+                    } else {
+                        body.cnode(data[i]).up();
+                    }
+                }
+            }
+            delete this._conn._data;
+            this._conn._data = [];
+            this._requests.push(
+                new Strophe.Request(body.tree(),
+                                    this._onRequestStateChange.bind(
+                                        this, this._conn._dataRecv.bind(this._conn)),
+                                    body.tree().getAttribute("rid")));
+            this._processRequest(this._requests.length - 1);
+        }
+
+        if (this._requests.length > 0) {
+            var time_elapsed = this._requests[0].age();
+            if (this._requests[0].dead !== null) {
+                if (this._requests[0].timeDead() >
+                    Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait)) {
+                    this._throttledRequestHandler();
+                }
+            }
+
+            if (time_elapsed > Math.floor(Strophe.TIMEOUT * this.wait)) {
+                Strophe.warn("Request " +
+                             this._requests[0].id +
+                             " timed out, over " + Math.floor(Strophe.TIMEOUT * this.wait) +
+                             " seconds since last activity");
+                this._throttledRequestHandler();
+            }
+        }
+    },
+
+    /** PrivateFunction: _onRequestStateChange
+     *  _Private_ handler for Strophe.Request state changes.
+     *
+     *  This function is called when the XMLHttpRequest readyState changes.
+     *  It contains a lot of error handling logic for the many ways that
+     *  requests can fail, and calls the request callback when requests
+     *  succeed.
+     *
+     *  Parameters:
+     *    (Function) func - The handler for the request.
+     *    (Strophe.Request) req - The request that is changing readyState.
+     */
+    _onRequestStateChange: function (func, req)
+    {
+        Strophe.debug("request id " + req.id +
+                      "." + req.sends + " state changed to " +
+                      req.xhr.readyState);
+
+        if (req.abort) {
+            req.abort = false;
+            return;
+        }
+
+        if(req.xhr.readyState == 2){ 
+           jQuery(document).trigger('ridChange', {rid: Number(req.rid)+1});
+        }
+        
+        // request complete
+        var reqStatus;
+        if (req.xhr.readyState == 4) {
+            reqStatus = 0;
+            try {
+                reqStatus = req.xhr.status;
+            } catch (e) {
+                // ignore errors from undefined status attribute.  works
+                // around a browser bug
+            }
+
+            if (typeof(reqStatus) == "undefined") {
+                reqStatus = 0;
+            }
+
+            if (this.disconnecting) {
+                if (reqStatus >= 400) {
+                    this._hitError(reqStatus);
+                    return;
+                }
+            }
+
+            var reqIs0 = (this._requests[0] == req);
+            var reqIs1 = (this._requests[1] == req);
+
+            if ((reqStatus > 0 && reqStatus < 500) || req.sends > 5) {
+                // remove from internal queue
+                this._removeRequest(req);
+                Strophe.debug("request id " +
+                              req.id +
+                              " should now be removed");
+            }
+
+            // request succeeded
+            if (reqStatus == 200) {
+                // if request 1 finished, or request 0 finished and request
+                // 1 is over Strophe.SECONDARY_TIMEOUT seconds old, we need to
+                // restart the other - both will be in the first spot, as the
+                // completed request has been removed from the queue already
+                if (reqIs1 ||
+                    (reqIs0 && this._requests.length > 0 &&
+                     this._requests[0].age() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait))) {
+                    this._restartRequest(0);
+                }
+                // call handler
+                Strophe.debug("request id " +
+                              req.id + "." +
+                              req.sends + " got 200");
+                func(req);
+                this.errors = 0;
+            } else {
+                Strophe.error("request id " +
+                              req.id + "." +
+                              req.sends + " error " + reqStatus +
+                              " happened");
+                if (reqStatus === 0 ||
+                    (reqStatus >= 400 && reqStatus < 600) ||
+                    reqStatus >= 12000) {
+                    this._hitError(reqStatus);
+                    if (reqStatus >= 400 && reqStatus < 500) {
+                        this._conn._changeConnectStatus(Strophe.Status.DISCONNECTING,
+                                                  null);
+                        this._conn._doDisconnect();
+                    }
+                }
+            }
+
+            if (!((reqStatus > 0 && reqStatus < 500) ||
+                  req.sends > 5)) {
+                this._throttledRequestHandler();
+            }
+        }
+    },
+
+    /** PrivateFunction: _processRequest
+     *  _Private_ function to process a request in the queue.
+     *
+     *  This function takes requests off the queue and sends them and
+     *  restarts dead requests.
+     *
+     *  Parameters:
+     *    (Integer) i - The index of the request in the queue.
+     */
+    _processRequest: function (i)
+    {
+        var self = this;
+        var req = this._requests[i];
+        var reqStatus = -1;
+
+        try {
+            if (req.xhr.readyState == 4) {
+                reqStatus = req.xhr.status;
+            }
+        } catch (e) {
+            Strophe.error("caught an error in _requests[" + i +
+                          "], reqStatus: " + reqStatus);
+        }
+
+        if (typeof(reqStatus) == "undefined") {
+            reqStatus = -1;
+        }
+
+        // make sure we limit the number of retries
+        if (req.sends > this.maxRetries) {
+            this._onDisconnectTimeout();
+            return;
+        }
+
+        var time_elapsed = req.age();
+        var primaryTimeout = (!isNaN(time_elapsed) &&
+                              time_elapsed > Math.floor(Strophe.TIMEOUT * this.wait));
+        var secondaryTimeout = (req.dead !== null &&
+                                req.timeDead() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait));
+        var requestCompletedWithServerError = (req.xhr.readyState == 4 &&
+                                               (reqStatus < 1 ||
+                                                reqStatus >= 500));
+        if (primaryTimeout || secondaryTimeout ||
+            requestCompletedWithServerError) {
+            if (secondaryTimeout) {
+                Strophe.error("Request " +
+                              this._requests[i].id +
+                              " timed out (secondary), restarting");
+            }
+            req.abort = true;
+            req.xhr.abort();
+            // setting to null fails on IE6, so set to empty function
+            req.xhr.onreadystatechange = function () {};
+            this._requests[i] = new Strophe.Request(req.xmlData,
+                                                    req.origFunc,
+                                                    req.rid,
+                                                    req.sends);
+            req = this._requests[i];
+        }
+
+        if (req.xhr.readyState === 0) {
+            Strophe.debug("request id " + req.id +
+                          "." + req.sends + " posting");
+
+            try {
+                req.xhr.open("POST", this._conn.service, this._conn.options.sync ? false : true);
+            } catch (e2) {
+                Strophe.error("XHR open failed.");
+                if (!this._conn.connected) {
+                    this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,
+                                              "bad-service");
+                }
+                this._conn.disconnect();
+                return;
+            }
+
+            // Fires the XHR request -- may be invoked immediately
+            // or on a gradually expanding retry window for reconnects
+            var sendFunc = function () {
+                req.date = new Date();
+                if (self._conn.options.customHeaders){
+                    var headers = self._conn.options.customHeaders;
+                    for (var header in headers) {
+                        if (headers.hasOwnProperty(header)) {
+                            req.xhr.setRequestHeader(header, headers[header]);
+                        }
+                    }
+                }
+                req.xhr.send(req.data);
+            };
+
+            // Implement progressive backoff for reconnects --
+            // First retry (send == 1) should also be instantaneous
+            if (req.sends > 1) {
+                // Using a cube of the retry number creates a nicely
+                // expanding retry window
+                var backoff = Math.min(Math.floor(Strophe.TIMEOUT * this.wait),
+                                       Math.pow(req.sends, 3)) * 1000;
+                setTimeout(sendFunc, backoff);
+            } else {
+                sendFunc();
+            }
+
+            req.sends++;
+
+            if (this._conn.xmlOutput !== Strophe.Connection.prototype.xmlOutput) {
+                if (req.xmlData.nodeName === this.strip && req.xmlData.childNodes.length) {
+                    this._conn.xmlOutput(req.xmlData.childNodes[0]);
+                } else {
+                    this._conn.xmlOutput(req.xmlData);
+                }
+            }
+            if (this._conn.rawOutput !== Strophe.Connection.prototype.rawOutput) {
+                this._conn.rawOutput(req.data);
+            }
+        } else {
+            Strophe.debug("_processRequest: " +
+                          (i === 0 ? "first" : "second") +
+                          " request has readyState of " +
+                          req.xhr.readyState);
+        }
+    },
+
+    /** PrivateFunction: _removeRequest
+     *  _Private_ function to remove a request from the queue.
+     *
+     *  Parameters:
+     *    (Strophe.Request) req - The request to remove.
+     */
+    _removeRequest: function (req)
+    {
+        Strophe.debug("removing request");
+
+        var i;
+        for (i = this._requests.length - 1; i >= 0; i--) {
+            if (req == this._requests[i]) {
+                this._requests.splice(i, 1);
+            }
+        }
+
+        // IE6 fails on setting to null, so set to empty function
+        req.xhr.onreadystatechange = function () {};
+
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _restartRequest
+     *  _Private_ function to restart a request that is presumed dead.
+     *
+     *  Parameters:
+     *    (Integer) i - The index of the request in the queue.
+     */
+    _restartRequest: function (i)
+    {
+        var req = this._requests[i];
+        if (req.dead === null) {
+            req.dead = new Date();
+        }
+
+        this._processRequest(i);
+    },
+
+    /** PrivateFunction: _reqToData
+     * _Private_ function to get a stanza out of a request.
+     *
+     * Tries to extract a stanza out of a Request Object.
+     * When this fails the current connection will be disconnected.
+     *
+     *  Parameters:
+     *    (Object) req - The Request.
+     *
+     *  Returns:
+     *    The stanza that was passed.
+     */
+    _reqToData: function (req)
+    {
+        try {
+            return req.getResponse();
+        } catch (e) {
+            if (e != "parsererror") { throw e; }
+            this._conn.disconnect("strophe-parsererror");
+        }
+    },
+
+    /** PrivateFunction: _sendTerminate
+     *  _Private_ function to send initial disconnect sequence.
+     *
+     *  This is the first step in a graceful disconnect.  It sends
+     *  the BOSH server a terminate body and includes an unavailable
+     *  presence if authentication has completed.
+     */
+    _sendTerminate: function (pres)
+    {
+        Strophe.info("_sendTerminate was called");
+        var body = this._buildBody().attrs({type: "terminate"});
+
+        if (pres) {
+            body.cnode(pres.tree());
+        }
+
+        var req = new Strophe.Request(body.tree(),
+                                      this._onRequestStateChange.bind(
+                                          this, this._conn._dataRecv.bind(this._conn)),
+                                      body.tree().getAttribute("rid"));
+
+        this._requests.push(req);
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _send
+     *  _Private_ part of the Connection.send function for BOSH
+     *
+     * Just triggers the RequestHandler to send the messages that are in the queue
+     */
+    _send: function () {
+        clearTimeout(this._conn._idleTimeout);
+        this._throttledRequestHandler();
+        this._conn._idleTimeout = setTimeout(this._conn._onIdle.bind(this._conn), 100);
+    },
+
+    /** PrivateFunction: _sendRestart
+     *
+     *  Send an xmpp:restart stanza.
+     */
+    _sendRestart: function ()
+    {
+        this._throttledRequestHandler();
+        clearTimeout(this._conn._idleTimeout);
+    },
+
+    /** PrivateFunction: _throttledRequestHandler
+     *  _Private_ function to throttle requests to the connection window.
+     *
+     *  This function makes sure we don't send requests so fast that the
+     *  request ids overflow the connection window in the case that one
+     *  request died.
+     */
+    _throttledRequestHandler: function ()
+    {
+        if (!this._requests) {
+            Strophe.debug("_throttledRequestHandler called with " +
+                          "undefined requests");
+        } else {
+            Strophe.debug("_throttledRequestHandler called with " +
+                          this._requests.length + " requests");
+        }
+
+        if (!this._requests || this._requests.length === 0) {
+            return;
+        }
+
+        if (this._requests.length > 0) {
+            this._processRequest(0);
+        }
+
+        if (this._requests.length > 1 &&
+            Math.abs(this._requests[0].rid -
+                     this._requests[1].rid) < this.window) {
+            this._processRequest(1);
+        }
+    }
+};
+
+/*
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+
+/* jshint undef: true, unused: true:, noarg: true, latedef: true */
+/*global document, window, clearTimeout, WebSocket,
+    DOMParser, Strophe, $build */
+
+/** Class: Strophe.WebSocket
+ *  _Private_ helper class that handles WebSocket Connections
+ *
+ *  The Strophe.WebSocket class is used internally by Strophe.Connection
+ *  to encapsulate WebSocket sessions. It is not meant to be used from user's code.
+ */
+
+/** File: websocket.js
+ *  A JavaScript library to enable XMPP over Websocket in Strophejs.
+ *
+ *  This file implements XMPP over WebSockets for Strophejs.
+ *  If a Connection is established with a Websocket url (ws://...)
+ *  Strophe will use WebSockets.
+ *  For more information on XMPP-over WebSocket see this RFC draft:
+ *  http://tools.ietf.org/html/draft-ietf-xmpp-websocket-00
+ *
+ *  WebSocket support implemented by Andreas Guth (andreas.guth at rwth-aachen.de)
+ */
+
+/** PrivateConstructor: Strophe.Websocket
+ *  Create and initialize a Strophe.WebSocket object.
+ *  Currently only sets the connection Object.
+ *
+ *  Parameters:
+ *    (Strophe.Connection) connection - The Strophe.Connection that will use WebSockets.
+ *
+ *  Returns:
+ *    A new Strophe.WebSocket object.
+ */
+Strophe.Websocket = function(connection) {
+    this._conn = connection;
+    this.strip = "stream:stream";
+
+    var service = connection.service;
+    if (service.indexOf("ws:") !== 0 && service.indexOf("wss:") !== 0) {
+        // If the service is not an absolute URL, assume it is a path and put the absolute
+        // URL together from options, current URL and the path.
+        var new_service = "";
+
+        if (connection.options.protocol === "ws" && window.location.protocol !== "https:") {
+            new_service += "ws";
+        } else {
+            new_service += "wss";
+        }
+
+        new_service += "://" + window.location.host;
+
+        if (service.indexOf("/") !== 0) {
+            new_service += window.location.pathname + service;
+        } else {
+            new_service += service;
+        }
+
+        connection.service = new_service;
+    }
+};
+
+Strophe.Websocket.prototype = {
+    /** PrivateFunction: _buildStream
+     *  _Private_ helper function to generate the <stream> start tag for WebSockets
+     *
+     *  Returns:
+     *    A Strophe.Builder with a <stream> element.
+     */
+    _buildStream: function ()
+    {
+        return $build("stream:stream", {
+            "to": this._conn.domain,
+            "xmlns": Strophe.NS.CLIENT,
+            "xmlns:stream": Strophe.NS.STREAM,
+            "version": '1.0'
+        });
+    },
+
+    /** PrivateFunction: _check_streamerror
+     * _Private_ checks a message for stream:error
+     *
+     *  Parameters:
+     *    (Strophe.Request) bodyWrap - The received stanza.
+     *    connectstatus - The ConnectStatus that will be set on error.
+     *  Returns:
+     *     true if there was a streamerror, false otherwise.
+     */
+    _check_streamerror: function (bodyWrap, connectstatus) {
+        var errors = bodyWrap.getElementsByTagName("stream:error");
+        if (errors.length === 0) {
+            return false;
+        }
+        var error = errors[0];
+
+        var condition = "";
+        var text = "";
+
+        var ns = "urn:ietf:params:xml:ns:xmpp-streams";
+        for (var i = 0; i < error.childNodes.length; i++) {
+            var e = error.childNodes[i];
+            if (e.getAttribute("xmlns") !== ns) {
+                break;
+            } if (e.nodeName === "text") {
+                text = e.textContent;
+            } else {
+                condition = e.nodeName;
+            }
+        }
+
+        var errorString = "WebSocket stream error: ";
+
+        if (condition) {
+            errorString += condition;
+        } else {
+            errorString += "unknown";
+        }
+
+        if (text) {
+            errorString += " - " + condition;
+        }
+
+        Strophe.error(errorString);
+
+        // close the connection on stream_error
+        this._conn._changeConnectStatus(connectstatus, condition);
+        this._conn._doDisconnect();
+        return true;
+    },
+
+    /** PrivateFunction: _reset
+     *  Reset the connection.
+     *
+     *  This function is called by the reset function of the Strophe Connection.
+     *  Is not needed by WebSockets.
+     */
+    _reset: function ()
+    {
+        return;
+    },
+
+    /** PrivateFunction: _connect
+     *  _Private_ function called by Strophe.Connection.connect
+     *
+     *  Creates a WebSocket for a connection and assigns Callbacks to it.
+     *  Does nothing if there already is a WebSocket.
+     */
+    _connect: function () {
+        // Ensure that there is no open WebSocket from a previous Connection.
+        this._closeSocket();
+
+        // Create the new WobSocket
+        this.socket = new WebSocket(this._conn.service, "xmpp");
+        this.socket.onopen = this._onOpen.bind(this);
+        this.socket.onerror = this._onError.bind(this);
+        this.socket.onclose = this._onClose.bind(this);
+        this.socket.onmessage = this._connect_cb_wrapper.bind(this);
+    },
+
+    /** PrivateFunction: _connect_cb
+     *  _Private_ function called by Strophe.Connection._connect_cb
+     *
+     * checks for stream:error
+     *
+     *  Parameters:
+     *    (Strophe.Request) bodyWrap - The received stanza.
+     */
+    _connect_cb: function(bodyWrap) {
+        var error = this._check_streamerror(bodyWrap, Strophe.Status.CONNFAIL);
+        if (error) {
+            return Strophe.Status.CONNFAIL;
+        }
+    },
+
+    /** PrivateFunction: _handleStreamStart
+     * _Private_ function that checks the opening stream:stream tag for errors.
+     *
+     * Disconnects if there is an error and returns false, true otherwise.
+     *
+     *  Parameters:
+     *    (Node) message - Stanza containing the stream:stream.
+     */
+    _handleStreamStart: function(message) {
+        var error = false;
+        // Check for errors in the stream:stream tag
+        var ns = message.getAttribute("xmlns");
+        if (typeof ns !== "string") {
+            error = "Missing xmlns in stream:stream";
+        } else if (ns !== Strophe.NS.CLIENT) {
+            error = "Wrong xmlns in stream:stream: " + ns;
+        }
+
+        var ns_stream = message.namespaceURI;
+        if (typeof ns_stream !== "string") {
+            error = "Missing xmlns:stream in stream:stream";
+        } else if (ns_stream !== Strophe.NS.STREAM) {
+            error = "Wrong xmlns:stream in stream:stream: " + ns_stream;
+        }
+
+        var ver = message.getAttribute("version");
+        if (typeof ver !== "string") {
+            error = "Missing version in stream:stream";
+        } else if (ver !== "1.0") {
+            error = "Wrong version in stream:stream: " + ver;
+        }
+
+        if (error) {
+            this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, error);
+            this._conn._doDisconnect();
+            return false;
+        }
+
+        return true;
+    },
+
+    /** PrivateFunction: _connect_cb_wrapper
+     * _Private_ function that handles the first connection messages.
+     *
+     * On receiving an opening stream tag this callback replaces itself with the real
+     * message handler. On receiving a stream error the connection is terminated.
+     */
+    _connect_cb_wrapper: function(message) {
+        if (message.data.indexOf("<stream:stream ") === 0 || message.data.indexOf("<?xml") === 0) {
+            // Strip the XML Declaration, if there is one
+            var data = message.data.replace(/^(<\?.*?\?>\s*)*/, "");
+            if (data === '') return;
+
+            //Make the initial stream:stream selfclosing to parse it without a SAX parser.
+            data = message.data.replace(/<stream:stream (.*[^\/])>/, "<stream:stream $1/>");
+
+            var streamStart = new DOMParser().parseFromString(data, "text/xml").documentElement;
+            this._conn.xmlInput(streamStart);
+            this._conn.rawInput(message.data);
+
+            //_handleStreamSteart will check for XML errors and disconnect on error
+            if (this._handleStreamStart(streamStart)) {
+
+                //_connect_cb will check for stream:error and disconnect on error
+                this._connect_cb(streamStart);
+
+                // ensure received stream:stream is NOT selfclosing and save it for following messages
+                this.streamStart = message.data.replace(/^<stream:(.*)\/>$/, "<stream:$1>");
+            }
+        } else if (message.data === "</stream:stream>") {
+            this._conn.rawInput(message.data);
+            this._conn.xmlInput(document.createElement("stream:stream"));
+            this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "Received closing stream");
+            this._conn._doDisconnect();
+            return;
+        } else {
+            var string = this._streamWrap(message.data);
+            var elem = new DOMParser().parseFromString(string, "text/xml").documentElement;
+            this.socket.onmessage = this._onMessage.bind(this);
+            this._conn._connect_cb(elem, null, message.data);
+        }
+    },
+
+    /** PrivateFunction: _disconnect
+     *  _Private_ function called by Strophe.Connection.disconnect
+     *
+     *  Disconnects and sends a last stanza if one is given
+     *
+     *  Parameters:
+     *    (Request) pres - This stanza will be sent before disconnecting.
+     */
+    _disconnect: function (pres)
+    {
+        if (this.socket.readyState !== WebSocket.CLOSED) {
+            if (pres) {
+                this._conn.send(pres);
+            }
+            var close = '</stream:stream>';
+            this._conn.xmlOutput(document.createElement("stream:stream"));
+            this._conn.rawOutput(close);
+            try {
+                this.socket.send(close);
+            } catch (e) {
+                Strophe.info("Couldn't send closing stream tag.");
+            }
+        }
+
+        this._conn._doDisconnect();
+    },
+
+    /** PrivateFunction: _doDisconnect
+     *  _Private_ function to disconnect.
+     *
+     *  Just closes the Socket for WebSockets
+     */
+    _doDisconnect: function ()
+    {
+        Strophe.info("WebSockets _doDisconnect was called");
+        this._closeSocket();
+    },
+
+    /** PrivateFunction _streamWrap
+     *  _Private_ helper function to wrap a stanza in a <stream> tag.
+     *  This is used so Strophe can process stanzas from WebSockets like BOSH
+     */
+    _streamWrap: function (stanza)
+    {
+        return this.streamStart + stanza + '</stream:stream>';
+    },
+
+
+    /** PrivateFunction: _closeSocket
+     *  _Private_ function to close the WebSocket.
+     *
+     *  Closes the socket if it is still open and deletes it
+     */
+    _closeSocket: function ()
+    {
+        if (this.socket) { try {
+            this.socket.close();
+        } catch (e) {} }
+        this.socket = null;
+    },
+
+    /** PrivateFunction: _emptyQueue
+     * _Private_ function to check if the message queue is empty.
+     *
+     *  Returns:
+     *    True, because WebSocket messages are send immediately after queueing.
+     */
+    _emptyQueue: function ()
+    {
+        return true;
+    },
+
+    /** PrivateFunction: _onClose
+     * _Private_ function to handle websockets closing.
+     *
+     * Nothing to do here for WebSockets
+     */
+    _onClose: function() {
+        if(this._conn.connected && !this._conn.disconnecting) {
+            Strophe.error("Websocket closed unexcectedly");
+            this._conn._doDisconnect();
+        } else {
+            Strophe.info("Websocket closed");
+        }
+    },
+
+    /** PrivateFunction: _no_auth_received
+     *
+     * Called on stream start/restart when no stream:features
+     * has been received.
+     */
+    _no_auth_received: function (_callback)
+    {
+        Strophe.error("Server did not send any auth methods");
+        this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "Server did not send any auth methods");
+        if (_callback) {
+            _callback = _callback.bind(this._conn);
+            _callback();
+        }
+        this._conn._doDisconnect();
+    },
+
+    /** PrivateFunction: _onDisconnectTimeout
+     *  _Private_ timeout handler for handling non-graceful disconnection.
+     *
+     *  This does nothing for WebSockets
+     */
+    _onDisconnectTimeout: function () {},
+
+    /** PrivateFunction: _onError
+     * _Private_ function to handle websockets errors.
+     *
+     * Parameters:
+     * (Object) error - The websocket error.
+     */
+    _onError: function(error) {
+        Strophe.error("Websocket error " + error);
+        this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "The WebSocket connection could not be established was disconnected.");
+        this._disconnect();
+    },
+
+    /** PrivateFunction: _onIdle
+     *  _Private_ function called by Strophe.Connection._onIdle
+     *
+     *  sends all queued stanzas
+     */
+    _onIdle: function () {
+        var data = this._conn._data;
+        if (data.length > 0 && !this._conn.paused) {
+            for (var i = 0; i < data.length; i++) {
+                if (data[i] !== null) {
+                    var stanza, rawStanza;
+                    if (data[i] === "restart") {
+                        stanza = this._buildStream();
+                        rawStanza = this._removeClosingTag(stanza);
+                        stanza = stanza.tree();
+                    } else {
+                        stanza = data[i];
+                        rawStanza = Strophe.serialize(stanza);
+                    }
+                    this._conn.xmlOutput(stanza);
+                    this._conn.rawOutput(rawStanza);
+                    this.socket.send(rawStanza);
+                }
+            }
+            this._conn._data = [];
+        }
+    },
+
+    /** PrivateFunction: _onMessage
+     * _Private_ function to handle websockets messages.
+     *
+     * This function parses each of the messages as if they are full documents. [TODO : We may actually want to use a SAX Push parser].
+     *
+     * Since all XMPP traffic starts with "<stream:stream version='1.0' xml:lang='en' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' id='3697395463' from='SERVER'>"
+     * The first stanza will always fail to be parsed...
+     * Addtionnaly, the seconds stanza will always be a <stream:features> with the stream NS defined in the previous stanza... so we need to 'force' the inclusion of the NS in this stanza!
+     *
+     * Parameters:
+     * (string) message - The websocket message.
+     */
+    _onMessage: function(message) {
+        var elem, data;
+        // check for closing stream
+        if (message.data === "</stream:stream>") {
+            var close = "</stream:stream>";
+            this._conn.rawInput(close);
+            this._conn.xmlInput(document.createElement("stream:stream"));
+            if (!this._conn.disconnecting) {
+                this._conn._doDisconnect();
+            }
+            return;
+        } else if (message.data.search("<stream:stream ") === 0) {
+            //Make the initial stream:stream selfclosing to parse it without a SAX parser.
+            data = message.data.replace(/<stream:stream (.*[^\/])>/, "<stream:stream $1/>");
+            elem = new DOMParser().parseFromString(data, "text/xml").documentElement;
+
+            if (!this._handleStreamStart(elem)) {
+                return;
+            }
+        } else {
+            data = this._streamWrap(message.data);
+            elem = new DOMParser().parseFromString(data, "text/xml").documentElement;
+        }
+
+        if (this._check_streamerror(elem, Strophe.Status.ERROR)) {
+            return;
+        }
+
+        //handle unavailable presence stanza before disconnecting
+        if (this._conn.disconnecting &&
+                elem.firstChild.nodeName === "presence" &&
+                elem.firstChild.getAttribute("type") === "unavailable") {
+            this._conn.xmlInput(elem);
+            this._conn.rawInput(Strophe.serialize(elem));
+            // if we are already disconnecting we will ignore the unavailable stanza and
+            // wait for the </stream:stream> tag before we close the connection
+            return;
+        }
+        this._conn._dataRecv(elem, message.data);
+    },
+
+    /** PrivateFunction: _onOpen
+     * _Private_ function to handle websockets connection setup.
+     *
+     * The opening stream tag is sent here.
+     */
+    _onOpen: function() {
+        Strophe.info("Websocket open");
+        var start = this._buildStream();
+        this._conn.xmlOutput(start.tree());
+
+        var startString = this._removeClosingTag(start);
+        this._conn.rawOutput(startString);
+        this.socket.send(startString);
+    },
+
+    /** PrivateFunction: _removeClosingTag
+     *  _Private_ function to Make the first <stream:stream> non-selfclosing
+     *
+     *  Parameters:
+     *      (Object) elem - The <stream:stream> tag.
+     *
+     *  Returns:
+     *      The stream:stream tag as String
+     */
+    _removeClosingTag: function(elem) {
+        var string = Strophe.serialize(elem);
+        string = string.replace(/<(stream:stream .*[^\/])\/>$/, "<$1>");
+        return string;
+    },
+
+    /** PrivateFunction: _reqToData
+     * _Private_ function to get a stanza out of a request.
+     *
+     * WebSockets don't use requests, so the passed argument is just returned.
+     *
+     *  Parameters:
+     *    (Object) stanza - The stanza.
+     *
+     *  Returns:
+     *    The stanza that was passed.
+     */
+    _reqToData: function (stanza)
+    {
+        return stanza;
+    },
+
+    /** PrivateFunction: _send
+     *  _Private_ part of the Connection.send function for WebSocket
+     *
+     * Just flushes the messages that are in the queue
+     */
+    _send: function () {
+        this._conn.flush();
+    },
+
+    /** PrivateFunction: _sendRestart
+     *
+     *  Send an xmpp:restart stanza.
+     */
+    _sendRestart: function ()
+    {
+        clearTimeout(this._conn._idleTimeout);
+        this._conn._onIdle.bind(this._conn)();
+    }
+};
+
+/*!
+ * Source: build/lib/strophe.muc.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+// Generated by CoffeeScript 1.3.3
+/*
+ *Plugin to implement the MUC extension.
+   http://xmpp.org/extensions/xep-0045.html
+ *Previous Author:
+    Nathan Zorn <nathan.zorn at gmail.com>
+ *Complete CoffeeScript rewrite:
+    Andreas Guth <guth at dbis.rwth-aachen.de>
+*/
+
+var Occupant, RoomConfig, XmppRoom,
+  __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+
+Strophe.addConnectionPlugin('muc', {
+  _connection: null,
+  rooms: {},
+  roomNames: [],
+  /*Function 
+  Initialize the MUC plugin. Sets the correct connection object and
+  extends the namesace.
+  */
+
+  init: function(conn) {
+    this._connection = conn;
+    this._muc_handler = null;
+    Strophe.addNamespace('MUC_OWNER', Strophe.NS.MUC + "#owner");
+    Strophe.addNamespace('MUC_ADMIN', Strophe.NS.MUC + "#admin");
+    Strophe.addNamespace('MUC_USER', Strophe.NS.MUC + "#user");
+    return Strophe.addNamespace('MUC_ROOMCONF', Strophe.NS.MUC + "#roomconfig");
+  },
+  /*Function
+  Join a multi-user chat room
+  Parameters:
+  (String) room - The multi-user chat room to join.
+  (String) nick - The nickname to use in the chat room. Optional
+  (Function) msg_handler_cb - The function call to handle messages from the
+  specified chat room.
+  (Function) pres_handler_cb - The function call back to handle presence
+  in the chat room.
+  (Function) roster_cb - The function call to handle roster info in the chat room
+  (String) password - The optional password to use. (password protected
+  rooms only)
+  (Object) history_attrs - Optional attributes for retrieving history
+  (XML DOM Element) extended_presence - Optional XML for extending presence
+  */
+
+  join: function(room, nick, msg_handler_cb, pres_handler_cb, roster_cb, password, history_attrs, extended_presence) {
+    var msg, room_nick, _ref,
+      _this = this;
+    room_nick = this.test_append_nick(room, nick);
+    msg = $pres({
+      from: this._connection.jid,
+      to: room_nick
+    }).c("x", {
+      xmlns: Strophe.NS.MUC
+    });
+    if (history_attrs != null) {
+      msg = msg.c("history", history_attrs).up();
+    }
+    if (password != null) {
+      msg.cnode(Strophe.xmlElement("password", [], password));
+    }
+    if (extended_presence != null) {
+      msg.up().cnode(extended_presence);
+    }
+    if ((_ref = this._muc_handler) == null) {
+      this._muc_handler = this._connection.addHandler(function(stanza) {
+        var from, handler, handlers, id, roomname, x, xmlns, xquery, _i, _len;
+        from = stanza.getAttribute('from');
+        if (!from) {
+          return true;
+        }
+        roomname = from.split("/")[0];
+        if (!_this.rooms[roomname]) {
+          return true;
+        }
+        room = _this.rooms[roomname];
+        handlers = {};
+        if (stanza.nodeName === "message") {
+          handlers = room._message_handlers;
+        } else if (stanza.nodeName === "presence") {
+          xquery = stanza.getElementsByTagName("x");
+          if (xquery.length > 0) {
+            for (_i = 0, _len = xquery.length; _i < _len; _i++) {
+              x = xquery[_i];
+              xmlns = x.getAttribute("xmlns");
+              if (xmlns && xmlns.match(Strophe.NS.MUC)) {
+                handlers = room._presence_handlers;
+                break;
+              }
+            }
+          }
+        }
+        for (id in handlers) {
+          handler = handlers[id];
+          if (!handler(stanza, room)) {
+            delete handlers[id];
+          }
+        }
+        return true;
+      });
+    }
+    if (!this.rooms.hasOwnProperty(room)) {
+      this.rooms[room] = new XmppRoom(this, room, nick, password);
+      this.roomNames.push(room);
+    }
+    if (pres_handler_cb) {
+      this.rooms[room].addHandler('presence', pres_handler_cb);
+    }
+    if (msg_handler_cb) {
+      this.rooms[room].addHandler('message', msg_handler_cb);
+    }
+    if (roster_cb) {
+      this.rooms[room].addHandler('roster', roster_cb);
+    }
+    return this._connection.send(msg);
+  },
+  /*Function
+  Leave a multi-user chat room
+  Parameters:
+  (String) room - The multi-user chat room to leave.
+  (String) nick - The nick name used in the room.
+  (Function) handler_cb - Optional function to handle the successful leave.
+  (String) exit_msg - optional exit message.
+  Returns:
+  iqid - The unique id for the room leave.
+  */
+
+  leave: function(room, nick, handler_cb, exit_msg) {
+    var id, presence, presenceid, room_nick;
+    id = this.roomNames.indexOf(room);
+    delete this.rooms[room];
+    if (id >= 0) {
+      this.roomNames.splice(id, 1);
+      if (this.roomNames.length === 0) {
+        this._connection.deleteHandler(this._muc_handler);
+        this._muc_handler = null;
+      }
+    }
+    room_nick = this.test_append_nick(room, nick);
+    presenceid = this._connection.getUniqueId();
+    presence = $pres({
+      type: "unavailable",
+      id: presenceid,
+      from: this._connection.jid,
+      to: room_nick
+    });
+    if (exit_msg != null) {
+      presence.c("status", exit_msg);
+    }
+    if (handler_cb != null) {
+      this._connection.addHandler(handler_cb, null, "presence", null, presenceid);
+    }
+    this._connection.send(presence);
+    return presenceid;
+  },
+  /*Function
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) nick - The nick name used in the chat room.
+  (String) message - The plaintext message to send to the room.
+  (String) html_message - The message to send to the room with html markup.
+  (String) type - "groupchat" for group chat messages o
+                  "chat" for private chat messages
+  Returns:
+  msgiq - the unique id used to send the message
+  */
+
+  message: function(room, nick, message, html_message, type) {
+    var msg, msgid, parent, room_nick;
+    room_nick = this.test_append_nick(room, nick);
+    type = type || (nick != null ? "chat" : "groupchat");
+    msgid = this._connection.getUniqueId();
+    msg = $msg({
+      to: room_nick,
+      from: this._connection.jid,
+      type: type,
+      id: msgid
+    }).c("body", {
+      xmlns: Strophe.NS.CLIENT
+    }).t(message);
+    msg.up();
+    if (html_message != null) {
+      msg.c("html", {
+        xmlns: Strophe.NS.XHTML_IM
+      }).c("body", {
+        xmlns: Strophe.NS.XHTML
+      }).t(html_message);
+      if (msg.node.childNodes.length === 0) {
+        parent = msg.node.parentNode;
+        msg.up().up();
+        msg.node.removeChild(parent);
+      } else {
+        msg.up().up();
+      }
+    }
+    msg.c("x", {
+      xmlns: "jabber:x:event"
+    }).c("composing");
+    this._connection.send(msg);
+    return msgid;
+  },
+  /*Function
+  Convenience Function to send a Message to all Occupants
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) message - The plaintext message to send to the room.
+  (String) html_message - The message to send to the room with html markup.
+  Returns:
+  msgiq - the unique id used to send the message
+  */
+
+  groupchat: function(room, message, html_message) {
+    return this.message(room, null, message, html_message);
+  },
+  /*Function
+  Send a mediated invitation.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) receiver - The invitation's receiver.
+  (String) reason - Optional reason for joining the room.
+  Returns:
+  msgiq - the unique id used to send the invitation
+  */
+
+  invite: function(room, receiver, reason) {
+    var invitation, msgid;
+    msgid = this._connection.getUniqueId();
+    invitation = $msg({
+      from: this._connection.jid,
+      to: room,
+      id: msgid
+    }).c('x', {
+      xmlns: Strophe.NS.MUC_USER
+    }).c('invite', {
+      to: receiver
+    });
+    if (reason != null) {
+      invitation.c('reason', reason);
+    }
+    this._connection.send(invitation);
+    return msgid;
+  },
+  /*Function
+  Send a direct invitation.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) receiver - The invitation's receiver.
+  (String) reason - Optional reason for joining the room.
+  (String) password - Optional password for the room.
+  Returns:
+  msgiq - the unique id used to send the invitation
+  */
+
+  directInvite: function(room, receiver, reason, password) {
+    var attrs, invitation, msgid;
+    msgid = this._connection.getUniqueId();
+    attrs = {
+      xmlns: 'jabber:x:conference',
+      jid: room
+    };
+    if (reason != null) {
+      attrs.reason = reason;
+    }
+    if (password != null) {
+      attrs.password = password;
+    }
+    invitation = $msg({
+      from: this._connection.jid,
+      to: receiver,
+      id: msgid
+    }).c('x', attrs);
+    this._connection.send(invitation);
+    return msgid;
+  },
+  /*Function
+  Queries a room for a list of occupants
+  (String) room - The multi-user chat room name.
+  (Function) success_cb - Optional function to handle the info.
+  (Function) error_cb - Optional function to handle an error.
+  Returns:
+  id - the unique id used to send the info request
+  */
+
+  queryOccupants: function(room, success_cb, error_cb) {
+    var attrs, info;
+    attrs = {
+      xmlns: Strophe.NS.DISCO_ITEMS
+    };
+    info = $iq({
+      from: this._connection.jid,
+      to: room,
+      type: 'get'
+    }).c('query', attrs);
+    return this._connection.sendIQ(info, success_cb, error_cb);
+  },
+  /*Function
+  Start a room configuration.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (Function) handler_cb - Optional function to handle the config form.
+  Returns:
+  id - the unique id used to send the configuration request
+  */
+
+  configure: function(room, handler_cb, error_cb) {
+    var config, stanza;
+    config = $iq({
+      to: room,
+      type: "get"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    });
+    stanza = config.tree();
+    return this._connection.sendIQ(stanza, handler_cb, error_cb);
+  },
+  /*Function
+  Cancel the room configuration
+  Parameters:
+  (String) room - The multi-user chat room name.
+  Returns:
+  id - the unique id used to cancel the configuration.
+  */
+
+  cancelConfigure: function(room) {
+    var config, stanza;
+    config = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    }).c("x", {
+      xmlns: "jabber:x:data",
+      type: "cancel"
+    });
+    stanza = config.tree();
+    return this._connection.sendIQ(stanza);
+  },
+  /*Function
+  Save a room configuration.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (Array) config- Form Object or an array of form elements used to configure the room.
+  Returns:
+  id - the unique id used to save the configuration.
+  */
+
+  saveConfiguration: function(room, config, success_cb, error_cb) {
+    var conf, iq, stanza, _i, _len;
+    iq = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    });
+    if (config instanceof Form) {
+      config.type = "submit";
+      iq.cnode(config.toXML());
+    } else {
+      iq.c("x", {
+        xmlns: "jabber:x:data",
+        type: "submit"
+      });
+      for (_i = 0, _len = config.length; _i < _len; _i++) {
+        conf = config[_i];
+        iq.cnode(conf).up();
+      }
+    }
+    stanza = iq.tree();
+    return this._connection.sendIQ(stanza, success_cb, error_cb);
+  },
+  /*Function
+  Parameters:
+  (String) room - The multi-user chat room name.
+  Returns:
+  id - the unique id used to create the chat room.
+  */
+
+  createInstantRoom: function(room, success_cb, error_cb) {
+    var roomiq;
+    roomiq = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    }).c("x", {
+      xmlns: "jabber:x:data",
+      type: "submit"
+    });
+    return this._connection.sendIQ(roomiq.tree(), success_cb, error_cb);
+  },
+  /*Function
+  Set the topic of the chat room.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) topic - Topic message.
+  */
+
+  setTopic: function(room, topic) {
+    var msg;
+    msg = $msg({
+      to: room,
+      from: this._connection.jid,
+      type: "groupchat"
+    }).c("subject", {
+      xmlns: "jabber:client"
+    }).t(topic);
+    return this._connection.send(msg.tree());
+  },
+  /*Function
+  Internal Function that Changes the role or affiliation of a member
+  of a MUC room. This function is used by modifyRole and modifyAffiliation.
+  The modification can only be done by a room moderator. An error will be
+  returned if the user doesn't have permission.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (Object) item - Object with nick and role or jid and affiliation attribute
+  (String) reason - Optional reason for the change.
+  (Function) handler_cb - Optional callback for success
+  (Function) error_cb - Optional callback for error
+  Returns:
+  iq - the id of the mode change request.
+  */
+
+  _modifyPrivilege: function(room, item, reason, handler_cb, error_cb) {
+    var iq;
+    iq = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_ADMIN
+    }).cnode(item.node);
+    if (reason != null) {
+      iq.c("reason", reason);
+    }
+    return this._connection.sendIQ(iq.tree(), handler_cb, error_cb);
+  },
+  /*Function
+  Changes the role of a member of a MUC room.
+  The modification can only be done by a room moderator. An error will be
+  returned if the user doesn't have permission.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) nick - The nick name of the user to modify.
+  (String) role - The new role of the user.
+  (String) affiliation - The new affiliation of the user.
+  (String) reason - Optional reason for the change.
+  (Function) handler_cb - Optional callback for success
+  (Function) error_cb - Optional callback for error
+  Returns:
+  iq - the id of the mode change request.
+  */
+
+  modifyRole: function(room, nick, role, reason, handler_cb, error_cb) {
+    var item;
+    item = $build("item", {
+      nick: nick,
+      role: role
+    });
+    return this._modifyPrivilege(room, item, reason, handler_cb, error_cb);
+  },
+  kick: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'none', reason, handler_cb, error_cb);
+  },
+  voice: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'participant', reason, handler_cb, error_cb);
+  },
+  mute: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'visitor', reason, handler_cb, error_cb);
+  },
+  op: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'moderator', reason, handler_cb, error_cb);
+  },
+  deop: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'participant', reason, handler_cb, error_cb);
+  },
+  /*Function
+  Changes the affiliation of a member of a MUC room.
+  The modification can only be done by a room moderator. An error will be
+  returned if the user doesn't have permission.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) jid  - The jid of the user to modify.
+  (String) affiliation - The new affiliation of the user.
+  (String) reason - Optional reason for the change.
+  (Function) handler_cb - Optional callback for success
+  (Function) error_cb - Optional callback for error
+  Returns:
+  iq - the id of the mode change request.
+  */
+
+  modifyAffiliation: function(room, jid, affiliation, reason, handler_cb, error_cb) {
+    var item;
+    item = $build("item", {
+      jid: jid,
+      affiliation: affiliation
+    });
+    return this._modifyPrivilege(room, item, reason, handler_cb, error_cb);
+  },
+  ban: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'outcast', reason, handler_cb, error_cb);
+  },
+  member: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'member', reason, handler_cb, error_cb);
+  },
+  revoke: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'none', reason, handler_cb, error_cb);
+  },
+  owner: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'owner', reason, handler_cb, error_cb);
+  },
+  admin: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'admin', reason, handler_cb, error_cb);
+  },
+  /*Function
+  Change the current users nick name.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) user - The new nick name.
+  */
+
+  changeNick: function(room, user) {
+    var presence, room_nick;
+    room_nick = this.test_append_nick(room, user);
+    presence = $pres({
+      from: this._connection.jid,
+      to: room_nick,
+      id: this._connection.getUniqueId()
+    });
+    return this._connection.send(presence.tree());
+  },
+  /*Function
+  Change the current users status.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) user - The current nick.
+  (String) show - The new show-text.
+  (String) status - The new status-text.
+  */
+
+  setStatus: function(room, user, show, status) {
+    var presence, room_nick;
+    room_nick = this.test_append_nick(room, user);
+    presence = $pres({
+      from: this._connection.jid,
+      to: room_nick
+    });
+    if (show != null) {
+      presence.c('show', show).up();
+    }
+    if (status != null) {
+      presence.c('status', status);
+    }
+    return this._connection.send(presence.tree());
+  },
+  /*Function
+  List all chat room available on a server.
+  Parameters:
+  (String) server - name of chat server.
+  (String) handle_cb - Function to call for room list return.
+  (String) error_cb - Function to call on error.
+  */
+
+  listRooms: function(server, handle_cb, error_cb) {
+    var iq;
+    iq = $iq({
+      to: server,
+      from: this._connection.jid,
+      type: "get"
+    }).c("query", {
+      xmlns: Strophe.NS.DISCO_ITEMS
+    });
+    return this._connection.sendIQ(iq, handle_cb, error_cb);
+  },
+  test_append_nick: function(room, nick) {
+    return room + (nick != null ? "/" + (Strophe.escapeNode(nick)) : "");
+  }
+});
+
+XmppRoom = (function() {
+
+  function XmppRoom(client, name, nick, password) {
+    this.client = client;
+    this.name = name;
+    this.nick = nick;
+    this.password = password;
+    this._roomRosterHandler = __bind(this._roomRosterHandler, this);
+
+    this._addOccupant = __bind(this._addOccupant, this);
+
+    this.roster = {};
+    this._message_handlers = {};
+    this._presence_handlers = {};
+    this._roster_handlers = {};
+    this._handler_ids = 0;
+    if (client.muc) {
+      this.client = client.muc;
+    }
+    this.name = Strophe.getBareJidFromJid(name);
+    this.addHandler('presence', this._roomRosterHandler);
+  }
+
+  XmppRoom.prototype.join = function(msg_handler_cb, pres_handler_cb, roster_cb) {
+    return this.client.join(this.name, this.nick, msg_handler_cb, pres_handler_cb, roster_cb, this.password);
+  };
+
+  XmppRoom.prototype.leave = function(handler_cb, message) {
+    this.client.leave(this.name, this.nick, handler_cb, message);
+    return delete this.client.rooms[this.name];
+  };
+
+  XmppRoom.prototype.message = function(nick, message, html_message, type) {
+    return this.client.message(this.name, nick, message, html_message, type);
+  };
+
+  XmppRoom.prototype.groupchat = function(message, html_message) {
+    return this.client.groupchat(this.name, message, html_message);
+  };
+
+  XmppRoom.prototype.invite = function(receiver, reason) {
+    return this.client.invite(this.name, receiver, reason);
+  };
+
+  XmppRoom.prototype.directInvite = function(receiver, reason) {
+    return this.client.directInvite(this.name, receiver, reason, this.password);
+  };
+
+  XmppRoom.prototype.configure = function(handler_cb) {
+    return this.client.configure(this.name, handler_cb);
+  };
+
+  XmppRoom.prototype.cancelConfigure = function() {
+    return this.client.cancelConfigure(this.name);
+  };
+
+  XmppRoom.prototype.saveConfiguration = function(config) {
+    return this.client.saveConfiguration(this.name, config);
+  };
+
+  XmppRoom.prototype.queryOccupants = function(success_cb, error_cb) {
+    return this.client.queryOccupants(this.name, success_cb, error_cb);
+  };
+
+  XmppRoom.prototype.setTopic = function(topic) {
+    return this.client.setTopic(this.name, topic);
+  };
+
+  XmppRoom.prototype.modifyRole = function(nick, role, reason, success_cb, error_cb) {
+    return this.client.modifyRole(this.name, nick, role, reason, success_cb, error_cb);
+  };
+
+  XmppRoom.prototype.kick = function(nick, reason, handler_cb, error_cb) {
+    return this.client.kick(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.voice = function(nick, reason, handler_cb, error_cb) {
+    return this.client.voice(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.mute = function(nick, reason, handler_cb, error_cb) {
+    return this.client.mute(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.op = function(nick, reason, handler_cb, error_cb) {
+    return this.client.op(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.deop = function(nick, reason, handler_cb, error_cb) {
+    return this.client.deop(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.modifyAffiliation = function(jid, affiliation, reason, success_cb, error_cb) {
+    return this.client.modifyAffiliation(this.name, jid, affiliation, reason, success_cb, error_cb);
+  };
+
+  XmppRoom.prototype.ban = function(jid, reason, handler_cb, error_cb) {
+    return this.client.ban(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.member = function(jid, reason, handler_cb, error_cb) {
+    return this.client.member(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.revoke = function(jid, reason, handler_cb, error_cb) {
+    return this.client.revoke(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.owner = function(jid, reason, handler_cb, error_cb) {
+    return this.client.owner(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.admin = function(jid, reason, handler_cb, error_cb) {
+    return this.client.admin(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.changeNick = function(nick) {
+    this.nick = nick;
+    return this.client.changeNick(this.name, nick);
+  };
+
+  XmppRoom.prototype.setStatus = function(show, status) {
+    return this.client.setStatus(this.name, this.nick, show, status);
+  };
+
+  /*Function
+  Adds a handler to the MUC room.
+    Parameters:
+  (String) handler_type - 'message', 'presence' or 'roster'.
+  (Function) handler - The handler function.
+  Returns:
+  id - the id of handler.
+  */
+
+
+  XmppRoom.prototype.addHandler = function(handler_type, handler) {
+    var id;
+    id = this._handler_ids++;
+    switch (handler_type) {
+      case 'presence':
+        this._presence_handlers[id] = handler;
+        break;
+      case 'message':
+        this._message_handlers[id] = handler;
+        break;
+      case 'roster':
+        this._roster_handlers[id] = handler;
+        break;
+      default:
+        this._handler_ids--;
+        return null;
+    }
+    return id;
+  };
+
+  /*Function
+  Removes a handler from the MUC room.
+  This function takes ONLY ids returned by the addHandler function
+  of this room. passing handler ids returned by connection.addHandler
+  may brake things!
+    Parameters:
+  (number) id - the id of the handler
+  */
+
+
+  XmppRoom.prototype.removeHandler = function(id) {
+    delete this._presence_handlers[id];
+    delete this._message_handlers[id];
+    return delete this._roster_handlers[id];
+  };
+
+  /*Function
+  Creates and adds an Occupant to the Room Roster.
+    Parameters:
+  (Object) data - the data the Occupant is filled with
+  Returns:
+  occ - the created Occupant.
+  */
+
+
+  XmppRoom.prototype._addOccupant = function(data) {
+    var occ;
+    occ = new Occupant(data, this);
+    this.roster[occ.nick] = occ;
+    return occ;
+  };
+
+  /*Function
+  The standard handler that managed the Room Roster.
+    Parameters:
+  (Object) pres - the presence stanza containing user information
+  */
+
+
+  XmppRoom.prototype._roomRosterHandler = function(pres) {
+    var data, handler, id, newnick, nick, _ref;
+    data = XmppRoom._parsePresence(pres);
+    nick = data.nick;
+    newnick = data.newnick || null;
+    switch (data.type) {
+      case 'error':
+        return;
+      case 'unavailable':
+        if (newnick) {
+          data.nick = newnick;
+          if (this.roster[nick] && this.roster[newnick]) {
+            this.roster[nick].update(this.roster[newnick]);
+            this.roster[newnick] = this.roster[nick];
+          }
+          if (this.roster[nick] && !this.roster[newnick]) {
+            this.roster[newnick] = this.roster[nick].update(data);
+          }
+        }
+        delete this.roster[nick];
+        break;
+      default:
+        if (this.roster[nick]) {
+          this.roster[nick].update(data);
+        } else {
+          this._addOccupant(data);
+        }
+    }
+    _ref = this._roster_handlers;
+    for (id in _ref) {
+      handler = _ref[id];
+      if (!handler(this.roster, this)) {
+        delete this._roster_handlers[id];
+      }
+    }
+    return true;
+  };
+
+  /*Function
+  Parses a presence stanza
+    Parameters:
+  (Object) data - the data extracted from the presence stanza
+  */
+
+
+  XmppRoom._parsePresence = function(pres) {
+    var a, c, c2, data, _i, _j, _len, _len1, _ref, _ref1, _ref2, _ref3, _ref4, _ref5, _ref6, _ref7;
+    data = {};
+    a = pres.attributes;
+    data.nick = Strophe.getResourceFromJid(a.from.textContent);
+    data.type = ((_ref = a.type) != null ? _ref.textContent : void 0) || null;
+    data.states = [];
+    _ref1 = pres.childNodes;
+    for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
+      c = _ref1[_i];
+      switch (c.nodeName) {
+        case "status":
+          data.status = c.textContent || null;
+          break;
+        case "show":
+          data.show = c.textContent || null;
+          break;
+        case "x":
+          a = c.attributes;
+          if (((_ref2 = a.xmlns) != null ? _ref2.textContent : void 0) === Strophe.NS.MUC_USER) {
+            _ref3 = c.childNodes;
+            for (_j = 0, _len1 = _ref3.length; _j < _len1; _j++) {
+              c2 = _ref3[_j];
+              switch (c2.nodeName) {
+                case "item":
+                  a = c2.attributes;
+                  data.affiliation = ((_ref4 = a.affiliation) != null ? _ref4.textContent : void 0) || null;
+                  data.role = ((_ref5 = a.role) != null ? _ref5.textContent : void 0) || null;
+                  data.jid = ((_ref6 = a.jid) != null ? _ref6.textContent : void 0) || null;
+                  data.newnick = ((_ref7 = a.nick) != null ? _ref7.textContent : void 0) || null;
+                  break;
+                case "status":
+                  if (c2.attributes.code) {
+                    data.states.push(c2.attributes.code.textContent);
+                  }
+              }
+            }
+          }
+      }
+    }
+    return data;
+  };
+
+  return XmppRoom;
+
+})();
+
+RoomConfig = (function() {
+
+  function RoomConfig(info) {
+    this.parse = __bind(this.parse, this);
+    if (info != null) {
+      this.parse(info);
+    }
+  }
+
+  RoomConfig.prototype.parse = function(result) {
+    var attr, attrs, child, field, identity, query, _i, _j, _k, _len, _len1, _len2, _ref;
+    query = result.getElementsByTagName("query")[0].childNodes;
+    this.identities = [];
+    this.features = [];
+    this.x = [];
+    for (_i = 0, _len = query.length; _i < _len; _i++) {
+      child = query[_i];
+      attrs = child.attributes;
+      switch (child.nodeName) {
+        case "identity":
+          identity = {};
+          for (_j = 0, _len1 = attrs.length; _j < _len1; _j++) {
+            attr = attrs[_j];
+            identity[attr.name] = attr.textContent;
+          }
+          this.identities.push(identity);
+          break;
+        case "feature":
+          this.features.push(attrs["var"].textContent);
+          break;
+        case "x":
+          attrs = child.childNodes[0].attributes;
+          if ((!attrs["var"].textContent === 'FORM_TYPE') || (!attrs.type.textContent === 'hidden')) {
+            break;
+          }
+          _ref = child.childNodes;
+          for (_k = 0, _len2 = _ref.length; _k < _len2; _k++) {
+            field = _ref[_k];
+            if (!(!field.attributes.type)) {
+              continue;
+            }
+            attrs = field.attributes;
+            this.x.push({
+              "var": attrs["var"].textContent,
+              label: attrs.label.textContent || "",
+              value: field.firstChild.textContent || ""
+            });
+          }
+      }
+    }
+    return {
+      "identities": this.identities,
+      "features": this.features,
+      "x": this.x
+    };
+  };
+
+  return RoomConfig;
+
+})();
+
+Occupant = (function() {
+
+  function Occupant(data, room) {
+    this.room = room;
+    this.update = __bind(this.update, this);
+
+    this.admin = __bind(this.admin, this);
+
+    this.owner = __bind(this.owner, this);
+
+    this.revoke = __bind(this.revoke, this);
+
+    this.member = __bind(this.member, this);
+
+    this.ban = __bind(this.ban, this);
+
+    this.modifyAffiliation = __bind(this.modifyAffiliation, this);
+
+    this.deop = __bind(this.deop, this);
+
+    this.op = __bind(this.op, this);
+
+    this.mute = __bind(this.mute, this);
+
+    this.voice = __bind(this.voice, this);
+
+    this.kick = __bind(this.kick, this);
+
+    this.modifyRole = __bind(this.modifyRole, this);
+
+    this.update(data);
+  }
+
+  Occupant.prototype.modifyRole = function(role, reason, success_cb, error_cb) {
+    return this.room.modifyRole(this.nick, role, reason, success_cb, error_cb);
+  };
+
+  Occupant.prototype.kick = function(reason, handler_cb, error_cb) {
+    return this.room.kick(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.voice = function(reason, handler_cb, error_cb) {
+    return this.room.voice(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.mute = function(reason, handler_cb, error_cb) {
+    return this.room.mute(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.op = function(reason, handler_cb, error_cb) {
+    return this.room.op(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.deop = function(reason, handler_cb, error_cb) {
+    return this.room.deop(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.modifyAffiliation = function(affiliation, reason, success_cb, error_cb) {
+    return this.room.modifyAffiliation(this.jid, affiliation, reason, success_cb, error_cb);
+  };
+
+  Occupant.prototype.ban = function(reason, handler_cb, error_cb) {
+    return this.room.ban(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.member = function(reason, handler_cb, error_cb) {
+    return this.room.member(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.revoke = function(reason, handler_cb, error_cb) {
+    return this.room.revoke(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.owner = function(reason, handler_cb, error_cb) {
+    return this.room.owner(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.admin = function(reason, handler_cb, error_cb) {
+    return this.room.admin(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.update = function(data) {
+    this.nick = data.nick || null;
+    this.affiliation = data.affiliation || null;
+    this.role = data.role || null;
+    this.jid = data.jid || null;
+    this.status = data.status || null;
+    this.show = data.show || null;
+    return this;
+  };
+
+  return Occupant;
+
+})();
+/*!
+ * Source: build/lib/strophe.disco.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+/*
+  Copyright 2010, François de Metz <francois at 2metz.fr>
+*/
+
+/**
+ * Disco Strophe Plugin
+ * Implement http://xmpp.org/extensions/xep-0030.html
+ * TODO: manage node hierarchies, and node on info request
+ */
+Strophe.addConnectionPlugin('disco',
+{
+    _connection: null,
+    _identities : [],
+    _features : [],
+    _items : [],
+    /** Function: init
+     * Plugin init
+     *
+     * Parameters:
+     *   (Strophe.Connection) conn - Strophe connection
+     */
+    init: function(conn)
+    {
+    this._connection = conn;
+        this._identities = [];
+        this._features   = [];
+        this._items      = [];
+        // disco info
+        conn.addHandler(this._onDiscoInfo.bind(this), Strophe.NS.DISCO_INFO, 'iq', 'get', null, null);
+        // disco items
+        conn.addHandler(this._onDiscoItems.bind(this), Strophe.NS.DISCO_ITEMS, 'iq', 'get', null, null);
+    },
+    /** Function: addIdentity
+     * See http://xmpp.org/registrar/disco-categories.html
+     * Parameters:
+     *   (String) category - category of identity (like client, automation, etc ...)
+     *   (String) type - type of identity (like pc, web, bot , etc ...)
+     *   (String) name - name of identity in natural language
+     *   (String) lang - lang of name parameter
+     *
+     * Returns:
+     *   Boolean
+     */
+    addIdentity: function(category, type, name, lang)
+    {
+        for (var i=0; i<this._identities.length; i++)
+        {
+            if (this._identities[i].category == category &&
+                this._identities[i].type == type &&
+                this._identities[i].name == name &&
+                this._identities[i].lang == lang)
+            {
+                return false;
+            }
+        }
+        this._identities.push({category: category, type: type, name: name, lang: lang});
+        return true;
+    },
+    /** Function: addFeature
+     *
+     * Parameters:
+     *   (String) var_name - feature name (like jabber:iq:version)
+     *
+     * Returns:
+     *   boolean
+     */
+    addFeature: function(var_name)
+    {
+        for (var i=0; i<this._features.length; i++)
+        {
+             if (this._features[i] == var_name)
+                 return false;
+        }
+        this._features.push(var_name);
+        return true;
+    },
+    /** Function: removeFeature
+     *
+     * Parameters:
+     *   (String) var_name - feature name (like jabber:iq:version)
+     *
+     * Returns:
+     *   boolean
+     */
+    removeFeature: function(var_name)
+    {
+        for (var i=0; i<this._features.length; i++)
+        {
+             if (this._features[i] === var_name){
+                 this._features.splice(i,1)
+                 return true;
+             }
+        }
+        return false;
+    },
+    /** Function: addItem
+     *
+     * Parameters:
+     *   (String) jid
+     *   (String) name
+     *   (String) node
+     *   (Function) call_back
+     *
+     * Returns:
+     *   boolean
+     */
+    addItem: function(jid, name, node, call_back)
+    {
+        if (node && !call_back)
+            return false;
+        this._items.push({jid: jid, name: name, node: node, call_back: call_back});
+        return true;
+    },
+    /** Function: info
+     * Info query
+     *
+     * Parameters:
+     *   (Function) call_back
+     *   (String) jid
+     *   (String) node
+     */
+    info: function(jid, node, success, error, timeout)
+    {
+        var attrs = {xmlns: Strophe.NS.DISCO_INFO};
+        if (node)
+            attrs.node = node;
+
+        var info = $iq({from:this._connection.jid,
+                         to:jid, type:'get'}).c('query', attrs);
+        this._connection.sendIQ(info, success, error, timeout);
+    },
+    /** Function: items
+     * Items query
+     *
+     * Parameters:
+     *   (Function) call_back
+     *   (String) jid
+     *   (String) node
+     */
+    items: function(jid, node, success, error, timeout)
+    {
+        var attrs = {xmlns: Strophe.NS.DISCO_ITEMS};
+        if (node)
+            attrs.node = node;
+
+        var items = $iq({from:this._connection.jid,
+                         to:jid, type:'get'}).c('query', attrs);
+        this._connection.sendIQ(items, success, error, timeout);
+    },
+
+    /** PrivateFunction: _buildIQResult
+     */
+    _buildIQResult: function(stanza, query_attrs)
+    {
+        var id   =  stanza.getAttribute('id');
+        var from = stanza.getAttribute('from');
+        var iqresult = $iq({type: 'result', id: id});
+
+        if (from !== null) {
+            iqresult.attrs({to: from});
+        }
+
+        return iqresult.c('query', query_attrs);
+    },
+
+    /** PrivateFunction: _onDiscoInfo
+     * Called when receive info request
+     */
+    _onDiscoInfo: function(stanza)
+    {
+        var node = stanza.getElementsByTagName('query')[0].getAttribute('node');
+        var attrs = {xmlns: Strophe.NS.DISCO_INFO};
+        if (node)
+        {
+            attrs.node = node;
+        }
+        var iqresult = this._buildIQResult(stanza, attrs);
+        for (var i=0; i<this._identities.length; i++)
+        {
+            var attrs = {category: this._identities[i].category,
+                         type    : this._identities[i].type};
+            if (this._identities[i].name)
+                attrs.name = this._identities[i].name;
+            if (this._identities[i].lang)
+                attrs['xml:lang'] = this._identities[i].lang;
+            iqresult.c('identity', attrs).up();
+        }
+        for (var i=0; i<this._features.length; i++)
+        {
+            iqresult.c('feature', {'var':this._features[i]}).up();
+        }
+        this._connection.send(iqresult.tree());
+        return true;
+    },
+    /** PrivateFunction: _onDiscoItems
+     * Called when receive items request
+     */
+    _onDiscoItems: function(stanza)
+    {
+        var query_attrs = {xmlns: Strophe.NS.DISCO_ITEMS};
+        var node = stanza.getElementsByTagName('query')[0].getAttribute('node');
+        if (node)
+        {
+            query_attrs.node = node;
+            var items = [];
+            for (var i = 0; i < this._items.length; i++)
+            {
+                if (this._items[i].node == node)
+                {
+                    items = this._items[i].call_back(stanza);
+                    break;
+                }
+            }
+        }
+        else
+        {
+            var items = this._items;
+        }
+        var iqresult = this._buildIQResult(stanza, query_attrs);
+        for (var i = 0; i < items.length; i++)
+        {
+            var attrs = {jid:  items[i].jid};
+            if (items[i].name)
+                attrs.name = items[i].name;
+            if (items[i].node)
+                attrs.node = items[i].node;
+            iqresult.c('item', attrs).up();
+        }
+        this._connection.send(iqresult.tree());
+        return true;
+    }
+});
+
+/*!
+ * Source: build/lib/strophe.caps.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+/**
+ * Entity Capabilities (XEP-0115)
+ * 
+ * Depends on disco plugin.
+ * 
+ * See: http://xmpp.org/extensions/xep-0115.html
+ * 
+ * Authors: - Michael Weibel <michael.weibel at gmail.com> - Klaus Herberth <klaus at jsxc.org>
+ * Copyright: - Michael Weibel <michael.weibel at gmail.com>
+ * 
+ * @license MIT
+ */
+
+(function($) {
+   Strophe.addConnectionPlugin('caps', {
+      /**
+       * Constant: HASH Hash used
+       * 
+       * Currently only sha-1 is supported.
+       */
+      HASH: 'sha-1',
+      /**
+       * Variable: node Client which is being used.
+       * 
+       * Can be overwritten as soon as Strophe has been initialized.
+       */
+      node: 'http://strophe.im/strophejs/',
+      /**
+       * PrivateVariable: _ver Own generated version string
+       */
+      _ver: '',
+      /**
+       * PrivateVariable: _connection Strophe connection
+       */
+      _connection: null,
+      /**
+       * PrivateVariable: _knownCapabilities A hashtable containing
+       * version-strings and their capabilities, serialized as string.
+       * 
+       * TODO: Maybe those caps shouldn't be serialized.
+       */
+      _knownCapabilities: JSON.parse(localStorage.getItem('strophe.caps._knownCapabilities')) || {},
+
+      /**
+       * PrivateVariable: _jidVerIndex A hashtable containing jids and their
+       * versions for better lookup of capabilities.
+       */
+      _jidVerIndex: JSON.parse(localStorage.getItem('strophe.caps._jidVerIndex')) || {},
+
+      /**
+       * Function: init Initialize plugin: - Add caps namespace - Add caps
+       * feature to disco plugin - Add handler for caps stanzas
+       * 
+       * Parameters: (Strophe.Connection) conn - Strophe connection
+       */
+      init: function(conn) {
+         this._connection = conn;
+
+         Strophe.addNamespace('CAPS', 'http://jabber.org/protocol/caps');
+
+         if (!this._connection.disco) {
+            throw "Caps plugin requires the disco plugin to be installed.";
+         }
+
+         this._connection.disco.addFeature(Strophe.NS.CAPS);
+         this._connection.addHandler(this._delegateCapabilities.bind(this), Strophe.NS.CAPS);
+      },
+
+      /**
+       * Function: generateCapsAttrs Returns the attributes for generating the
+       * "c"-stanza containing the own version
+       * 
+       * Returns: (Object) - attributes
+       */
+      generateCapsAttrs: function() {
+         return {
+            'xmlns': Strophe.NS.CAPS,
+            'hash': this.HASH,
+            'node': this.node,
+            'ver': this.generateVer()
+         };
+      },
+
+      /**
+       * Function: generateVer Returns the base64 encoded version string
+       * (encoded itself with sha1)
+       * 
+       * Returns: (String) - version
+       */
+      generateVer: function() {
+         if (this._ver !== "") {
+            return this._ver;
+         }
+
+         var ver = "", identities = this._connection.disco._identities.sort(this._sortIdentities), identitiesLen = identities.length, features = this._connection.disco._features.sort(), featuresLen = features.length;
+         for (var i = 0; i < identitiesLen; i++) {
+            var curIdent = identities[i];
+            ver += curIdent.category + "/" + curIdent.type + "/" + curIdent.lang + "/" + curIdent.name + "<";
+         }
+         for (var i = 0; i < featuresLen; i++) {
+            ver += features[i] + '<';
+         }
+
+         this._ver = b64_sha1(ver);
+         return this._ver;
+      },
+
+      /**
+       * Function: getCapabilitiesByJid Returns serialized capabilities of a jid
+       * (if available). Otherwise null.
+       * 
+       * Parameters: (String) jid - Jabber id
+       * 
+       * Returns: (String|null) - capabilities, serialized; or null when not
+       * available.
+       */
+      getCapabilitiesByJid: function(jid) {
+         if (this._jidVerIndex[jid]) {
+            return this._knownCapabilities[this._jidVerIndex[jid]];
+         }
+         return null;
+      },
+      hasFeatureByJid: function(jid, feature) {
+         if (this._jidVerIndex[jid] && feature !== null && typeof feature !== 'undefined') {
+            if(!$.isArray(feature)){
+               feature = $.makeArray(feature);
+            };
+            
+            var i;
+            for (i = 0; i < feature.length; i++) {
+               if (this._knownCapabilities[this._jidVerIndex[jid]]['features'].indexOf(feature[i]) < 0)
+                  return false;
+            }
+            return true;
+         }
+         return false;
+      },
+
+      /**
+       * PrivateFunction: _delegateCapabilities Checks if the version has
+       * already been saved. If yes: do nothing. If no: Request capabilities
+       * 
+       * Parameters: (Strophe.Builder) stanza - Stanza
+       * 
+       * Returns: (Boolean)
+       */
+      _delegateCapabilities: function(stanza) {
+         var from = stanza.getAttribute('from'), c = stanza.querySelector('c'), ver = c.getAttribute('ver'), node = c.getAttribute('node');
+         if (!this._knownCapabilities[ver]) {
+            return this._requestCapabilities(from, node, ver);
+         } else {
+            this._jidVerIndex[from] = ver;
+         }
+         if (!this._jidVerIndex[from] || !this._jidVerIndex[from] !== ver) {
+            this._jidVerIndex[from] = ver;
+         }
+
+         localStorage.setItem('strophe.caps._jidVerIndex', JSON.stringify(this._jidVerIndex));
+         $(document).trigger('caps.strophe', [ from ]);
+
+         return true;
+      },
+
+      /**
+       * PrivateFunction: _requestCapabilities Requests capabilities from the
+       * one which sent the caps-info stanza. This is done using disco info.
+       * 
+       * Additionally, it registers a handler for handling the reply.
+       * 
+       * Parameters: (String) to - Destination jid (String) node - Node
+       * attribute of the caps-stanza (String) ver - Version of the caps-stanza
+       * 
+       * Returns: (Boolean) - true
+       */
+      _requestCapabilities: function(to, node, ver) {
+         if (to !== this._connection.jid) {
+            var id = this._connection.disco.info(to, node + '#' + ver);
+            this._connection.addHandler(this._handleDiscoInfoReply.bind(this), Strophe.NS.DISCO_INFO, 'iq', 'result', id, to);
+         }
+         return true;
+      },
+
+      /**
+       * PrivateFunction: _handleDiscoInfoReply Parses the disco info reply and
+       * adds the version & it's capabilities to the _knownCapabilities
+       * variable. Additionally, it adds the jid & the version to the
+       * _jidVerIndex variable for a better lookup.
+       * 
+       * Parameters: (Strophe.Builder) stanza - Disco info stanza
+       * 
+       * Returns: (Boolean) - false, to automatically remove the handler.
+       */
+      _handleDiscoInfoReply: function(stanza) {
+         var query = stanza.querySelector('query'), node = query.getAttribute('node').split('#'), ver = node[1], from = stanza.getAttribute('from');
+         if (!this._knownCapabilities[ver]) {
+            var childNodes = query.childNodes, childNodesLen = childNodes.length;
+            this._knownCapabilities[ver] = {
+               features: [],
+               identities: []
+            };
+            for (var i = 0; i < childNodesLen; i++) {
+               var node = childNodes[i];
+               if (node.nodeName == 'feature') {
+                  this._knownCapabilities[ver]['features'].push(node.getAttribute('var'));
+               } else if (node.nodeName == 'identity') {
+                  this._knownCapabilities[ver]['identities'].push(this._attributesToJsObject(node.attributes));
+               } else {
+                  if (_knownCapabilities[ver][node.nodeName])
+                     _knownCapabilities[ver][node.nodeName] = [];
+                  this._knownCapabilities[ver][node.nodeName].push(this._attributesToJsObject(node.attributes));
+               }
+            }
+            this._jidVerIndex[from] = ver;
+         } else if (!this._jidVerIndex[from] || !this._jidVerIndex[from] !== ver) {
+            this._jidVerIndex[from] = ver;
+         }
+
+         localStorage.setItem('strophe.caps._jidVerIndex', JSON.stringify(this._jidVerIndex));
+         localStorage.setItem('strophe.caps._knownCapabilities', JSON.stringify(this._knownCapabilities));
+         $(document).trigger('caps.strophe', [ from ]);
+
+         return false;
+      },
+
+      _attributesToJsObject: function(attr) {
+         var obj = {};
+
+         for (i = 0; i < attr.length; i++)
+            obj[attr[i].name] = attr[i].value;
+
+         return obj;
+      },
+
+      /**
+       * PrivateFunction: _sortIdentities Sorts two identities according the
+       * sorting requirements in XEP-0115.
+       * 
+       * Parameters: (Object) a - Identity a (Object) b - Identity b
+       * 
+       * Returns: (Integer) - 1, 0 or -1; according to which one's greater.
+       */
+      _sortIdentities: function(a, b) {
+         if (a.category > b.category) {
+            return 1;
+         }
+         if (a.category < b.category) {
+            return -1;
+         }
+         if (a.type > b.type) {
+            return 1;
+         }
+         if (a.type < b.type) {
+            return -1;
+         }
+         if (a.lang > b.lang) {
+            return 1;
+         }
+         if (a.lang < b.lang) {
+            return -1;
+         }
+         return 0;
+      }
+   });
+}(jQuery));
+/*!
+ * Source: build/lib/strophe.vcard.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+// Generated by CoffeeScript 1.3.3
+/*
+Plugin to implement the vCard extension.
+http://xmpp.org/extensions/xep-0054.html
+
+Author: Nathan Zorn (nathan.zorn at gmail.com)
+CoffeeScript port: Andreas Guth (guth at dbis.rwth-aachen.de)
+*/
+
+/* jslint configuration:
+*/
+
+/* global document, window, setTimeout, clearTimeout, console,
+    XMLHttpRequest, ActiveXObject,
+    Base64, MD5,
+    Strophe, $build, $msg, $iq, $pres
+*/
+
+var buildIq;
+
+buildIq = function(type, jid, vCardEl) {
+  var iq;
+  iq = $iq(jid ? {
+    type: type,
+    to: jid
+  } : {
+    type: type
+  });
+  iq.c("vCard", {
+    xmlns: Strophe.NS.VCARD
+  });
+  if (vCardEl) {
+    iq.cnode(vCardEl);
+  }
+  return iq;
+};
+
+Strophe.addConnectionPlugin('vcard', {
+  _connection: null,
+  init: function(conn) {
+    this._connection = conn;
+    return Strophe.addNamespace('VCARD', 'vcard-temp');
+  },
+  /*Function
+    Retrieve a vCard for a JID/Entity
+    Parameters:
+    (Function) handler_cb - The callback function used to handle the request.
+    (String) jid - optional - The name of the entity to request the vCard
+       If no jid is given, this function retrieves the current user's vcard.
+  */
+
+  get: function(handler_cb, jid, error_cb) {
+    var iq;
+    iq = buildIq("get", jid);
+    return this._connection.sendIQ(iq, handler_cb, error_cb);
+  },
+  /* Function
+      Set an entity's vCard.
+  */
+
+  set: function(handler_cb, vCardEl, jid, error_cb) {
+    var iq;
+    iq = buildIq("set", jid, vCardEl);
+    return this._connection.sendIQ(iq, handler_cb, error_rb);
+  }
+});
+/*!
+ * Source: build/lib/strophe.jingle/strophe.jingle.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+/* jshint -W117 */
+(function($){
+Strophe.addConnectionPlugin('jingle', {
+    connection: null,
+    sessions: {},
+    jid2session: {},
+    ice_config: {iceServers: []},
+    pc_constraints: {},
+    media_constraints: {
+        mandatory: {
+            'OfferToReceiveAudio': true,
+            'OfferToReceiveVideo': true
+        }
+        // MozDontOfferDataChannel: true when this is firefox
+    },
+    localStream: null,
+
+    init: function (conn) {
+        this.connection = conn;
+        if (this.connection.disco) {
+            // http://xmpp.org/extensions/xep-0167.html#support
+            // http://xmpp.org/extensions/xep-0176.html#support
+            this.connection.disco.addFeature('urn:xmpp:jingle:1');
+            this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
+            this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
+            this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
+            this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
+
+
+            // this is dealt with by SDP O/A so we don't need to annouce this
+            //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
+            //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
+            this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
+            //this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
+            //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
+        }
+        this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
+    },
+    onJingle: function (iq) {
+        var sid = $(iq).find('jingle').attr('sid');
+        var action = $(iq).find('jingle').attr('action');
+        // send ack first
+        var ack = $iq({type: 'result',
+              to: iq.getAttribute('from'),
+              id: iq.getAttribute('id')
+        });
+        console.log('on jingle ' + action);
+        var sess = this.sessions[sid];
+        if ('session-initiate' != action) {
+            if (sess === null) {
+                ack.type = 'error';
+                ack.c('error', {type: 'cancel'})
+                   .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
+                   .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+                this.connection.send(ack);
+                return true;
+            }
+            // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
+            // local jid is not checked
+            if (Strophe.getBareJidFromJid(iq.getAttribute('from')) != Strophe.getBareJidFromJid(sess.peerjid)) {
+                console.warn('jid mismatch for session id', sid, iq.getAttribute('from'), sess.peerjid);
+                ack.type = 'error';
+                ack.c('error', {type: 'cancel'})
+                   .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
+                   .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+                this.connection.send(ack);
+                return true;
+            }
+        } else if (sess !== undefined) {
+            // existing session with same session id
+            // this might be out-of-order if the sess.peerjid is the same as from
+            ack.type = 'error';
+            ack.c('error', {type: 'cancel'})
+               .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
+            console.warn('duplicate session id', sid);
+            this.connection.send(ack);
+            return true;
+        }
+        // FIXME: check for a defined action
+        this.connection.send(ack);
+        // see http://xmpp.org/extensions/xep-0166.html#concepts-session
+        switch (action) {
+        case 'session-initiate':
+            sess = new JingleSession($(iq).attr('to'), $(iq).find('jingle').attr('sid'), this.connection);
+            // configure session
+            if (this.localStream) {
+                sess.localStreams.push(this.localStream);
+            }
+            sess.media_constraints = this.media_constraints;
+            sess.pc_constraints = this.pc_constraints;
+            sess.ice_config = this.ice_config;
+
+            sess.initiate($(iq).attr('from'), false);
+            sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
+
+            if ($(iq).find('>jingle>muted[xmlns="http://jitsi.org/protocol/meet#startmuted"]').length) {
+                console.log('got a request to start muted');
+                sess.startmuted = true;
+            }
+
+            this.sessions[sess.sid] = sess;
+            this.jid2session[sess.peerjid] = sess;
+
+            // the callback should either 
+            // .sendAnswer and .accept
+            // or .sendTerminate -- not necessarily synchronus
+            $(document).trigger('callincoming.jingle', [sess.sid]);
+            break;
+        case 'session-accept':
+            sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
+            sess.accept();
+            $(document).trigger('callaccepted.jingle', [sess.sid]);
+            break;
+        case 'session-terminate':
+            console.log('terminating...');
+            sess.terminate();
+            this.terminate(sess.sid);
+            if ($(iq).find('>jingle>reason').length) {
+                $(document).trigger('callterminated.jingle', [
+                    sess.sid,
+                    $(iq).find('>jingle>reason>:first')[0].tagName,
+                    $(iq).find('>jingle>reason>text').text()
+                ]);
+            } else {
+                $(document).trigger('callterminated.jingle', [sess.sid]);
+            }
+            break;
+        case 'transport-info':
+            sess.addIceCandidate($(iq).find('>jingle>content'));
+            break;
+        case 'session-info':
+            var affected;
+            if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+                $(document).trigger('ringing.jingle', [sess.sid]);
+            } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+                affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+                $(document).trigger('mute.jingle', [sess.sid, affected]);
+            } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+                affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+                $(document).trigger('unmute.jingle', [sess.sid, affected]);
+            }
+            break;
+        case 'addsource': // FIXME: proprietary
+            sess.addSource($(iq).find('>jingle>content'));
+            break;
+        case 'removesource': // FIXME: proprietary
+            sess.removeSource($(iq).find('>jingle>content'));
+            break;
+        default:
+            console.warn('jingle action not implemented', action);
+            break;
+        }
+        return true;
+    },
+    initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
+        var sess = new JingleSession(myjid || this.connection.jid,
+                                     Math.random().toString(36).substr(2, 12), // random string
+                                     this.connection);
+        // configure session
+        if (this.localStream) {
+            sess.localStreams.push(this.localStream);
+        }
+        sess.media_constraints = this.media_constraints;
+        sess.pc_constraints = this.pc_constraints;
+        sess.ice_config = this.ice_config;
+
+        sess.initiate(peerjid, true);
+        this.sessions[sess.sid] = sess;
+        this.jid2session[sess.peerjid] = sess;
+        sess.sendOffer();
+        return sess;
+    },
+    terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
+        if (sid === null || sid === undefined) {
+            for (sid in this.sessions) {
+                if (this.sessions[sid].state != 'ended') {
+                    this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+                    this.sessions[sid].terminate();
+                }
+                delete this.jid2session[this.sessions[sid].peerjid];
+                delete this.sessions[sid];
+            }
+        } else if (this.sessions.hasOwnProperty(sid)) {
+            if (this.sessions[sid].state != 'ended') {
+                this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+                this.sessions[sid].terminate();
+            }
+            delete this.jid2session[this.sessions[sid].peerjid];
+            delete this.sessions[sid];
+        }
+    },
+    terminateByJid: function (jid) {
+        if (this.jid2session.hasOwnProperty(jid)) {
+            var sess = this.jid2session[jid];
+            if (sess) {
+                sess.terminate();
+                console.log('peer went away silently', jid);
+                delete this.sessions[sess.sid];
+                delete this.jid2session[jid];
+                $(document).trigger('callterminated.jingle', [sess.sid, 'gone']);
+            }
+        }
+    },
+    getStunAndTurnCredentials: function () {
+        // get stun and turn configuration from server via xep-0215
+        // uses time-limited credentials as described in
+        // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
+        //
+        // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
+        // for a prosody module which implements this
+        //
+        // currently, this doesn't work with updateIce and therefore credentials with a long
+        // validity have to be fetched before creating the peerconnection
+        // TODO: implement refresh via updateIce as described in
+        //      https://code.google.com/p/webrtc/issues/detail?id=1650
+        var self = this;
+        this.connection.sendIQ(
+            $iq({type: 'get', to: this.connection.domain})
+                .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
+            function (res) {
+                var iceservers = [];
+                $(res).find('>services>service').each(function (idx, el) {
+                    el = $(el);
+                    var dict = {};
+                    switch (el.attr('type')) {
+                    case 'stun':
+                        dict.url = 'stun:' + el.attr('host');
+                        if (el.attr('port')) {
+                            dict.url += ':' + el.attr('port');
+                        }
+                        iceservers.push(dict);
+                        break;
+                    case 'turn':
+                        dict.url = 'turn:';
+                        if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
+                            if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
+                                dict.url += el.attr('username') + '@';
+                            } else {
+                                dict.username = el.attr('username'); // only works in M28
+                            }
+                        }
+                        dict.url += el.attr('host');
+                        if (el.attr('port') && el.attr('port') != '3478') {
+                            dict.url += ':' + el.attr('port');
+                        }
+                        if (el.attr('transport') && el.attr('transport') != 'udp') {
+                            dict.url += '?transport=' + el.attr('transport');
+                        }
+                        if (el.attr('password')) {
+                            dict.credential = el.attr('password');
+                        }
+                        iceservers.push(dict);
+                        break;
+                    }
+                });
+                self.ice_config.iceServers = iceservers;
+            },
+            function (err) {
+                console.warn('getting turn credentials failed', err);
+                console.warn('is mod_turncredentials or similar installed?');
+            }
+        );
+        // implement push?
+    }
+});
+}(jQuery));
+
+/*!
+ * Source: build/lib/strophe.jingle/strophe.jingle.session.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+/* jshint -W117 */
+// Jingle stuff
+var JingleSession;
+
+(function($){
+JingleSession = function(me, sid, connection) {
+    this.me = me;
+    this.sid = sid;
+    this.connection = connection;
+    this.initiator = null;
+    this.responder = null;
+    this.isInitiator = null;
+    this.peerjid = null;
+    this.state = null;
+    this.peerconnection = null;
+    this.remoteStream = null;
+    this.localSDP = null;
+    this.remoteSDP = null;
+    this.localStreams = [];
+    this.relayedStreams = [];
+    this.remoteStreams = [];
+    this.startTime = null;
+    this.stopTime = null;
+    this.media_constraints = null;
+    this.pc_constraints = null;
+    this.ice_config = {};
+    this.drip_container = [];
+
+    this.usetrickle = true;
+    this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718
+    this.usedrip = false; // dripping is sending trickle candidates not one-by-one
+
+    this.hadstuncandidate = false;
+    this.hadturncandidate = false;
+    this.lasticecandidate = false;
+
+    this.statsinterval = null;
+
+    this.reason = null;
+
+    this.addssrc = [];
+    this.removessrc = [];
+    this.pendingop = null;
+
+    this.wait = true;
+
+    // XEP-0172 support, non-standard
+    this.nickname = null;
+
+    // non-standard "please start muted" support for colibri/meet
+    this.startmuted = false;
+
+    // Filter for testcases with ICE Candidates
+    this.filter_candidates = null;
+}
+
+JingleSession.prototype.initiate = function (peerjid, isInitiator) {
+    var self = this;
+    if (this.state !== null) {
+        console.error('attempt to initiate on session ' + this.sid +
+                  'in state ' + this.state);
+        return;
+    }
+    this.isInitiator = isInitiator;
+    this.state = 'pending';
+    this.initiator = isInitiator ? this.me : peerjid;
+    this.responder = !isInitiator ? this.me : peerjid;
+    this.peerjid = peerjid;
+    //console.log('create PeerConnection ' + JSON.stringify(this.ice_config));
+    try {
+        this.peerconnection = new RTCPeerconnection(this.ice_config,
+                                                     this.pc_constraints);
+    } catch (e) {
+        console.error('Failed to create PeerConnection, exception: ',
+                      e.message);
+        console.error(e);
+        return;
+    }
+    this.hadstuncandidate = false;
+    this.hadturncandidate = false;
+    this.lasticecandidate = false;
+    this.peerconnection.onicecandidate = function (event) {
+        self.sendIceCandidate(event.candidate);
+    };
+    this.peerconnection.onaddstream = function (event) {
+        self.remoteStream = event.stream;
+        self.remoteStreams.push(event.stream);
+        $(document).trigger('remotestreamadded.jingle', [event, self.sid]);
+    };
+    this.peerconnection.onremovestream = function (event) {
+        self.remoteStream = null;
+        // FIXME: remove from this.remoteStreams
+        $(document).trigger('remotestreamremoved.jingle', [event, self.sid]);
+    };
+    this.peerconnection.onsignalingstatechange = function (event) {
+        if (!(self && self.peerconnection)) return;
+    };
+    this.peerconnection.oniceconnectionstatechange = function (event) {
+        if (!(self && self.peerconnection)) return;
+        switch (self.peerconnection.iceConnectionState) {
+        case 'connected':
+            this.startTime = new Date();
+            break;
+        case 'disconnected':
+            this.stopTime = new Date();
+            break;
+        }
+        $(document).trigger('iceconnectionstatechange.jingle', [self.sid, self]);
+    };
+    // add any local and relayed stream
+    this.localStreams.forEach(function(stream) {
+        self.peerconnection.addStream(stream);
+    });
+    this.relayedStreams.forEach(function(stream) {
+        self.peerconnection.addStream(stream);
+    });
+};
+
+JingleSession.prototype.accept = function () {
+    var self = this;
+    this.state = 'active';
+
+    var pranswer = this.peerconnection.localDescription;
+    if (!pranswer || pranswer.type != 'pranswer') {
+        return;
+    }
+    console.log('going from pranswer to answer');
+    if (this.usetrickle) {
+        // remove candidates already sent from session-accept
+        var lines = SDPUtil.find_lines(pranswer.sdp, 'a=candidate:');
+        for (var i = 0; i < lines.length; i++) {
+            pranswer.sdp = pranswer.sdp.replace(lines[i] + '\r\n', '');
+        }
+    }
+    while (SDPUtil.find_line(pranswer.sdp, 'a=inactive')) {
+        // FIXME: change any inactive to sendrecv or whatever they were originally
+        pranswer.sdp = pranswer.sdp.replace('a=inactive', 'a=sendrecv');
+    }
+    var prsdp = new SDP(pranswer.sdp);
+    var accept = $iq({to: this.peerjid,
+             type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-accept',
+           initiator: this.initiator,
+           responder: this.responder,
+           sid: this.sid });
+    prsdp.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder');
+    this.connection.sendIQ(accept,
+        function () {
+            var ack = {};
+            ack.source = 'answer';
+            $(document).trigger('ack.jingle', [self.sid, ack]);
+        },
+        function (stanza) {
+            var error = ($(stanza).find('error').length) ? {
+                code: $(stanza).find('error').attr('code'),
+                reason: $(stanza).find('error :first')[0].tagName,
+            }:{};
+            error.source = 'answer';
+            $(document).trigger('error.jingle', [self.sid, error]);
+        },
+    10000);
+
+    var sdp = this.peerconnection.localDescription.sdp;
+    while (SDPUtil.find_line(sdp, 'a=inactive')) {
+        // FIXME: change any inactive to sendrecv or whatever they were originally
+        sdp = sdp.replace('a=inactive', 'a=sendrecv');
+    }
+    this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),
+        function () {
+            //console.log('setLocalDescription success');
+            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+        },
+        function (e) {
+            console.error('setLocalDescription failed', e);
+        }
+    );
+};
+
+JingleSession.prototype.terminate = function (reason) {
+    this.state = 'ended';
+    this.reason = reason;
+    this.peerconnection.close();
+    if (this.statsinterval !== null) {
+        window.clearInterval(this.statsinterval);
+        this.statsinterval = null;
+    }
+};
+
+JingleSession.prototype.active = function () {
+    return this.state == 'active';
+};
+
+JingleSession.prototype.sendIceCandidate = function (candidate) {
+    var self = this;
+    if (candidate && !this.lasticecandidate) {
+        var ice = SDPUtil.iceparams(this.localSDP.media[candidate.sdpMLineIndex], this.localSDP.session);
+        var jcand = SDPUtil.candidateToJingle(candidate.candidate);
+        if (!(ice && jcand)) {
+            console.error('failed to get ice && jcand');
+            return;
+        }
+        ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+
+        if (jcand.type === 'srflx') {
+            this.hadstuncandidate = true;
+        } else if (jcand.type === 'relay') {
+            this.hadturncandidate = true;
+        }
+
+        if(this.filter_candidates === null || jcand.type === this.filter_candidates) {
+            if (this.usetrickle) {
+                console.log('sendIceCandidate using trickle');
+                if (this.usedrip) {
+                    if (this.drip_container.length === 0) {
+                        // start 20ms callout
+                        window.setTimeout(function () {
+                            console.log('sending drip container');
+                            if (self.drip_container.length === 0) return;
+                            self.sendIceCandidates(self.drip_container);
+                            self.drip_container = [];
+                        }, 20);
+
+                    }
+                    this.drip_container.push(event.candidate);
+                    return;
+                } else {
+                    console.log('sending single candidate');
+                    self.sendIceCandidates([event.candidate]);
+                }
+            }
+        }
+    } else {
+        console.log('sendIceCandidate: last candidate...');
+        if (!this.usetrickle) {
+            console.log('should send full offer now...');
+            var init = $iq({to: this.peerjid,
+                       type: 'set'})
+                .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+                   action: this.peerconnection.localDescription.type == 'offer' ? 'session-initiate' : 'session-accept',
+                   initiator: this.initiator,
+                   sid: this.sid});
+            if (this.nickname != null) {
+                init.c('nick', {xmlns:'http://jabber.org/protocol/nick'}).t(this.nickname).up();
+            }
+            if (this.startmuted) {
+                init.c('muted', {xmlns:'http://jitsi.org/protocol/meet#startmuted'}).up();
+            }
+            this.localSDP = new SDP(this.peerconnection.localDescription.sdp);
+            this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder');
+            console.log('try to send ack(offer)...');
+            this.connection.sendIQ(init,
+                function () {
+                    console.log('Sent session initiate (ACK, offer)...');
+                    var ack = {};
+                    ack.source = 'offer';
+                    $(document).trigger('ack.jingle', [self.sid, ack]);
+                },
+                function (stanza) {
+                    self.state = 'error';
+                    self.peerconnection.close();
+                    var error = ($(stanza).find('error').length) ? {
+                        code: $(stanza).find('error').attr('code'),
+                        reason: $(stanza).find('error :first')[0].tagName,
+                    }:{};
+                    error.source = 'offer';
+                    $(document).trigger('error.jingle', [self.sid, error]);
+                },
+            10000);
+        }
+        this.lasticecandidate = true;
+        console.log('Have we encountered any srflx candidates? ' + this.hadstuncandidate);
+        console.log('Have we encountered any relay candidates? ' + this.hadturncandidate);
+
+        if (!(this.hadstuncandidate || this.hadturncandidate) && this.peerconnection.signalingState != 'closed') {
+            console.log('no candidates found!');
+            $(document).trigger('nostuncandidates.jingle', [this.sid]);
+        }
+    }
+};
+
+JingleSession.prototype.sendIceCandidates = function (candidates) {
+    console.log('sendIceCandidates', candidates);
+    var cand = $iq({to: this.peerjid, type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'transport-info',
+           initiator: this.initiator,
+           sid: this.sid});
+    for (var mid = 0; mid < this.localSDP.media.length; mid++) {
+        var cands = candidates.filter(function (el) { return el.sdpMLineIndex == mid; });
+        if (cands.length > 0) {
+            var ice = SDPUtil.iceparams(this.localSDP.media[mid], this.localSDP.session);
+            ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+            cand.c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',
+                   name: cands[0].sdpMid
+            }).c('transport', ice);
+            for (var i = 0; i < cands.length; i++) {
+                cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();
+            }
+            // add fingerprint
+            if (SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session)) {
+                var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session));
+                tmp.required = true;
+                cand.c('fingerprint').t(tmp.fingerprint);
+                delete tmp.fingerprint;
+                cand.attrs(tmp);
+                cand.up();
+            }
+            cand.up(); // transport
+            cand.up(); // content
+        }
+    }
+    // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340
+    //console.log('was this the last candidate', this.lasticecandidate);
+    console.log('try to send ack(transportinfo)...');
+    this.connection.sendIQ(cand,
+        function () {
+            var ack = {};
+            ack.source = 'transportinfo';
+            console.log('Sent session initiate (ACK, transportinfo)...');
+            $(document).trigger('ack.jingle', [this.sid, ack]);
+        },
+        function (stanza) {
+            var error = ($(stanza).find('error').length) ? {
+                code: $(stanza).find('error').attr('code'),
+                reason: $(stanza).find('error :first')[0].tagName,
+            }:{};
+            error.source = 'transportinfo';
+            $(document).trigger('error.jingle', [this.sid, error]);
+        },
+    10000);
+};
+
+
+JingleSession.prototype.sendOffer = function () {
+    //console.log('sendOffer...');
+    var self = this;
+    this.peerconnection.createOffer(function (sdp) {
+            self.createdOffer(sdp);
+        },
+        function (e) {
+            console.error('createOffer failed', e);
+        },
+        this.media_constraints
+    );
+};
+
+JingleSession.prototype.createdOffer = function (sdp) {
+    //console.log('createdOffer', sdp);
+    var self = this;
+    this.localSDP = new SDP(sdp.sdp);
+    //this.localSDP.mangle();
+    if (this.usetrickle) {
+        var init = $iq({to: this.peerjid,
+                   type: 'set'})
+            .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+               action: 'session-initiate',
+               initiator: this.initiator,
+               sid: this.sid});
+        if (this.nickname != null) {
+            init.c('nick', {xmlns:'http://jabber.org/protocol/nick'}).t(this.nickname).up();
+        }
+        if (this.startmuted) {
+            init.c('muted', {xmlns:'http://jitsi.org/protocol/meet#startmuted'}).up();
+        }
+        this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder');
+        this.connection.sendIQ(init,
+            function () {
+                var ack = {};
+                ack.source = 'offer';
+                $(document).trigger('ack.jingle', [self.sid, ack]);
+            },
+            function (stanza) {
+                self.state = 'error';
+                self.peerconnection.close();
+                var error = ($(stanza).find('error').length) ? {
+                    code: $(stanza).find('error').attr('code'),
+                    reason: $(stanza).find('error :first')[0].tagName,
+                }:{};
+                error.source = 'offer';
+                $(document).trigger('error.jingle', [self.sid, error]);
+            },
+        10000);
+    }
+    sdp.sdp = this.localSDP.raw;
+    this.peerconnection.setLocalDescription(sdp, 
+        function () {
+            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+            //console.log('setLocalDescription success');
+        },
+        function (e) {
+            console.error('setLocalDescription failed', e);
+        }
+    );
+    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
+    for (var i = 0; i < cands.length; i++) {
+        var cand = SDPUtil.parse_icecandidate(cands[i]);
+        if (cand.type == 'srflx') {
+            this.hadstuncandidate = true;
+        } else if (cand.type == 'relay') {
+            this.hadturncandidate = true;
+        }
+    }
+};
+
+JingleSession.prototype.setRemoteDescription = function (elem, desctype) {
+    //console.log('setting remote description... ', desctype);
+    this.remoteSDP = new SDP('');
+    this.remoteSDP.fromJingle(elem);
+    if (this.peerconnection.remoteDescription !== null) {
+        console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);
+        if (this.peerconnection.remoteDescription.type == 'pranswer') {
+            var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);
+            for (var i = 0; i < pranswer.media.length; i++) {
+                // make sure we have ice ufrag and pwd
+                if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {
+                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {
+                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\r\n';
+                    } else {
+                        console.warn('no ice ufrag?');
+                    }
+                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {
+                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\r\n';
+                    } else {
+                        console.warn('no ice pwd?');
+                    }
+                }
+                // copy over candidates
+                var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');
+                for (var j = 0; j < lines.length; j++) {
+                    this.remoteSDP.media[i] += lines[j] + '\r\n';
+                }
+            }
+            this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
+        }
+    }
+    var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});
+    
+    this.peerconnection.setRemoteDescription(remotedesc,
+        function () {
+            //console.log('setRemoteDescription success');
+        },
+        function (e) {
+            console.error('setRemoteDescription error', e);
+        }
+    );
+};
+
+JingleSession.prototype.addIceCandidate = function (elem) {
+    var self = this;
+    if (this.peerconnection.signalingState == 'closed') {
+        return;
+    }
+    if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {
+        console.log('trickle ice candidate arriving before session accept...');
+        // create a PRANSWER for setRemoteDescription
+        if (!this.remoteSDP) {
+            var cobbled = 'v=0\r\n' +
+                'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
+                's=-\r\n' +
+                't=0 0\r\n';
+            // first, take some things from the local description
+            for (var i = 0; i < this.localSDP.media.length; i++) {
+                cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\r\n';
+                cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\r\n') + '\r\n';
+                if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {
+                    cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\r\n';
+                }
+                cobbled += 'a=inactive\r\n';
+            }
+            this.remoteSDP = new SDP(cobbled);
+        }
+        // then add things like ice and dtls from remote candidate
+        elem.each(function () {
+            for (var i = 0; i < self.remoteSDP.media.length; i++) {
+                if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+                        self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+                    if (!SDPUtil.find_line(self.remoteSDP.media[i], 'a=ice-ufrag:')) {
+                        var tmp = $(this).find('transport');
+                        self.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\r\n';
+                        self.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\r\n';
+                        tmp = $(this).find('transport>fingerprint');
+                        if (tmp.length) {
+                            self.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\r\n';
+                        } else {
+                            console.log('no dtls fingerprint (webrtc issue #1718?)');
+                            self.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\r\n';
+                        }
+                        break;
+                    }
+                }
+            }
+        });
+        this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
+
+        // we need a complete SDP with ice-ufrag/ice-pwd in all parts
+        // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts
+        // but it could be in the session part as well. since the code above constructs this sdp this can't happen however
+        var iscomplete = this.remoteSDP.media.filter(function (mediapart) {
+            return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');
+        }).length == this.remoteSDP.media.length;
+
+        if (iscomplete) {
+            console.log('setting pranswer');
+            try {
+                this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }),
+                    function() {
+                    },
+                    function(e) {
+                        console.log('setRemoteDescription pranswer failed', e.toString());
+                    });
+            } catch (e) {
+                console.error('setting pranswer failed', e);
+            }
+        } else {
+            //console.log('not yet setting pranswer');
+        }
+    }
+    // operate on each content element
+    elem.each(function () {
+        // would love to deactivate this, but firefox still requires it
+        var idx = -1;
+        var i;
+        for (i = 0; i < self.remoteSDP.media.length; i++) {
+            if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+                self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+                idx = i;
+                break;
+            }
+        }
+        if (idx == -1) { // fall back to localdescription
+            for (i = 0; i < self.localSDP.media.length; i++) {
+                if (SDPUtil.find_line(self.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+                    self.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+                    idx = i;
+                    break;
+                }
+            }
+        }
+        var name = $(this).attr('name');
+        // TODO: check ice-pwd and ice-ufrag?
+        $(this).find('transport>candidate').each(function () {
+            var line, candidate;
+            line = SDPUtil.candidateFromJingle(this);
+            candidate = new RTCIceCandidate({sdpMLineIndex: idx,
+                                            sdpMid: name,
+                                            candidate: line});
+            try {
+                self.peerconnection.addIceCandidate(candidate);
+            } catch (e) {
+                console.error('addIceCandidate failed', e.toString(), line);
+            }
+        });
+    });
+};
+
+JingleSession.prototype.sendAnswer = function (provisional) {
+    //console.log('createAnswer', provisional);
+    var self = this;
+    this.peerconnection.createAnswer(
+        function (sdp) {
+            self.createdAnswer(sdp, provisional);
+        },
+        function (e) {
+            console.error('createAnswer failed', e);
+        },
+        this.media_constraints
+    );
+};
+
+JingleSession.prototype.createdAnswer = function (sdp, provisional) {
+    //console.log('createAnswer callback');
+    var self = this;
+    this.localSDP = new SDP(sdp.sdp);
+    //this.localSDP.mangle();
+    this.usepranswer = provisional === true;
+
+    if (this.startmuted) {
+        console.log('we got a request to start muted...');
+        this.connection.jingle.localStream.getAudioTracks().forEach(function (track) {
+            track.enabled = false;
+        });
+        // doing this freezes local video, too (which probably means it should be replaced
+        // by a symbol
+        this.connection.jingle.localStream.getVideoTracks().forEach(function (track) {
+            track.enabled = false;
+        });
+
+        // set video to recvonly
+        this.localSDP.media[1] = this.localSDP.media[1].replace('a=sendrecv', 'a=recvonly');
+        // and remove a=ssrc lines. Weird things happen otherwise
+        SDPUtil.find_lines(this.localSDP.media[1], 'a=ssrc:').forEach(function (line) {
+            self.localSDP.media[1] = self.localSDP.media[1].replace(line + '\r\n', '');
+        });
+        this.localSDP.raw = this.localSDP.session + this.localSDP.media.join('');
+    }
+
+    if (this.usetrickle) {
+        if (!this.usepranswer) {
+            var accept = $iq({to: this.peerjid,
+                     type: 'set'})
+                .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+                   action: 'session-accept',
+                   initiator: this.initiator,
+                   responder: this.responder,
+                   sid: this.sid });
+            this.localSDP.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder');
+            this.connection.sendIQ(accept,
+                function () {
+                    var ack = {};
+                    ack.source = 'answer';
+                    $(document).trigger('ack.jingle', [self.sid, ack]);
+                },
+                function (stanza) {
+                    var error = ($(stanza).find('error').length) ? {
+                        code: $(stanza).find('error').attr('code'),
+                        reason: $(stanza).find('error :first')[0].tagName,
+                    }:{};
+                    error.source = 'answer';
+                    $(document).trigger('error.jingle', [self.sid, error]);
+                },
+            10000);
+        } else {
+            sdp.type = 'pranswer';
+            for (var i = 0; i < this.localSDP.media.length; i++) {
+                this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\r\n', 'a=inactive\r\n');
+            }
+            this.localSDP.raw = this.localSDP.session + this.localSDP.media.join('');
+        }
+    }
+    sdp.sdp = this.localSDP.raw;
+    this.peerconnection.setLocalDescription(sdp,
+        function () {
+            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+            //console.log('setLocalDescription success');
+        },
+        function (e) {
+            console.error('setLocalDescription failed', e);
+        }
+    );
+    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
+    for (var j = 0; j < cands.length; j++) {
+        var cand = SDPUtil.parse_icecandidate(cands[j]);
+        if (cand.type == 'srflx') {
+            this.hadstuncandidate = true;
+        } else if (cand.type == 'relay') {
+            this.hadturncandidate = true;
+        }
+    }
+};
+
+JingleSession.prototype.sendTerminate = function (reason, text) {
+    var self = this,
+        term = $iq({to: this.peerjid,
+               type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-terminate',
+           initiator: this.initiator,
+           sid: this.sid})
+        .c('reason')
+        .c(reason || 'success');
+        
+    if (text) {
+        term.up().c('text').t(text);
+    }
+    
+    this.connection.sendIQ(term,
+        function () {
+            self.peerconnection.close();
+            self.peerconnection = null;
+            self.terminate();
+            var ack = {};
+            ack.source = 'terminate';
+            $(document).trigger('ack.jingle', [self.sid, ack]);
+        },
+        function (stanza) {
+            var error = ($(stanza).find('error').length) ? {
+                code: $(stanza).find('error').attr('code'),
+                reason: $(stanza).find('error :first')[0].tagName,
+            }:{};
+            $(document).trigger('ack.jingle', [self.sid, error]);
+        },
+    10000);
+    if (this.statsinterval !== null) {
+        window.clearInterval(this.statsinterval);
+        this.statsinterval = null;
+    }
+};
+
+
+JingleSession.prototype.addSource = function (elem) {
+    console.log('addssrc', new Date().getTime());
+    console.log('ice', this.peerconnection.iceConnectionState);
+    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+    var self = this;
+    $(elem).each(function (idx, content) {
+        var name = $(content).attr('name');
+        var lines = '';
+        tmp = $(content).find('>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+        tmp.each(function () {
+            var ssrc = $(this).attr('ssrc');
+            $(this).find('>parameter').each(function () {
+                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
+                if ($(this).attr('value') && $(this).attr('value').length)
+                    lines += ':' + $(this).attr('value');
+                lines += '\r\n';
+            });
+        });
+        sdp.media.forEach(function(media, idx) {
+            if (!SDPUtil.find_line(media, 'a=mid:' + name))
+                return;
+            sdp.media[idx] += lines;
+            if (!self.addssrc[idx]) self.addssrc[idx] = '';
+            self.addssrc[idx] += lines;
+        });
+        sdp.raw = sdp.session + sdp.media.join('');
+    });
+    this.modifySources();
+};
+
+JingleSession.prototype.removeSource = function (elem) {
+    console.log('removessrc', new Date().getTime());
+    console.log('ice', this.peerconnection.iceConnectionState);
+    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+    var self = this;
+    $(elem).each(function (idx, content) {
+        var name = $(content).attr('name');
+        var lines = '';
+        tmp = $(content).find('>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+        tmp.each(function () {
+            var ssrc = $(this).attr('ssrc');
+            $(this).find('>parameter').each(function () {
+                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
+                if ($(this).attr('value') && $(this).attr('value').length)
+                    lines += ':' + $(this).attr('value');
+                lines += '\r\n';
+            });
+        });
+        sdp.media.forEach(function(media, idx) {
+            if (!SDPUtil.find_line(media, 'a=mid:' + name))
+                return;
+            sdp.media[idx] += lines;
+            if (!self.addssrc[idx]) self.removessrc[idx] = '';
+            self.removessrc[idx] += lines;
+        });
+        sdp.raw = sdp.session + sdp.media.join('');
+    });
+    this.modifySources();
+};
+
+JingleSession.prototype.modifySources = function() {
+    var self = this;
+    if (this.peerconnection.signalingState == 'closed') return;
+    if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null)) return;
+    if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {
+        console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);
+        this.wait = true;
+        window.setTimeout(function() { self.modifySources(); }, 250);
+        return;
+    }
+    if (this.wait) {
+        window.setTimeout(function() { self.modifySources(); }, 2500);
+        this.wait = false;
+        return;
+    }
+
+    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+    // add sources
+    this.addssrc.forEach(function(lines, idx) {
+        sdp.media[idx] += lines;
+    });
+    this.addssrc = [];
+
+    // remove sources
+    this.removessrc.forEach(function(lines, idx) {
+        lines = lines.split('\r\n');
+        lines.pop(); // remove empty last element;
+        lines.forEach(function(line) {
+            sdp.media[idx] = sdp.media[idx].replace(line + '\r\n', '');
+        });
+    });
+    this.removessrc = [];
+
+    sdp.raw = sdp.session + sdp.media.join('');
+    this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),
+        function() {
+            self.peerconnection.createAnswer(
+                function(modifiedAnswer) {
+                    // change video direction, see https://github.com/jitsi/jitmeet/issues/41
+                    if (self.pendingop !== null) {
+                        var sdp = new SDP(modifiedAnswer.sdp);
+                        if (sdp.media.length > 1) {
+                            switch(self.pendingop) {
+                            case 'mute':
+                                sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
+                                break;
+                            case 'unmute':
+                                sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
+                                break;
+                            }
+                            sdp.raw = sdp.session + sdp.media.join('');
+                            modifiedAnswer.sdp = sdp.raw;
+                        }
+                        self.pendingop = null;
+                    }
+
+                    self.peerconnection.setLocalDescription(modifiedAnswer,
+                        function() {
+                            //console.log('modified setLocalDescription ok');
+                            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+                        },
+                        function(error) {
+                            console.log('modified setLocalDescription failed');
+                        }
+                    );
+                },
+                function(error) {
+                    console.log('modified answer failed');
+                }
+            );
+        },
+        function(error) {
+            console.log('modify failed');
+        }
+    );
+};
+
+// SDP-based mute by going recvonly/sendrecv
+// FIXME: should probably black out the screen as well
+JingleSession.prototype.hardMuteVideo = function (muted) {
+    this.pendingop = muted ? 'mute' : 'unmute';
+    this.modifySources();
+
+    this.connection.jingle.localStream.getVideoTracks().forEach(function (track) {
+        track.enabled = !muted;
+    });
+};
+
+JingleSession.prototype.sendMute = function (muted, content) {
+    var info = $iq({to: this.peerjid,
+             type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-info',
+           initiator: this.initiator,
+           sid: this.sid });
+    info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
+    info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});
+    if (content) {
+        info.attrs({'name': content});
+    }
+    this.connection.send(info);
+};
+
+JingleSession.prototype.sendRinging = function () {
+    var info = $iq({to: this.peerjid,
+             type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-info',
+           initiator: this.initiator,
+           sid: this.sid });
+    info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
+    this.connection.send(info);
+};
+
+JingleSession.prototype.getStats = function (interval) {
+    var self = this;
+    var recv = {audio: 0, video: 0};
+    var lost = {audio: 0, video: 0};
+    var lastrecv = {audio: 0, video: 0};
+    var lastlost = {audio: 0, video: 0};
+    var loss = {audio: 0, video: 0};
+    var delta = {audio: 0, video: 0};
+    this.statsinterval = window.setInterval(function () {
+        if (self && self.peerconnection && self.peerconnection.getStats) {
+            self.peerconnection.getStats(function (stats) {
+                var results = stats.result();
+                // TODO: there are so much statistics you can get from this..
+                for (var i = 0; i < results.length; ++i) {
+                    if (results[i].type == 'ssrc') {
+                        var packetsrecv = results[i].stat('packetsReceived');
+                        var packetslost = results[i].stat('packetsLost');
+                        if (packetsrecv && packetslost) {
+                            packetsrecv = parseInt(packetsrecv, 10);
+                            packetslost = parseInt(packetslost, 10);
+                            
+                            if (results[i].stat('googFrameRateReceived')) {
+                                lastlost.video = lost.video;
+                                lastrecv.video = recv.video;
+                                recv.video = packetsrecv;
+                                lost.video = packetslost;
+                            } else {
+                                lastlost.audio = lost.audio;
+                                lastrecv.audio = recv.audio;
+                                recv.audio = packetsrecv;
+                                lost.audio = packetslost;
+                            }
+                        }
+                    }
+                }
+                delta.audio = recv.audio - lastrecv.audio;
+                delta.video = recv.video - lastrecv.video;
+                loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;
+                loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;
+                $(document).trigger('packetloss.jingle', [self.sid, loss]);
+            });
+        }
+    }, interval || 3000);
+    return this.statsinterval;
+};
+
+}(jQuery));
+
+/*!
+ * Source: build/lib/strophe.jingle/strophe.jingle.sdp.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+/* jshint -W117 */
+var SDP;
+
+(function($){
+// SDP STUFF
+SDP = function(sdp) {
+    this.media = sdp.split('\r\nm=');
+    for (var i = 1; i < this.media.length; i++) {
+        this.media[i] = 'm=' + this.media[i];
+        if (i != this.media.length - 1) {
+            this.media[i] += '\r\n';
+        }
+    }
+    this.session = this.media.shift() + '\r\n';
+    this.raw = this.session + this.media.join('');
+}
+
+// remove iSAC and CN from SDP
+SDP.prototype.mangle = function () {
+    var i, j, mline, lines, rtpmap, newdesc;
+    for (i = 0; i < this.media.length; i++) {
+        lines = this.media[i].split('\r\n');
+        lines.pop(); // remove empty last element
+        mline = SDPUtil.parse_mline(lines.shift());
+        if (mline.media != 'audio')
+            continue;
+        newdesc = '';
+        mline.fmt.length = 0;
+        for (j = 0; j < lines.length; j++) {
+            if (lines[j].substr(0, 9) == 'a=rtpmap:') {
+                rtpmap = SDPUtil.parse_rtpmap(lines[j]);
+                if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')
+                    continue;
+                mline.fmt.push(rtpmap.id);
+                newdesc += lines[j] + '\r\n';
+            } else {
+                newdesc += lines[j] + '\r\n';
+            }
+        }
+        this.media[i] = SDPUtil.build_mline(mline) + '\r\n';
+        this.media[i] += newdesc;
+    }
+    this.raw = this.session + this.media.join('');
+};
+
+// remove lines matching prefix from session section
+SDP.prototype.removeSessionLines = function(prefix) {
+    var self = this;
+    var lines = SDPUtil.find_lines(this.session, prefix);
+    lines.forEach(function(line) {
+        self.session = self.session.replace(line + '\r\n', '');
+    });
+    this.raw = this.session + this.media.join('');
+    return lines;
+}
+// remove lines matching prefix from a media section specified by mediaindex
+// TODO: non-numeric mediaindex could match mid
+SDP.prototype.removeMediaLines = function(mediaindex, prefix) {
+    var self = this;
+    var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);
+    lines.forEach(function(line) {
+        self.media[mediaindex] = self.media[mediaindex].replace(line + '\r\n', '');
+    });
+    this.raw = this.session + this.media.join('');
+    return lines;
+}
+
+// add content's to a jingle element
+SDP.prototype.toJingle = function (elem, thecreator) {
+    var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;
+    var self = this;
+    // new bundle plan
+    if (SDPUtil.find_line(this.session, 'a=group:')) {
+        lines = SDPUtil.find_lines(this.session, 'a=group:');
+        for (i = 0; i < lines.length; i++) {
+            tmp = lines[i].split(' ');
+            var semantics = tmp.shift().substr(8);
+            elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', semantics:semantics});
+            for (j = 0; j < tmp.length; j++) {
+                elem.c('content', {name: tmp[j]}).up();
+            }
+            elem.up();
+        }
+    }
+    // old bundle plan, to be removed
+    var bundle = [];
+    if (SDPUtil.find_line(this.session, 'a=group:BUNDLE')) {
+        bundle = SDPUtil.find_line(this.session, 'a=group:BUNDLE ').split(' ');
+        bundle.shift();
+    }
+    for (i = 0; i < this.media.length; i++) {
+        mline = SDPUtil.parse_mline(this.media[i].split('\r\n')[0]);
+        if (!(mline.media == 'audio' || mline.media == 'video')) {
+            continue;
+        }
+        if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {
+            ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first
+        } else {
+            ssrc = false;
+        }
+
+        elem.c('content', {creator: thecreator, name: mline.media});
+        if (SDPUtil.find_line(this.media[i], 'a=mid:')) {
+            // prefer identifier from a=mid if present
+            var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));
+            elem.attrs({ name: mid });
+
+            // old BUNDLE plan, to be removed
+            if (bundle.indexOf(mid) != -1) {
+                elem.c('bundle', {xmlns: 'http://estos.de/ns/bundle'}).up();
+                bundle.splice(bundle.indexOf(mid), 1);
+            }
+        }
+        if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length) {
+            elem.c('description',
+                 {xmlns: 'urn:xmpp:jingle:apps:rtp:1',
+                  media: mline.media });
+            if (ssrc) {
+                elem.attrs({ssrc: ssrc});
+            }
+            for (j = 0; j < mline.fmt.length; j++) {
+                rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);
+                elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));
+                // put any 'a=fmtp:' + mline.fmt[j] lines into <param name=foo value=bar/>
+                if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {
+                    tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));
+                    for (k = 0; k < tmp.length; k++) {
+                        elem.c('parameter', tmp[k]).up();
+                    }
+                }
+                this.RtcpFbToJingle(i, elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb
+
+                elem.up();
+            }
+            if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {
+                elem.c('encryption', {required: 1});
+                var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);
+                crypto.forEach(function(line) {
+                    elem.c('crypto', SDPUtil.parse_crypto(line)).up();
+                });
+                elem.up(); // end of encryption
+            }
+
+            if (ssrc) {
+                // new style mapping
+                elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+                // FIXME: group by ssrc and support multiple different ssrcs
+                var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');
+                ssrclines.forEach(function(line) {
+                    idx = line.indexOf(' ');
+                    var linessrc = line.substr(0, idx).substr(7);
+                    if (linessrc != ssrc) {
+                        elem.up();
+                        ssrc = linessrc;
+                        elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+                    }
+                    var kv = line.substr(idx + 1);
+                    elem.c('parameter');
+                    if (kv.indexOf(':') == -1) {
+                        elem.attrs({ name: kv });
+                    } else {
+                        elem.attrs({ name: kv.split(':', 2)[0] });
+                        elem.attrs({ value: kv.split(':', 2)[1] });
+                    }
+                    elem.up();
+                });
+                elem.up();
+
+                // old proprietary mapping, to be removed at some point
+                tmp = SDPUtil.parse_ssrc(this.media[i]);
+                tmp.xmlns = 'http://estos.de/ns/ssrc';
+                tmp.ssrc = ssrc;
+                elem.c('ssrc', tmp).up(); // ssrc is part of description
+            }
+
+            if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {
+                elem.c('rtcp-mux').up();
+            }
+
+            // XEP-0293 -- map a=rtcp-fb:*
+            this.RtcpFbToJingle(i, elem, '*');
+
+            // XEP-0294
+            if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {
+                lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');
+                for (j = 0; j < lines.length; j++) {
+                    tmp = SDPUtil.parse_extmap(lines[j]);
+                    elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',
+                                    uri: tmp.uri,
+                                    id: tmp.value });
+                    if (tmp.hasOwnProperty('direction')) {
+                        switch (tmp.direction) {
+                        case 'sendonly':
+                            elem.attrs({senders: 'responder'});
+                            break;
+                        case 'recvonly':
+                            elem.attrs({senders: 'initiator'});
+                            break;
+                        case 'sendrecv':
+                            elem.attrs({senders: 'both'});
+                            break;
+                        case 'inactive':
+                            elem.attrs({senders: 'none'});
+                            break;
+                        }
+                    }
+                    // TODO: handle params
+                    elem.up();
+                }
+            }
+            elem.up(); // end of description
+        }
+
+        // map ice-ufrag/pwd, dtls fingerprint, candidates
+        this.TransportToJingle(i, elem);
+
+        if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {
+            elem.attrs({senders: 'both'});
+        } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {
+            elem.attrs({senders: 'initiator'});
+        } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {
+            elem.attrs({senders: 'responder'});
+        } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {
+            elem.attrs({senders: 'none'});
+        }
+        if (mline.port == '0') {
+            // estos hack to reject an m-line
+            elem.attrs({senders: 'rejected'});
+        }
+        elem.up(); // end of content
+    }
+    elem.up();
+    return elem;
+};
+
+SDP.prototype.TransportToJingle = function (mediaindex, elem) {
+    var i = mediaindex;
+    var tmp;
+    var self = this;
+    elem.c('transport');
+
+    // XEP-0320
+    var fingerprints = SDPUtil.find_lines(this.media[mediaindex], 'a=fingerprint:', this.session);
+    fingerprints.forEach(function(line) {
+        tmp = SDPUtil.parse_fingerprint(line);
+        tmp.xmlns = 'urn:xmpp:tmp:jingle:apps:dtls:0';
+        // tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0'; -- FIXME: update receivers first
+        elem.c('fingerprint').t(tmp.fingerprint);
+        delete tmp.fingerprint;
+        line = SDPUtil.find_line(self.media[mediaindex], 'a=setup:', self.session);
+        if (line) {
+            tmp.setup = line.substr(8);
+        }
+        elem.attrs(tmp);
+        elem.up(); // end of fingerprint
+    });
+    tmp = SDPUtil.iceparams(this.media[mediaindex], this.session);
+    if (tmp) {
+        tmp.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+        elem.attrs(tmp);
+        // XEP-0176
+        if (SDPUtil.find_line(this.media[mediaindex], 'a=candidate:', this.session)) { // add any a=candidate lines
+            var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=candidate:', this.session);
+            lines.forEach(function (line) {
+                elem.c('candidate', SDPUtil.candidateToJingle(line)).up();
+            });
+        }
+    }
+    elem.up(); // end of transport
+}
+
+SDP.prototype.RtcpFbToJingle = function (mediaindex, elem, payloadtype) { // XEP-0293
+    var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=rtcp-fb:' + payloadtype);
+    lines.forEach(function (line) {
+        var tmp = SDPUtil.parse_rtcpfb(line);
+        if (tmp.type == 'trr-int') {
+            elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});
+            elem.up();
+        } else {
+            elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});
+            if (tmp.params.length > 0) {
+                elem.attrs({'subtype': tmp.params[0]});
+            }
+            elem.up();
+        }
+    });
+};
+
+SDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293
+    var media = '';
+    var tmp = elem.find('>rtcp-fb-trr-int[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
+    if (tmp.length) {
+        media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';
+        if (tmp.attr('value')) {
+            media += tmp.attr('value');
+        } else {
+            media += '0';
+        }
+        media += '\r\n';
+    }
+    tmp = elem.find('>rtcp-fb[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
+    tmp.each(function () {
+        media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');
+        if ($(this).attr('subtype')) {
+            media += ' ' + $(this).attr('subtype');
+        }
+        media += '\r\n';
+    });
+    return media;
+};
+
+// construct an SDP from a jingle stanza
+SDP.prototype.fromJingle = function (jingle) {
+    var self = this;
+    this.raw = 'v=0\r\n' +
+        'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
+        's=-\r\n' +
+        't=0 0\r\n';
+    // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8
+    if ($(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').length) {
+        $(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').each(function (idx, group) {
+            var contents = $(group).find('>content').map(function (idx, content) {
+                return content.getAttribute('name');
+            }).get();
+            if (contents.length > 0) {
+                self.raw += 'a=group:' + (group.getAttribute('semantics') || group.getAttribute('type')) + ' ' + contents.join(' ') + '\r\n';
+            }
+        });
+    } else if ($(jingle).find('>group[xmlns="urn:ietf:rfc:5888"]').length) {
+        // temporary namespace, not to be used. to be removed soon.
+        $(jingle).find('>group[xmlns="urn:ietf:rfc:5888"]').each(function (idx, group) {
+            var contents = $(group).find('>content').map(function (idx, content) {
+                return content.getAttribute('name');
+            }).get();
+            if (group.getAttribute('type') !== null && contents.length > 0) {
+                self.raw += 'a=group:' + group.getAttribute('type') + ' ' + contents.join(' ') + '\r\n';
+            }
+        });
+    } else {
+        // for backward compability, to be removed soon
+        // assume all contents are in the same bundle group, can be improved upon later
+        var bundle = $(jingle).find('>content').filter(function (idx, content) {
+            //elem.c('bundle', {xmlns:'http://estos.de/ns/bundle'});
+            return $(content).find('>bundle').length > 0;
+        }).map(function (idx, content) {
+            return content.getAttribute('name');
+        }).get();
+        if (bundle.length) {
+            this.raw += 'a=group:BUNDLE ' + bundle.join(' ') + '\r\n';
+        }
+    }
+
+    this.session = this.raw;
+    jingle.find('>content').each(function () {
+        var m = self.jingle2media($(this));
+        self.media.push(m);
+    });
+
+    // reconstruct msid-semantic -- apparently not necessary
+    /*
+    var msid = SDPUtil.parse_ssrc(this.raw);
+    if (msid.hasOwnProperty('mslabel')) {
+        this.session += "a=msid-semantic: WMS " + msid.mslabel + "\r\n";
+    }
+    */
+
+    this.raw = this.session + this.media.join('');
+};
+
+// translate a jingle content element into an an SDP media part
+SDP.prototype.jingle2media = function (content) {
+    var media = '',
+        desc = content.find('description'),
+        ssrc = desc.attr('ssrc'),
+        self = this,
+        tmp;
+
+    tmp = { media: desc.attr('media') };
+    tmp.port = '1';
+    if (content.attr('senders') == 'rejected') {
+        // estos hack to reject an m-line.
+        tmp.port = '0';
+    }
+    if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {
+        tmp.proto = 'RTP/SAVPF';
+    } else {
+        tmp.proto = 'RTP/AVPF';
+    }
+    tmp.fmt = desc.find('payload-type').map(function () { return this.getAttribute('id'); }).get();
+    media += SDPUtil.build_mline(tmp) + '\r\n';
+    media += 'c=IN IP4 0.0.0.0\r\n';
+    media += 'a=rtcp:1 IN IP4 0.0.0.0\r\n';
+    tmp = content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]');
+    if (tmp.length) {
+        if (tmp.attr('ufrag')) {
+            media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\r\n';
+        }
+        if (tmp.attr('pwd')) {
+            media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\r\n';
+        }
+        tmp.find('>fingerprint').each(function () {
+            // FIXME: check namespace at some point
+            media += 'a=fingerprint:' + this.getAttribute('hash');
+            media += ' ' + $(this).text();
+            media += '\r\n';
+            if (this.getAttribute('setup')) {
+                media += 'a=setup:' + this.getAttribute('setup') + '\r\n';
+            }
+        });
+    }
+    switch (content.attr('senders')) {
+    case 'initiator':
+        media += 'a=sendonly\r\n';
+        break;
+    case 'responder':
+        media += 'a=recvonly\r\n';
+        break;
+    case 'none':
+        media += 'a=inactive\r\n';
+        break;
+    case 'both':
+        media += 'a=sendrecv\r\n';
+        break;
+    }
+    media += 'a=mid:' + content.attr('name') + '\r\n';
+
+    // <description><rtcp-mux/></description>
+    // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though
+    // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html
+    if (desc.find('rtcp-mux').length) {
+        media += 'a=rtcp-mux\r\n';
+    }
+
+    if (desc.find('encryption').length) {
+        desc.find('encryption>crypto').each(function () {
+            media += 'a=crypto:' + this.getAttribute('tag');
+            media += ' ' + this.getAttribute('crypto-suite');
+            media += ' ' + this.getAttribute('key-params');
+            if (this.getAttribute('session-params')) {
+                media += ' ' + this.getAttribute('session-params');
+            }
+            media += '\r\n';
+        });
+    }
+    desc.find('payload-type').each(function () {
+        media += SDPUtil.build_rtpmap(this) + '\r\n';
+        if ($(this).find('>parameter').length) {
+            media += 'a=fmtp:' + this.getAttribute('id') + ' ';
+            media += $(this).find('parameter').map(function () { return (this.getAttribute('name') ? (this.getAttribute('name') + '=') : '') + this.getAttribute('value'); }).get().join(';');
+            media += '\r\n';
+        }
+        // xep-0293
+        media += self.RtcpFbFromJingle($(this), this.getAttribute('id'));
+    });
+
+    // xep-0293
+    media += self.RtcpFbFromJingle(desc, '*');
+
+    // xep-0294
+    tmp = desc.find('>rtp-hdrext[xmlns="urn:xmpp:jingle:apps:rtp:rtp-hdrext:0"]');
+    tmp.each(function () {
+        media += 'a=extmap:' + this.getAttribute('id') + ' ' + this.getAttribute('uri') + '\r\n';
+    });
+
+    content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]>candidate').each(function () {
+        media += SDPUtil.candidateFromJingle(this);
+    });
+
+    tmp = content.find('description>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+    tmp.each(function () {
+        var ssrc = this.getAttribute('ssrc');
+        $(this).find('>parameter').each(function () {
+            media += 'a=ssrc:' + ssrc + ' ' + this.getAttribute('name');
+            if (this.getAttribute('value') && this.getAttribute('value').length)
+                media += ':' + this.getAttribute('value');
+            media += '\r\n';
+        });
+    });
+
+    if (tmp.length === 0) {
+        // fallback to proprietary mapping of a=ssrc lines
+        tmp = content.find('description>ssrc[xmlns="http://estos.de/ns/ssrc"]');
+        if (tmp.length) {
+            media += 'a=ssrc:' + ssrc + ' cname:' + tmp.attr('cname') + '\r\n';
+            media += 'a=ssrc:' + ssrc + ' msid:' + tmp.attr('msid') + '\r\n';
+            media += 'a=ssrc:' + ssrc + ' mslabel:' + tmp.attr('mslabel') + '\r\n';
+            media += 'a=ssrc:' + ssrc + ' label:' + tmp.attr('label') + '\r\n';
+        }
+    }
+    return media;
+};
+
+SDPUtil = {
+    iceparams: function (mediadesc, sessiondesc) {
+        var data = null;
+        if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
+            SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
+            data = {
+                ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
+                pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
+            };
+        }
+        return data;
+    },
+    parse_iceufrag: function (line) {
+        return line.substring(12);
+    },
+    build_iceufrag: function (frag) {
+        return 'a=ice-ufrag:' + frag;
+    },
+    parse_icepwd: function (line) {
+        return line.substring(10);
+    },
+    build_icepwd: function (pwd) {
+        return 'a=ice-pwd:' + pwd;
+    },
+    parse_mid: function (line) {
+        return line.substring(6);
+    },
+    parse_mline: function (line) {
+        var parts = line.substring(2).split(' '),
+        data = {};
+        data.media = parts.shift();
+        data.port = parts.shift();
+        data.proto = parts.shift();
+        if (parts[parts.length - 1] === '') { // trailing whitespace
+            parts.pop();
+        }
+        data.fmt = parts;
+        return data;
+    },
+    build_mline: function (mline) {
+        return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
+    },
+    parse_rtpmap: function (line) {
+        var parts = line.substring(9).split(' '),
+            data = {};
+        data.id = parts.shift();
+        parts = parts[0].split('/');
+        data.name = parts.shift();
+        data.clockrate = parts.shift();
+        data.channels = parts.length ? parts.shift() : '1';
+        return data;
+    },
+    build_rtpmap: function (el) {
+        var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
+        if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
+            line += '/' + el.getAttribute('channels');
+        }
+        return line;
+    },
+    parse_crypto: function (line) {
+        var parts = line.substring(9).split(' '),
+        data = {};
+        data.tag = parts.shift();
+        data['crypto-suite'] = parts.shift();
+        data['key-params'] = parts.shift();
+        if (parts.length) {
+            data['session-params'] = parts.join(' ');
+        }
+        return data;
+    },
+    parse_fingerprint: function (line) { // RFC 4572
+        var parts = line.substring(14).split(' '),
+        data = {};
+        data.hash = parts.shift();
+        data.fingerprint = parts.shift();
+        // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
+        return data;
+    },
+    parse_fmtp: function (line) {
+        var parts = line.split(' '),
+            i, key, value,
+            data = [];
+        parts.shift();
+        parts = parts.join(' ').split(';');
+        for (i = 0; i < parts.length; i++) {
+            key = parts[i].split('=')[0];
+            while (key.length && key[0] == ' ') {
+                key = key.substring(1);
+            }
+            value = parts[i].split('=')[1];
+            if (key && value) {
+                data.push({name: key, value: value});
+            } else if (key) {
+                // rfc 4733 (DTMF) style stuff
+                data.push({name: '', value: key});
+            }
+        }
+        return data;
+    },
+    parse_icecandidate: function (line) {
+        var candidate = {},
+            elems = line.split(' ');
+        candidate.foundation = elems[0].substring(12);
+        candidate.component = elems[1];
+        candidate.protocol = elems[2].toLowerCase();
+        candidate.priority = elems[3];
+        candidate.ip = elems[4];
+        candidate.port = elems[5];
+        // elems[6] => "typ"
+        candidate.type = elems[7];
+        candidate.generation = 0; // default value, may be overwritten below
+        for (var i = 8; i < elems.length; i += 2) {
+            switch (elems[i]) {
+            case 'raddr':
+                candidate['rel-addr'] = elems[i + 1];
+                break;
+            case 'rport':
+                candidate['rel-port'] = elems[i + 1];
+                break;
+            case 'generation':
+                candidate.generation = elems[i + 1];
+                break;
+            case 'tcptype':
+                candidate.tcptype = elems[i + 1];
+                break;
+            default: // TODO
+                console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+            }
+        }
+        candidate.network = '1';
+        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+        return candidate;
+    },
+    build_icecandidate: function (cand) {
+        var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
+        line += ' ';
+        switch (cand.type) {
+        case 'srflx':
+        case 'prflx':
+        case 'relay':
+            if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
+                line += 'raddr';
+                line += ' ';
+                line += cand['rel-addr'];
+                line += ' ';
+                line += 'rport';
+                line += ' ';
+                line += cand['rel-port'];
+                line += ' ';
+            }
+            break;
+        }
+        if (cand.hasOwnAttribute('tcptype')) {
+            line += 'tcptype';
+            line += ' ';
+            line += cand.tcptype;
+            line += ' ';
+        }
+        line += 'generation';
+        line += ' ';
+        line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
+        return line;
+    },
+    parse_ssrc: function (desc) {
+        // proprietary mapping of a=ssrc lines
+        // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
+        // and parse according to that
+        var lines = desc.split('\r\n'),
+            data = {};
+        for (var i = 0; i < lines.length; i++) {
+            if (lines[i].substring(0, 7) == 'a=ssrc:') {
+                var idx = lines[i].indexOf(' ');
+                data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
+            }
+        }
+        return data;
+    },
+    parse_rtcpfb: function (line) {
+        var parts = line.substr(10).split(' ');
+        var data = {};
+        data.pt = parts.shift();
+        data.type = parts.shift();
+        data.params = parts;
+        return data;
+    },
+    parse_extmap: function (line) {
+        var parts = line.substr(9).split(' ');
+        var data = {};
+        data.value = parts.shift();
+        if (data.value.indexOf('/') != -1) {
+            data.direction = data.value.substr(data.value.indexOf('/') + 1);
+            data.value = data.value.substr(0, data.value.indexOf('/'));
+        } else {
+            data.direction = 'both';
+        }
+        data.uri = parts.shift();
+        data.params = parts;
+        return data;
+    },
+    find_line: function (haystack, needle, sessionpart) {
+        var lines = haystack.split('\r\n');
+        for (var i = 0; i < lines.length; i++) {
+            if (lines[i].substring(0, needle.length) == needle) {
+                return lines[i];
+            }
+        }
+        if (!sessionpart) {
+            return false;
+        }
+        // search session part
+        lines = sessionpart.split('\r\n');
+        for (var j = 0; j < lines.length; j++) {
+            if (lines[j].substring(0, needle.length) == needle) {
+                return lines[j];
+            }
+        }
+        return false;
+    },
+    find_lines: function (haystack, needle, sessionpart) {
+        var lines = haystack.split('\r\n'),
+            needles = [];
+        for (var i = 0; i < lines.length; i++) {
+            if (lines[i].substring(0, needle.length) == needle)
+                needles.push(lines[i]);
+        }
+        if (needles.length || !sessionpart) {
+            return needles;
+        }
+        // search session part
+        lines = sessionpart.split('\r\n');
+        for (var j = 0; j < lines.length; j++) {
+            if (lines[j].substring(0, needle.length) == needle) {
+                needles.push(lines[j]);
+            }
+        }
+        return needles;
+    },
+    candidateToJingle: function (line) {
+        // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
+        //      <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>
+        if (line.indexOf('candidate:') === 0) {
+            line = 'a=' + line;
+        } else if (line.substring(0, 12) != 'a=candidate:') {            
+			console.log('parseCandidate called with a line that is not a candidate line');
+            console.log(line);
+            return null;
+        }
+        if (line.substring(line.length - 2) == '\r\n') // chomp it
+            line = line.substring(0, line.length - 2);
+        var candidate = {},
+            elems = line.split(' '),
+            i;
+        if (elems[6] != 'typ') {
+            console.log('did not find typ in the right place');
+            console.log(line);
+            return null;
+        }
+        candidate.foundation = elems[0].substring(12);
+        candidate.component = elems[1];
+        candidate.protocol = elems[2].toLowerCase();
+        candidate.priority = elems[3];
+        candidate.ip = elems[4];
+        candidate.port = elems[5];
+        // elems[6] => "typ"
+        candidate.type = elems[7];
+
+        candidate.generation = '0'; // fippo from jitsi-meet: default, may be overwritten below
+
+        for (i = 8; i < elems.length; i += 2) {
+            switch (elems[i]) {
+            case 'raddr':
+                candidate['rel-addr'] = elems[i + 1];
+                break;
+            case 'rport':
+                candidate['rel-port'] = elems[i + 1];
+                break;
+            case 'generation':
+                candidate.generation = elems[i + 1];
+                break;
+            case 'tcptype':
+                candidate.tcptype = elems[i + 1];
+                break;
+            default: // TODO
+                console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+            }
+        }
+        candidate.network = '1';
+        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+        return candidate;
+    },
+    candidateFromJingle: function (cand) {
+        var line = 'a=candidate:';
+        line += cand.getAttribute('foundation');
+        line += ' ';
+        line += cand.getAttribute('component');
+        line += ' ';
+        line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
+        line += ' ';
+        line += cand.getAttribute('priority');
+        line += ' ';
+        line += cand.getAttribute('ip');
+        line += ' ';
+        line += cand.getAttribute('port');
+        line += ' ';
+        line += 'typ';
+        line += ' ' + cand.getAttribute('type');
+        line += ' ';
+        switch (cand.getAttribute('type')) {
+        case 'srflx':
+        case 'prflx':
+        case 'relay':
+            if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
+                line += 'raddr';
+                line += ' ';
+                line += cand.getAttribute('rel-addr');
+                line += ' ';
+                line += 'rport';
+                line += ' ';
+                line += cand.getAttribute('rel-port');
+                line += ' ';
+            }
+            break;
+        }
+        line += 'generation';
+        line += ' ';
+        line += cand.getAttribute('generation') || '0';
+        return line + '\r\n';
+    }
+};
+}(jQuery));
+
+/*!
+ * Source: build/lib/strophe.jingle/strophe.jingle.adapter.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+/* jshint -W117 */
+var setupRTC, getUserMediaWithConstraints, TraceablePeerConnection;
+
+(function($){
+TraceablePeerConnection = function(ice_config, constraints) {
+    var self = this;
+    var RTCPeerconnection = navigator.mozGetUserMedia ? mozRTCPeerConnection : webkitRTCPeerConnection;
+    this.peerconnection = new RTCPeerconnection(ice_config, constraints);
+    this.updateLog = [];
+    this.stats = {};
+    this.statsinterval = null;
+    this.maxstats = 300; // limit to 300 values, i.e. 5 minutes; set to 0 to disable
+
+    // override as desired
+    this.trace = function(what, info) {
+        //console.warn('WTRACE', what, info);
+        self.updateLog.push({
+            time: new Date(),
+            type: what,
+            value: info || ""
+        });
+    };
+    this.onicecandidate = null;
+    this.peerconnection.onicecandidate = function (event) {
+        self.trace('onicecandidate', JSON.stringify(event.candidate, null, ' '));
+        if (self.onicecandidate !== null) {
+            self.onicecandidate(event);
+        }
+    };
+    this.onaddstream = null;
+    this.peerconnection.onaddstream = function (event) {
+        self.trace('onaddstream', event.stream.id);
+        if (self.onaddstream !== null) {
+            self.onaddstream(event);
+        }
+    };
+    this.onremovestream = null;
+    this.peerconnection.onremovestream = function (event) {
+        self.trace('onremovestream', event.stream.id);
+        if (self.onremovestream !== null) {
+            self.onremovestream(event);
+        }
+    };
+    this.onsignalingstatechange = null;
+    this.peerconnection.onsignalingstatechange = function (event) {
+        self.trace('onsignalingstatechange', self.signalingState);
+        if (self.onsignalingstatechange !== null) {
+            self.onsignalingstatechange(event);
+        }
+    };
+    this.oniceconnectionstatechange = null;
+    this.peerconnection.oniceconnectionstatechange = function (event) {
+        self.trace('oniceconnectionstatechange', self.iceConnectionState);
+        if (self.oniceconnectionstatechange !== null) {
+            self.oniceconnectionstatechange(event);
+        }
+    };
+    this.onnegotiationneeded = null;
+    this.peerconnection.onnegotiationneeded = function (event) {
+        self.trace('onnegotiationneeded');
+        if (self.onnegotiationneeded !== null) {
+            self.onnegotiationneeded(event);
+        }
+    };
+    self.ondatachannel = null;
+    this.peerconnection.ondatachannel = function (event) {
+        self.trace('ondatachannel', event);
+        if (self.ondatachannel !== null) {
+            self.ondatachannel(event);
+        }
+    }
+    if (!navigator.mozGetUserMedia) {
+        this.statsinterval = window.setInterval(function() {
+            self.peerconnection.getStats(function(stats) {
+                var results = stats.result();
+                for (var i = 0; i < results.length; ++i) {
+                    //console.log(results[i].type, results[i].id, results[i].names())
+                    var now = new Date();
+                    results[i].names().forEach(function (name) {
+                        var id = results[i].id + '-' + name;
+                        if (!self.stats[id]) {
+                            self.stats[id] = {
+                                startTime: now,
+                                endTime: now,
+                                values: [],
+                                times: []
+                            };
+                        }
+                        self.stats[id].values.push(results[i].stat(name));
+                        self.stats[id].times.push(now.getTime());
+                        if (self.stats[id].values.length > self.maxstats) {
+                            self.stats[id].values.shift();
+                            self.stats[id].times.shift();
+                        }
+                        self.stats[id].endTime = now;
+                    });
+                }
+            });
+
+        }, 1000);
+    }
+};
+
+dumpSDP = function(description) {
+    return 'type: ' + description.type + '\r\n' + description.sdp;
+}
+
+if (TraceablePeerConnection.prototype.__defineGetter__ !== undefined) {
+    TraceablePeerConnection.prototype.__defineGetter__('signalingState', function() { return this.peerconnection.signalingState; });
+    TraceablePeerConnection.prototype.__defineGetter__('iceConnectionState', function() { return this.peerconnection.iceConnectionState; });
+    TraceablePeerConnection.prototype.__defineGetter__('localDescription', function() { return this.peerconnection.localDescription; });
+    TraceablePeerConnection.prototype.__defineGetter__('remoteDescription', function() { return this.peerconnection.remoteDescription; });
+}
+
+TraceablePeerConnection.prototype.addStream = function (stream) {
+    this.trace('addStream', stream.id);
+    this.peerconnection.addStream(stream);
+};
+
+TraceablePeerConnection.prototype.removeStream = function (stream) {
+    this.trace('removeStream', stream.id);
+    this.peerconnection.removeStream(stream);
+};
+
+TraceablePeerConnection.prototype.createDataChannel = function (label, opts) {
+    this.trace('createDataChannel', label, opts);
+    this.peerconnection.createDataChannel(label, opts);
+}
+
+TraceablePeerConnection.prototype.setLocalDescription = function (description, successCallback, failureCallback) {
+    var self = this;
+    this.trace('setLocalDescription', dumpSDP(description));
+    this.peerconnection.setLocalDescription(description, 
+        function () {
+            self.trace('setLocalDescriptionOnSuccess');
+            successCallback();
+        },
+        function (err) {
+            self.trace('setLocalDescriptionOnFailure', err);
+            failureCallback(err);
+        }
+    );
+    /*
+    if (this.statsinterval === null && this.maxstats > 0) {
+        // start gathering stats
+    }
+    */
+};
+
+TraceablePeerConnection.prototype.setRemoteDescription = function (description, successCallback, failureCallback) {
+    var self = this;
+    this.trace('setRemoteDescription', dumpSDP(description));
+    this.peerconnection.setRemoteDescription(description, 
+        function () {
+            self.trace('setRemoteDescriptionOnSuccess');
+            successCallback();
+        },
+        function (err) {
+            self.trace('setRemoteDescriptionOnFailure', err);
+            failureCallback(err);
+        }
+    );
+    /*
+    if (this.statsinterval === null && this.maxstats > 0) {
+        // start gathering stats
+    }
+    */
+};
+
+TraceablePeerConnection.prototype.close = function () {
+    this.trace('stop');
+    if (this.statsinterval !== null) {
+        window.clearInterval(this.statsinterval);
+        this.statsinterval = null;
+    }
+    this.peerconnection.close();
+};
+
+TraceablePeerConnection.prototype.createOffer = function (successCallback, failureCallback, constraints) {
+    var self = this;
+    this.trace('createOffer', JSON.stringify(constraints, null, ' '));
+    this.peerconnection.createOffer(
+        function (offer) {
+            self.trace('createOfferOnSuccess', dumpSDP(offer));
+            successCallback(offer);
+        },
+        function(err) {
+            self.trace('createOfferOnFailure', err);
+            failureCallback(err);
+        },
+        constraints
+    );
+};
+
+TraceablePeerConnection.prototype.createAnswer = function (successCallback, failureCallback, constraints) {
+    var self = this;
+    this.trace('createAnswer', JSON.stringify(constraints, null, ' '));
+    this.peerconnection.createAnswer(
+        function (answer) {
+            self.trace('createAnswerOnSuccess', dumpSDP(answer));
+            successCallback(answer);
+        },
+        function(err) {
+            self.trace('createAnswerOnFailure', err);
+            failureCallback(err);
+        },
+        constraints
+    );
+};
+
+TraceablePeerConnection.prototype.addIceCandidate = function (candidate, successCallback, failureCallback) {
+    var self = this;
+    this.trace('addIceCandidate', JSON.stringify(candidate, null, ' '));
+    this.peerconnection.addIceCandidate(candidate);
+    /* maybe later
+    this.peerconnection.addIceCandidate(candidate, 
+        function () {                                
+            self.trace('addIceCandidateOnSuccess');
+            successCallback();
+        },
+        function (err) {
+            self.trace('addIceCandidateOnFailure', err);
+            failureCallback(err);
+        }
+    );
+    */
+};
+
+TraceablePeerConnection.prototype.getStats = function(callback, errback) {
+    if (navigator.mozGetUserMedia) {
+        // ignore for now...
+    } else {
+        this.peerconnection.getStats(callback);
+    }
+};
+
+// mozilla chrome compat layer -- very similar to adapter.js
+setupRTC = function (){
+    var RTC = null;
+    if (navigator.mozGetUserMedia) {
+        console.log('This appears to be Firefox');
+        var version = parseInt(navigator.userAgent.match(/Firefox\/([0-9]+)\./)[1], 10);
+        if (version >= 22) {
+            RTC = {
+                peerconnection: mozRTCPeerConnection,
+                browser: 'firefox',
+                getUserMedia: navigator.mozGetUserMedia.bind(navigator),
+                attachMediaStream: function (element, stream) {
+                    element[0].mozSrcObject = stream;
+                    element[0].play();
+                },
+                pc_constraints: {}
+            };
+            if (!MediaStream.prototype.getVideoTracks)
+                MediaStream.prototype.getVideoTracks = function () { return []; };
+            if (!MediaStream.prototype.getAudioTracks)
+                MediaStream.prototype.getAudioTracks = function () { return []; };
+            RTCSessionDescription = mozRTCSessionDescription;
+            RTCIceCandidate = mozRTCIceCandidate;
+        }
+    } else if (navigator.webkitGetUserMedia) {
+        console.log('This appears to be Chrome');
+        RTC = {
+            peerconnection: webkitRTCPeerConnection,
+            browser: 'chrome',
+            getUserMedia: navigator.webkitGetUserMedia.bind(navigator),
+            attachMediaStream: function (element, stream) {
+                element.attr('src', webkitURL.createObjectURL(stream));
+            },
+            // DTLS should now be enabled by default but..
+            pc_constraints: {'optional': [{'DtlsSrtpKeyAgreement': 'true'}]} 
+        };
+        if (navigator.userAgent.indexOf('Android') != -1) {
+            RTC.pc_constraints = {}; // disable DTLS on Android
+        }
+        if (!webkitMediaStream.prototype.getVideoTracks) {
+            webkitMediaStream.prototype.getVideoTracks = function () {
+                return this.videoTracks;
+            };
+        }
+        if (!webkitMediaStream.prototype.getAudioTracks) {
+            webkitMediaStream.prototype.getAudioTracks = function () {
+                return this.audioTracks;
+            };
+        }
+    }
+    if (RTC === null) {
+        try { console.log('Browser does not appear to be WebRTC-capable'); } catch (e) { }
+    }
+    return RTC;
+};
+
+getUserMediaWithConstraints = function(um, resolution, bandwidth, fps) {
+    var constraints = {audio: false, video: false};
+
+    if (um.indexOf('video') >= 0) {
+        constraints.video = {mandatory: {}};// same behaviour as true
+    }
+    if (um.indexOf('audio') >= 0) {
+        constraints.audio = {};// same behaviour as true
+    }
+    if (um.indexOf('screen') >= 0) {
+        constraints.video = {
+            "mandatory": {
+                "chromeMediaSource": "screen"
+            }
+        };
+    }
+
+    if (resolution && !constraints.video) {
+        constraints.video = {mandatory: {}};// same behaviour as true
+    }
+    // see https://code.google.com/p/chromium/issues/detail?id=143631#c9 for list of supported resolutions
+    switch (resolution) {
+    // 16:9 first
+    case '1080':
+    case 'fullhd':
+        constraints.video.mandatory.minWidth = 1920;
+        constraints.video.mandatory.minHeight = 1080;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+    case '720':
+    case 'hd':
+        constraints.video.mandatory.minWidth = 1280;
+        constraints.video.mandatory.minHeight = 720;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+    case '360':
+        constraints.video.mandatory.minWidth = 640;
+        constraints.video.mandatory.minHeight = 360;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+    case '180':
+        constraints.video.mandatory.minWidth = 320;
+        constraints.video.mandatory.minHeight = 180;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+        // 4:3
+    case '960':
+        constraints.video.mandatory.minWidth = 960;
+        constraints.video.mandatory.minHeight = 720;
+        break;
+    case '640':
+    case 'vga':
+        constraints.video.mandatory.minWidth = 640;
+        constraints.video.mandatory.minHeight = 480;
+        break;
+    case '320':
+        constraints.video.mandatory.minWidth = 320;
+        constraints.video.mandatory.minHeight = 240;
+        break;
+    default:
+        if (navigator.userAgent.indexOf('Android') != -1) {
+            constraints.video.mandatory.minWidth = 320;
+            constraints.video.mandatory.minHeight = 240;
+            constraints.video.mandatory.maxFrameRate = 15;
+        }
+        break;
+    }
+
+    if (bandwidth) { // doesn't work currently, see webrtc issue 1846
+        if (!constraints.video) constraints.video = {mandatory: {}};//same behaviour as true
+        constraints.video.optional = [{bandwidth: bandwidth}];
+    }
+    if (fps) { // for some cameras it might be necessary to request 30fps
+        // so they choose 30fps mjpg over 10fps yuy2
+        if (!constraints.video) constraints.video = {mandatory: {}};// same behaviour as tru;
+        constraints.video.mandatory.minFrameRate = fps;
+    }
+ 
+    try {
+        RTC.getUserMedia(constraints,
+                function (stream) {
+                    console.log('onUserMediaSuccess');
+                    $(document).trigger('mediaready.jingle', [stream]);
+                },
+                function (error) {
+                    console.warn('Failed to get access to local media. Error ', error);
+                    $(document).trigger('mediafailure.jingle', [error]);
+                });
+    } catch (e) {
+        console.error('GUM failed: ', e);
+        $(document).trigger('mediafailure.jingle');
+    }
+}
+}(jQuery));
+
+/*!
+ * Source: build/lib/otr/build/dep/salsa20.js, license: AGPL3, url: https://github.com/neoatlantis/node-salsa20 */
+// Salsa20 implementation
+// Contributed to Cryptocat by Dmitry Chestnykh
+// 21-01-2013
+
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define(factory)
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory()
+  } else {
+    root.Salsa20 = factory()
+  }
+
+}(this, function () {
+
+    function Salsa20(key, nonce) {
+        // Constants.
+        this.rounds = 20; // number of Salsa rounds
+        this.sigmaWords = [0x61707865, 0x3320646e, 0x79622d32, 0x6b206574];
+
+        // State.
+        this.keyWords = [];           // key words
+        this.nonceWords = [0, 0];     // nonce words
+        this.counterWords = [0, 0];   // block counter words
+
+        // Output buffer.
+        this.block = [];        // output block of 64 bytes
+        this.blockUsed = 64;     // number of block bytes used
+
+        this.setKey(key);
+        this.setNonce(nonce);
+    }
+
+    // setKey sets the key to the given 32-byte array.
+    Salsa20.prototype.setKey = function(key) {
+        for (var i = 0, j = 0; i < 8; i++, j += 4) {
+            this.keyWords[i] = (key[j] & 0xff)        |
+                              ((key[j+1] & 0xff)<<8)  |
+                              ((key[j+2] & 0xff)<<16) |
+                              ((key[j+3] & 0xff)<<24);
+        }
+        this._reset();
+    };
+
+    // setNonce sets the nonce to the given 8-byte array.
+    Salsa20.prototype.setNonce = function(nonce) {
+        this.nonceWords[0] = (nonce[0] & 0xff)      |
+                            ((nonce[1] & 0xff)<<8)  |
+                            ((nonce[2] & 0xff)<<16) |
+                            ((nonce[3] & 0xff)<<24);
+        this.nonceWords[1] = (nonce[4] & 0xff)      |
+                            ((nonce[5] & 0xff)<<8)  |
+                            ((nonce[6] & 0xff)<<16) |
+                            ((nonce[7] & 0xff)<<24);
+        this._reset();
+    };
+
+    // getBytes returns the next numberOfBytes bytes of stream.
+    Salsa20.prototype.getBytes = function(numberOfBytes) {
+        var out = new Array(numberOfBytes);
+        for (var i = 0; i < numberOfBytes; i++) {
+            if (this.blockUsed == 64) {
+                this._generateBlock();
+                this._incrementCounter();
+                this.blockUsed = 0;
+            }
+            out[i] = this.block[this.blockUsed];
+            this.blockUsed++;
+        }
+        return out;
+    };
+
+    Salsa20.prototype.getHexString = function(numberOfBytes) {
+        var hex=['0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f'];
+        var out = [];
+        var bytes = this.getBytes(numberOfBytes);
+        for(var i = 0; i < bytes.length; i++) {
+            out.push(hex[(bytes[i] >> 4) & 15]);
+            out.push(hex[bytes[i] & 15]);
+        }
+        return out.join('');
+    };
+
+    // Private methods.
+
+    Salsa20.prototype._reset = function() {
+        this.counterWords[0] = 0;
+        this.counterWords[1] = 0;
+        this.blockUsed = 64;
+    };
+
+    // _incrementCounter increments block counter.
+    Salsa20.prototype._incrementCounter = function() {
+        // Note: maximum 2^64 blocks.
+        this.counterWords[0] = (this.counterWords[0] + 1) & 0xffffffff;
+        if (this.counterWords[0] == 0) {
+            this.counterWords[1] = (this.counterWords[1] + 1) & 0xffffffff;
+        }
+    };
+
+    // _generateBlock generates 64 bytes from key, nonce, and counter,
+    // and puts the result into this.block.
+    Salsa20.prototype._generateBlock = function() {
+        var j0 = this.sigmaWords[0],
+            j1 = this.keyWords[0],
+            j2 = this.keyWords[1],
+            j3 = this.keyWords[2],
+            j4 = this.keyWords[3],
+            j5 = this.sigmaWords[1],
+            j6 = this.nonceWords[0],
+            j7 = this.nonceWords[1],
+            j8 = this.counterWords[0],
+            j9 = this.counterWords[1],
+            j10 = this.sigmaWords[2],
+            j11 = this.keyWords[4],
+            j12 = this.keyWords[5],
+            j13 = this.keyWords[6],
+            j14 = this.keyWords[7],
+            j15 = this.sigmaWords[3];
+
+            var x0 = j0, x1 = j1, x2 = j2, x3 = j3, x4 = j4, x5 = j5, x6 = j6, x7 = j7,
+                x8 = j8, x9 = j9, x10 = j10, x11 = j11, x12 = j12, x13 = j13, x14 = j14, x15 = j15;
+
+            var u;
+
+            for (var i = 0; i < this.rounds; i += 2) {
+                u = x0 + x12;
+                x4 ^= (u<<7) | (u>>>(32-7));
+                u = x4 + x0;
+                x8 ^= (u<<9) | (u>>>(32-9));
+                u = x8 + x4;
+                x12 ^= (u<<13) | (u>>>(32-13));
+                u = x12 + x8;
+                x0 ^= (u<<18) | (u>>>(32-18));
+
+                u = x5 + x1;
+                x9 ^= (u<<7) | (u>>>(32-7));
+                u = x9 + x5;
+                x13 ^= (u<<9) | (u>>>(32-9));
+                u = x13 + x9;
+                x1 ^= (u<<13) | (u>>>(32-13));
+                u = x1 + x13;
+                x5 ^= (u<<18) | (u>>>(32-18));
+
+                u = x10 + x6;
+                x14 ^= (u<<7) | (u>>>(32-7));
+                u = x14 + x10;
+                x2 ^= (u<<9) | (u>>>(32-9));
+                u = x2 + x14;
+                x6 ^= (u<<13) | (u>>>(32-13));
+                u = x6 + x2;
+                x10 ^= (u<<18) | (u>>>(32-18));
+
+                u = x15 + x11;
+                x3 ^= (u<<7) | (u>>>(32-7));
+                u = x3 + x15;
+                x7 ^= (u<<9) | (u>>>(32-9));
+                u = x7 + x3;
+                x11 ^= (u<<13) | (u>>>(32-13));
+                u = x11 + x7;
+                x15 ^= (u<<18) | (u>>>(32-18));
+
+                u = x0 + x3;
+                x1 ^= (u<<7) | (u>>>(32-7));
+                u = x1 + x0;
+                x2 ^= (u<<9) | (u>>>(32-9));
+                u = x2 + x1;
+                x3 ^= (u<<13) | (u>>>(32-13));
+                u = x3 + x2;
+                x0 ^= (u<<18) | (u>>>(32-18));
+
+                u = x5 + x4;
+                x6 ^= (u<<7) | (u>>>(32-7));
+                u = x6 + x5;
+                x7 ^= (u<<9) | (u>>>(32-9));
+                u = x7 + x6;
+                x4 ^= (u<<13) | (u>>>(32-13));
+                u = x4 + x7;
+                x5 ^= (u<<18) | (u>>>(32-18));
+
+                u = x10 + x9;
+                x11 ^= (u<<7) | (u>>>(32-7));
+                u = x11 + x10;
+                x8 ^= (u<<9) | (u>>>(32-9));
+                u = x8 + x11;
+                x9 ^= (u<<13) | (u>>>(32-13));
+                u = x9 + x8;
+                x10 ^= (u<<18) | (u>>>(32-18));
+
+                u = x15 + x14;
+                x12 ^= (u<<7) | (u>>>(32-7));
+                u = x12 + x15;
+                x13 ^= (u<<9) | (u>>>(32-9));
+                u = x13 + x12;
+                x14 ^= (u<<13) | (u>>>(32-13));
+                u = x14 + x13;
+                x15 ^= (u<<18) | (u>>>(32-18));
+            }
+
+            x0 += j0;
+            x1 += j1;
+            x2 += j2;
+            x3 += j3;
+            x4 += j4;
+            x5 += j5;
+            x6 += j6;
+            x7 += j7;
+            x8 += j8;
+            x9 += j9;
+            x10 += j10;
+            x11 += j11;
+            x12 += j12;
+            x13 += j13;
+            x14 += j14;
+            x15 += j15;
+
+            this.block[ 0] = ( x0 >>>  0) & 0xff; this.block[ 1] = ( x0 >>>  8) & 0xff;
+            this.block[ 2] = ( x0 >>> 16) & 0xff; this.block[ 3] = ( x0 >>> 24) & 0xff;
+            this.block[ 4] = ( x1 >>>  0) & 0xff; this.block[ 5] = ( x1 >>>  8) & 0xff;
+            this.block[ 6] = ( x1 >>> 16) & 0xff; this.block[ 7] = ( x1 >>> 24) & 0xff;
+            this.block[ 8] = ( x2 >>>  0) & 0xff; this.block[ 9] = ( x2 >>>  8) & 0xff;
+            this.block[10] = ( x2 >>> 16) & 0xff; this.block[11] = ( x2 >>> 24) & 0xff;
+            this.block[12] = ( x3 >>>  0) & 0xff; this.block[13] = ( x3 >>>  8) & 0xff;
+            this.block[14] = ( x3 >>> 16) & 0xff; this.block[15] = ( x3 >>> 24) & 0xff;
+            this.block[16] = ( x4 >>>  0) & 0xff; this.block[17] = ( x4 >>>  8) & 0xff;
+            this.block[18] = ( x4 >>> 16) & 0xff; this.block[19] = ( x4 >>> 24) & 0xff;
+            this.block[20] = ( x5 >>>  0) & 0xff; this.block[21] = ( x5 >>>  8) & 0xff;
+            this.block[22] = ( x5 >>> 16) & 0xff; this.block[23] = ( x5 >>> 24) & 0xff;
+            this.block[24] = ( x6 >>>  0) & 0xff; this.block[25] = ( x6 >>>  8) & 0xff;
+            this.block[26] = ( x6 >>> 16) & 0xff; this.block[27] = ( x6 >>> 24) & 0xff;
+            this.block[28] = ( x7 >>>  0) & 0xff; this.block[29] = ( x7 >>>  8) & 0xff;
+            this.block[30] = ( x7 >>> 16) & 0xff; this.block[31] = ( x7 >>> 24) & 0xff;
+            this.block[32] = ( x8 >>>  0) & 0xff; this.block[33] = ( x8 >>>  8) & 0xff;
+            this.block[34] = ( x8 >>> 16) & 0xff; this.block[35] = ( x8 >>> 24) & 0xff;
+            this.block[36] = ( x9 >>>  0) & 0xff; this.block[37] = ( x9 >>>  8) & 0xff;
+            this.block[38] = ( x9 >>> 16) & 0xff; this.block[39] = ( x9 >>> 24) & 0xff;
+            this.block[40] = (x10 >>>  0) & 0xff; this.block[41] = (x10 >>>  8) & 0xff;
+            this.block[42] = (x10 >>> 16) & 0xff; this.block[43] = (x10 >>> 24) & 0xff;
+            this.block[44] = (x11 >>>  0) & 0xff; this.block[45] = (x11 >>>  8) & 0xff;
+            this.block[46] = (x11 >>> 16) & 0xff; this.block[47] = (x11 >>> 24) & 0xff;
+            this.block[48] = (x12 >>>  0) & 0xff; this.block[49] = (x12 >>>  8) & 0xff;
+            this.block[50] = (x12 >>> 16) & 0xff; this.block[51] = (x12 >>> 24) & 0xff;
+            this.block[52] = (x13 >>>  0) & 0xff; this.block[53] = (x13 >>>  8) & 0xff;
+            this.block[54] = (x13 >>> 16) & 0xff; this.block[55] = (x13 >>> 24) & 0xff;
+            this.block[56] = (x14 >>>  0) & 0xff; this.block[57] = (x14 >>>  8) & 0xff;
+            this.block[58] = (x14 >>> 16) & 0xff; this.block[59] = (x14 >>> 24) & 0xff;
+            this.block[60] = (x15 >>>  0) & 0xff; this.block[61] = (x15 >>>  8) & 0xff;
+            this.block[62] = (x15 >>> 16) & 0xff; this.block[63] = (x15 >>> 24) & 0xff;
+    };
+
+  return Salsa20
+
+}))
+/*!
+ * Source: build/lib/otr/build/dep/bigint.js, license: public domain, url: www.leemon.com */
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define(factory.bind(root, root.crypto || root.msCrypto))
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory(require('crypto'))
+  } else {
+    root.BigInt = factory(root.crypto || root.msCrypto)
+  }
+
+}(this, function (crypto) {
+
+  ////////////////////////////////////////////////////////////////////////////////////////
+  // Big Integer Library v. 5.5
+  // Created 2000, last modified 2013
+  // Leemon Baird
+  // www.leemon.com
+  //
+  // Version history:
+  // v 5.5  17 Mar 2013
+  //   - two lines of a form like "if (x<0) x+=n" had the "if" changed to "while" to
+  //     handle the case when x<-n. (Thanks to James Ansell for finding that bug)
+  // v 5.4  3 Oct 2009
+  //   - added "var i" to greaterShift() so i is not global. (Thanks to Péter Szabó for finding that bug)
+  //
+  // v 5.3  21 Sep 2009
+  //   - added randProbPrime(k) for probable primes
+  //   - unrolled loop in mont_ (slightly faster)
+  //   - millerRabin now takes a bigInt parameter rather than an int
+  //
+  // v 5.2  15 Sep 2009
+  //   - fixed capitalization in call to int2bigInt in randBigInt
+  //     (thanks to Emili Evripidou, Reinhold Behringer, and Samuel Macaleese for finding that bug)
+  //
+  // v 5.1  8 Oct 2007 
+  //   - renamed inverseModInt_ to inverseModInt since it doesn't change its parameters
+  //   - added functions GCD and randBigInt, which call GCD_ and randBigInt_
+  //   - fixed a bug found by Rob Visser (see comment with his name below)
+  //   - improved comments
+  //
+  // This file is public domain.   You can use it for any purpose without restriction.
+  // I do not guarantee that it is correct, so use it at your own risk.  If you use 
+  // it for something interesting, I'd appreciate hearing about it.  If you find 
+  // any bugs or make any improvements, I'd appreciate hearing about those too.
+  // It would also be nice if my name and URL were left in the comments.  But none 
+  // of that is required.
+  //
+  // This code defines a bigInt library for arbitrary-precision integers.
+  // A bigInt is an array of integers storing the value in chunks of bpe bits, 
+  // little endian (buff[0] is the least significant word).
+  // Negative bigInts are stored two's complement.  Almost all the functions treat
+  // bigInts as nonnegative.  The few that view them as two's complement say so
+  // in their comments.  Some functions assume their parameters have at least one 
+  // leading zero element. Functions with an underscore at the end of the name put
+  // their answer into one of the arrays passed in, and have unpredictable behavior 
+  // in case of overflow, so the caller must make sure the arrays are big enough to 
+  // hold the answer.  But the average user should never have to call any of the 
+  // underscored functions.  Each important underscored function has a wrapper function 
+  // of the same name without the underscore that takes care of the details for you.  
+  // For each underscored function where a parameter is modified, that same variable 
+  // must not be used as another argument too.  So, you cannot square x by doing 
+  // multMod_(x,x,n).  You must use squareMod_(x,n) instead, or do y=dup(x); multMod_(x,y,n).
+  // Or simply use the multMod(x,x,n) function without the underscore, where
+  // such issues never arise, because non-underscored functions never change
+  // their parameters; they always allocate new memory for the answer that is returned.
+  //
+  // These functions are designed to avoid frequent dynamic memory allocation in the inner loop.
+  // For most functions, if it needs a BigInt as a local variable it will actually use
+  // a global, and will only allocate to it only when it's not the right size.  This ensures
+  // that when a function is called repeatedly with same-sized parameters, it only allocates
+  // memory on the first call.
+  //
+  // Note that for cryptographic purposes, the calls to Math.random() must 
+  // be replaced with calls to a better pseudorandom number generator.
+  //
+  // In the following, "bigInt" means a bigInt with at least one leading zero element,
+  // and "integer" means a nonnegative integer less than radix.  In some cases, integer 
+  // can be negative.  Negative bigInts are 2s complement.
+  // 
+  // The following functions do not modify their inputs.
+  // Those returning a bigInt, string, or Array will dynamically allocate memory for that value.
+  // Those returning a boolean will return the integer 0 (false) or 1 (true).
+  // Those returning boolean or int will not allocate memory except possibly on the first 
+  // time they're called with a given parameter size.
+  // 
+  // bigInt  add(x,y)               //return (x+y) for bigInts x and y.  
+  // bigInt  addInt(x,n)            //return (x+n) where x is a bigInt and n is an integer.
+  // string  bigInt2str(x,base)     //return a string form of bigInt x in a given base, with 2 <= base <= 95
+  // int     bitSize(x)             //return how many bits long the bigInt x is, not counting leading zeros
+  // bigInt  dup(x)                 //return a copy of bigInt x
+  // boolean equals(x,y)            //is the bigInt x equal to the bigint y?
+  // boolean equalsInt(x,y)         //is bigint x equal to integer y?
+  // bigInt  expand(x,n)            //return a copy of x with at least n elements, adding leading zeros if needed
+  // Array   findPrimes(n)          //return array of all primes less than integer n
+  // bigInt  GCD(x,y)               //return greatest common divisor of bigInts x and y (each with same number of elements).
+  // boolean greater(x,y)           //is x>y?  (x and y are nonnegative bigInts)
+  // boolean greaterShift(x,y,shift)//is (x <<(shift*bpe)) > y?
+  // bigInt  int2bigInt(t,n,m)      //return a bigInt equal to integer t, with at least n bits and m array elements
+  // bigInt  inverseMod(x,n)        //return (x**(-1) mod n) for bigInts x and n.  If no inverse exists, it returns null
+  // int     inverseModInt(x,n)     //return x**(-1) mod n, for integers x and n.  Return 0 if there is no inverse
+  // boolean isZero(x)              //is the bigInt x equal to zero?
+  // boolean millerRabin(x,b)       //does one round of Miller-Rabin base integer b say that bigInt x is possibly prime? (b is bigInt, 1<b<x)
+  // boolean millerRabinInt(x,b)    //does one round of Miller-Rabin base integer b say that bigInt x is possibly prime? (b is int,    1<b<x)
+  // bigInt  mod(x,n)               //return a new bigInt equal to (x mod n) for bigInts x and n.
+  // int     modInt(x,n)            //return x mod n for bigInt x and integer n.
+  // bigInt  mult(x,y)              //return x*y for bigInts x and y. This is faster when y<x.
+  // bigInt  multMod(x,y,n)         //return (x*y mod n) for bigInts x,y,n.  For greater speed, let y<x.
+  // boolean negative(x)            //is bigInt x negative?
+  // bigInt  powMod(x,y,n)          //return (x**y mod n) where x,y,n are bigInts and ** is exponentiation.  0**0=1. Faster for odd n.
+  // bigInt  randBigInt(n,s)        //return an n-bit random BigInt (n>=1).  If s=1, then the most significant of those n bits is set to 1.
+  // bigInt  randTruePrime(k)       //return a new, random, k-bit, true prime bigInt using Maurer's algorithm.
+  // bigInt  randProbPrime(k)       //return a new, random, k-bit, probable prime bigInt (probability it's composite less than 2^-80).
+  // bigInt  str2bigInt(s,b,n,m)    //return a bigInt for number represented in string s in base b with at least n bits and m array elements
+  // bigInt  sub(x,y)               //return (x-y) for bigInts x and y.  Negative answers will be 2s complement
+  // bigInt  trim(x,k)              //return a copy of x with exactly k leading zero elements
+  //
+  //
+  // The following functions each have a non-underscored version, which most users should call instead.
+  // These functions each write to a single parameter, and the caller is responsible for ensuring the array 
+  // passed in is large enough to hold the result. 
+  //
+  // void    addInt_(x,n)          //do x=x+n where x is a bigInt and n is an integer
+  // void    add_(x,y)             //do x=x+y for bigInts x and y
+  // void    copy_(x,y)            //do x=y on bigInts x and y
+  // void    copyInt_(x,n)         //do x=n on bigInt x and integer n
+  // void    GCD_(x,y)             //set x to the greatest common divisor of bigInts x and y, (y is destroyed).  (This never overflows its array).
+  // boolean inverseMod_(x,n)      //do x=x**(-1) mod n, for bigInts x and n. Returns 1 (0) if inverse does (doesn't) exist
+  // void    mod_(x,n)             //do x=x mod n for bigInts x and n. (This never overflows its array).
+  // void    mult_(x,y)            //do x=x*y for bigInts x and y.
+  // void    multMod_(x,y,n)       //do x=x*y  mod n for bigInts x,y,n.
+  // void    powMod_(x,y,n)        //do x=x**y mod n, where x,y,n are bigInts (n is odd) and ** is exponentiation.  0**0=1.
+  // void    randBigInt_(b,n,s)    //do b = an n-bit random BigInt. if s=1, then nth bit (most significant bit) is set to 1. n>=1.
+  // void    randTruePrime_(ans,k) //do ans = a random k-bit true random prime (not just probable prime) with 1 in the msb.
+  // void    sub_(x,y)             //do x=x-y for bigInts x and y. Negative answers will be 2s complement.
+  //
+  // The following functions do NOT have a non-underscored version. 
+  // They each write a bigInt result to one or more parameters.  The caller is responsible for
+  // ensuring the arrays passed in are large enough to hold the results. 
+  //
+  // void addShift_(x,y,ys)       //do x=x+(y<<(ys*bpe))
+  // void carry_(x)               //do carries and borrows so each element of the bigInt x fits in bpe bits.
+  // void divide_(x,y,q,r)        //divide x by y giving quotient q and remainder r
+  // int  divInt_(x,n)            //do x=floor(x/n) for bigInt x and integer n, and return the remainder. (This never overflows its array).
+  // int  eGCD_(x,y,d,a,b)        //sets a,b,d to positive bigInts such that d = GCD_(x,y) = a*x-b*y
+  // void halve_(x)               //do x=floor(|x|/2)*sgn(x) for bigInt x in 2's complement.  (This never overflows its array).
+  // void leftShift_(x,n)         //left shift bigInt x by n bits.  n<bpe.
+  // void linComb_(x,y,a,b)       //do x=a*x+b*y for bigInts x and y and integers a and b
+  // void linCombShift_(x,y,b,ys) //do x=x+b*(y<<(ys*bpe)) for bigInts x and y, and integers b and ys
+  // void mont_(x,y,n,np)         //Montgomery multiplication (see comments where the function is defined)
+  // void multInt_(x,n)           //do x=x*n where x is a bigInt and n is an integer.
+  // void rightShift_(x,n)        //right shift bigInt x by n bits. (This never overflows its array).
+  // void squareMod_(x,n)         //do x=x*x  mod n for bigInts x,n
+  // void subShift_(x,y,ys)       //do x=x-(y<<(ys*bpe)). Negative answers will be 2s complement.
+  //
+  // The following functions are based on algorithms from the _Handbook of Applied Cryptography_
+  //    powMod_()           = algorithm 14.94, Montgomery exponentiation
+  //    eGCD_,inverseMod_() = algorithm 14.61, Binary extended GCD_
+  //    GCD_()              = algorothm 14.57, Lehmer's algorithm
+  //    mont_()             = algorithm 14.36, Montgomery multiplication
+  //    divide_()           = algorithm 14.20  Multiple-precision division
+  //    squareMod_()        = algorithm 14.16  Multiple-precision squaring
+  //    randTruePrime_()    = algorithm  4.62, Maurer's algorithm
+  //    millerRabin()       = algorithm  4.24, Miller-Rabin algorithm
+  //
+  // Profiling shows:
+  //     randTruePrime_() spends:
+  //         10% of its time in calls to powMod_()
+  //         85% of its time in calls to millerRabin()
+  //     millerRabin() spends:
+  //         99% of its time in calls to powMod_()   (always with a base of 2)
+  //     powMod_() spends:
+  //         94% of its time in calls to mont_()  (almost always with x==y)
+  //
+  // This suggests there are several ways to speed up this library slightly:
+  //     - convert powMod_ to use a Montgomery form of k-ary window (or maybe a Montgomery form of sliding window)
+  //         -- this should especially focus on being fast when raising 2 to a power mod n
+  //     - convert randTruePrime_() to use a minimum r of 1/3 instead of 1/2 with the appropriate change to the test
+  //     - tune the parameters in randTruePrime_(), including c, m, and recLimit
+  //     - speed up the single loop in mont_() that takes 95% of the runtime, perhaps by reducing checking
+  //       within the loop when all the parameters are the same length.
+  //
+  // There are several ideas that look like they wouldn't help much at all:
+  //     - replacing trial division in randTruePrime_() with a sieve (that speeds up something taking almost no time anyway)
+  //     - increase bpe from 15 to 30 (that would help if we had a 32*32->64 multiplier, but not with JavaScript's 32*32->32)
+  //     - speeding up mont_(x,y,n,np) when x==y by doing a non-modular, non-Montgomery square
+  //       followed by a Montgomery reduction.  The intermediate answer will be twice as long as x, so that
+  //       method would be slower.  This is unfortunate because the code currently spends almost all of its time
+  //       doing mont_(x,x,...), both for randTruePrime_() and powMod_().  A faster method for Montgomery squaring
+  //       would have a large impact on the speed of randTruePrime_() and powMod_().  HAC has a couple of poorly-worded
+  //       sentences that seem to imply it's faster to do a non-modular square followed by a single
+  //       Montgomery reduction, but that's obviously wrong.
+  ////////////////////////////////////////////////////////////////////////////////////////
+
+  //globals
+
+  // The number of significant bits in the fraction of a JavaScript
+  // floating-point number is 52, independent of platform.
+  // See: https://github.com/arlolra/otr/issues/41
+
+  var bpe = 26;          // bits stored per array element
+  var radix = 1 << bpe;  // equals 2^bpe
+  var mask = radix - 1;  // AND this with an array element to chop it down to bpe bits
+
+  //the digits for converting to different bases
+  var digitsStr='0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz_=!@#$%^&*()[]{}|;:,.<>/?`~ \\\'\"+-';
+
+  var one=int2bigInt(1,1,1);     //constant used in powMod_()
+
+  //the following global variables are scratchpad memory to 
+  //reduce dynamic memory allocation in the inner loop
+  var t=new Array(0);
+  var ss=t;       //used in mult_()
+  var s0=t;       //used in multMod_(), squareMod_()
+  var s1=t;       //used in powMod_(), multMod_(), squareMod_()
+  var s2=t;       //used in powMod_(), multMod_()
+  var s3=t;       //used in powMod_()
+  var s4=t, s5=t; //used in mod_()
+  var s6=t;       //used in bigInt2str()
+  var s7=t;       //used in powMod_()
+  var T=t;        //used in GCD_()
+  var sa=t;       //used in mont_()
+  var mr_x1=t, mr_r=t, mr_a=t;                                      //used in millerRabin()
+  var eg_v=t, eg_u=t, eg_A=t, eg_B=t, eg_C=t, eg_D=t;               //used in eGCD_(), inverseMod_()
+  var md_q1=t, md_q2=t, md_q3=t, md_r=t, md_r1=t, md_r2=t, md_tt=t; //used in mod_()
+
+  var primes=t, pows=t, s_i=t, s_i2=t, s_R=t, s_rm=t, s_q=t, s_n1=t;
+  var s_a=t, s_r2=t, s_n=t, s_b=t, s_d=t, s_x1=t, s_x2=t, s_aa=t; //used in randTruePrime_()
+    
+  var rpprb=t; //used in randProbPrimeRounds() (which also uses "primes")
+
+  ////////////////////////////////////////////////////////////////////////////////////////
+
+
+  //return array of all primes less than integer n
+  function findPrimes(n) {
+    var i,s,p,ans;
+    s=new Array(n);
+    for (i=0;i<n;i++)
+      s[i]=0;
+    s[0]=2;
+    p=0;    //first p elements of s are primes, the rest are a sieve
+    for(;s[p]<n;) {                  //s[p] is the pth prime
+      for(i=s[p]*s[p]; i<n; i+=s[p]) //mark multiples of s[p]
+        s[i]=1;
+      p++;
+      s[p]=s[p-1]+1;
+      for(; s[p]<n && s[s[p]]; s[p]++); //find next prime (where s[p]==0)
+    }
+    ans=new Array(p);
+    for(i=0;i<p;i++)
+      ans[i]=s[i];
+    return ans;
+  }
+
+
+  //does a single round of Miller-Rabin base b consider x to be a possible prime?
+  //x is a bigInt, and b is an integer, with b<x
+  function millerRabinInt(x,b) {
+    if (mr_x1.length!=x.length) {
+      mr_x1=dup(x);
+      mr_r=dup(x);
+      mr_a=dup(x);
+    }
+
+    copyInt_(mr_a,b);
+    return millerRabin(x,mr_a);
+  }
+
+  //does a single round of Miller-Rabin base b consider x to be a possible prime?
+  //x and b are bigInts with b<x
+  function millerRabin(x,b) {
+    var i,j,k,s;
+
+    if (mr_x1.length!=x.length) {
+      mr_x1=dup(x);
+      mr_r=dup(x);
+      mr_a=dup(x);
+    }
+
+    copy_(mr_a,b);
+    copy_(mr_r,x);
+    copy_(mr_x1,x);
+
+    addInt_(mr_r,-1);
+    addInt_(mr_x1,-1);
+
+    //s=the highest power of two that divides mr_r
+
+    /*
+    k=0;
+    for (i=0;i<mr_r.length;i++)
+      for (j=1;j<mask;j<<=1)
+        if (x[i] & j) {
+          s=(k<mr_r.length+bpe ? k : 0); 
+           i=mr_r.length;
+           j=mask;
+        } else
+          k++;
+    */
+
+    /* http://www.javascripter.net/math/primes/millerrabinbug-bigint54.htm */
+    if (isZero(mr_r)) return 0;
+    for (k=0; mr_r[k]==0; k++);
+    for (i=1,j=2; mr_r[k]%j==0; j*=2,i++ );
+    s = k*bpe + i - 1;
+    /* end */
+
+    if (s)                
+      rightShift_(mr_r,s);
+
+    powMod_(mr_a,mr_r,x);
+
+    if (!equalsInt(mr_a,1) && !equals(mr_a,mr_x1)) {
+      j=1;
+      while (j<=s-1 && !equals(mr_a,mr_x1)) {
+        squareMod_(mr_a,x);
+        if (equalsInt(mr_a,1)) {
+          return 0;
+        }
+        j++;
+      }
+      if (!equals(mr_a,mr_x1)) {
+        return 0;
+      }
+    }
+    return 1;  
+  }
+
+  //returns how many bits long the bigInt is, not counting leading zeros.
+  function bitSize(x) {
+    var j,z,w;
+    for (j=x.length-1; (x[j]==0) && (j>0); j--);
+    for (z=0,w=x[j]; w; (w>>=1),z++);
+    z+=bpe*j;
+    return z;
+  }
+
+  //return a copy of x with at least n elements, adding leading zeros if needed
+  function expand(x,n) {
+    var ans=int2bigInt(0,(x.length>n ? x.length : n)*bpe,0);
+    copy_(ans,x);
+    return ans;
+  }
+
+  //return a k-bit true random prime using Maurer's algorithm.
+  function randTruePrime(k) {
+    var ans=int2bigInt(0,k,0);
+    randTruePrime_(ans,k);
+    return trim(ans,1);
+  }
+
+  //return a k-bit random probable prime with probability of error < 2^-80
+  function randProbPrime(k) {
+    if (k>=600) return randProbPrimeRounds(k,2); //numbers from HAC table 4.3
+    if (k>=550) return randProbPrimeRounds(k,4);
+    if (k>=500) return randProbPrimeRounds(k,5);
+    if (k>=400) return randProbPrimeRounds(k,6);
+    if (k>=350) return randProbPrimeRounds(k,7);
+    if (k>=300) return randProbPrimeRounds(k,9);
+    if (k>=250) return randProbPrimeRounds(k,12); //numbers from HAC table 4.4
+    if (k>=200) return randProbPrimeRounds(k,15);
+    if (k>=150) return randProbPrimeRounds(k,18);
+    if (k>=100) return randProbPrimeRounds(k,27);
+                return randProbPrimeRounds(k,40); //number from HAC remark 4.26 (only an estimate)
+  }
+
+  //return a k-bit probable random prime using n rounds of Miller Rabin (after trial division with small primes)
+  function randProbPrimeRounds(k,n) {
+    var ans, i, divisible, B; 
+    B=30000;  //B is largest prime to use in trial division
+    ans=int2bigInt(0,k,0);
+    
+    //optimization: try larger and smaller B to find the best limit.
+    
+    if (primes.length==0)
+      primes=findPrimes(30000);  //check for divisibility by primes <=30000
+
+    if (rpprb.length!=ans.length)
+      rpprb=dup(ans);
+
+    for (;;) { //keep trying random values for ans until one appears to be prime
+      //optimization: pick a random number times L=2*3*5*...*p, plus a 
+      //   random element of the list of all numbers in [0,L) not divisible by any prime up to p.
+      //   This can reduce the amount of random number generation.
+      
+      randBigInt_(ans,k,0); //ans = a random odd number to check
+      ans[0] |= 1; 
+      divisible=0;
+    
+      //check ans for divisibility by small primes up to B
+      for (i=0; (i<primes.length) && (primes[i]<=B); i++)
+        if (modInt(ans,primes[i])==0 && !equalsInt(ans,primes[i])) {
+          divisible=1;
+          break;
+        }      
+      
+      //optimization: change millerRabin so the base can be bigger than the number being checked, then eliminate the while here.
+      
+      //do n rounds of Miller Rabin, with random bases less than ans
+      for (i=0; i<n && !divisible; i++) {
+        randBigInt_(rpprb,k,0);
+        while(!greater(ans,rpprb)) //pick a random rpprb that's < ans
+          randBigInt_(rpprb,k,0);
+        if (!millerRabin(ans,rpprb))
+          divisible=1;
+      }
+      
+      if(!divisible)
+        return ans;
+    }  
+  }
+
+  //return a new bigInt equal to (x mod n) for bigInts x and n.
+  function mod(x,n) {
+    var ans=dup(x);
+    mod_(ans,n);
+    return trim(ans,1);
+  }
+
+  //return (x+n) where x is a bigInt and n is an integer.
+  function addInt(x,n) {
+    var ans=expand(x,x.length+1);
+    addInt_(ans,n);
+    return trim(ans,1);
+  }
+
+  //return x*y for bigInts x and y. This is faster when y<x.
+  function mult(x,y) {
+    var ans=expand(x,x.length+y.length);
+    mult_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x**y mod n) where x,y,n are bigInts and ** is exponentiation.  0**0=1. Faster for odd n.
+  function powMod(x,y,n) {
+    var ans=expand(x,n.length);  
+    powMod_(ans,trim(y,2),trim(n,2),0);  //this should work without the trim, but doesn't
+    return trim(ans,1);
+  }
+
+  //return (x-y) for bigInts x and y.  Negative answers will be 2s complement
+  function sub(x,y) {
+    var ans=expand(x,(x.length>y.length ? x.length+1 : y.length+1)); 
+    sub_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x+y) for bigInts x and y.  
+  function add(x,y) {
+    var ans=expand(x,(x.length>y.length ? x.length+1 : y.length+1)); 
+    add_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x**(-1) mod n) for bigInts x and n.  If no inverse exists, it returns null
+  function inverseMod(x,n) {
+    var ans=expand(x,n.length); 
+    var s;
+    s=inverseMod_(ans,n);
+    return s ? trim(ans,1) : null;
+  }
+
+  //return (x*y mod n) for bigInts x,y,n.  For greater speed, let y<x.
+  function multMod(x,y,n) {
+    var ans=expand(x,n.length);
+    multMod_(ans,y,n);
+    return trim(ans,1);
+  }
+
+  //generate a k-bit true random prime using Maurer's algorithm,
+  //and put it into ans.  The bigInt ans must be large enough to hold it.
+  function randTruePrime_(ans,k) {
+    var c,w,m,pm,dd,j,r,B,divisible,z,zz,recSize,recLimit;
+
+    if (primes.length==0)
+      primes=findPrimes(30000);  //check for divisibility by primes <=30000
+
+    if (pows.length==0) {
+      pows=new Array(512);
+      for (j=0;j<512;j++) {
+        pows[j]=Math.pow(2,j/511.0-1.0);
+      }
+    }
+
+    //c and m should be tuned for a particular machine and value of k, to maximize speed
+    c=0.1;  //c=0.1 in HAC
+    m=20;   //generate this k-bit number by first recursively generating a number that has between k/2 and k-m bits
+    recLimit=20; //stop recursion when k <=recLimit.  Must have recLimit >= 2
+
+    if (s_i2.length!=ans.length) {
+      s_i2=dup(ans);
+      s_R =dup(ans);
+      s_n1=dup(ans);
+      s_r2=dup(ans);
+      s_d =dup(ans);
+      s_x1=dup(ans);
+      s_x2=dup(ans);
+      s_b =dup(ans);
+      s_n =dup(ans);
+      s_i =dup(ans);
+      s_rm=dup(ans);
+      s_q =dup(ans);
+      s_a =dup(ans);
+      s_aa=dup(ans);
+    }
+
+    if (k <= recLimit) {  //generate small random primes by trial division up to its square root
+      pm=(1<<((k+2)>>1))-1; //pm is binary number with all ones, just over sqrt(2^k)
+      copyInt_(ans,0);
+      for (dd=1;dd;) {
+        dd=0;
+        ans[0]= 1 | (1<<(k-1)) | randomBitInt(k);  //random, k-bit, odd integer, with msb 1
+        for (j=1;(j<primes.length) && ((primes[j]&pm)==primes[j]);j++) { //trial division by all primes 3...sqrt(2^k)
+          if (0==(ans[0]%primes[j])) {
+            dd=1;
+            break;
+          }
+        }
+      }
+      carry_(ans);
+      return;
+    }
+
+    B=c*k*k;    //try small primes up to B (or all the primes[] array if the largest is less than B).
+    if (k>2*m)  //generate this k-bit number by first recursively generating a number that has between k/2 and k-m bits
+      for (r=1; k-k*r<=m; )
+        r=pows[randomBitInt(9)];   //r=Math.pow(2,Math.random()-1);
+    else
+      r=0.5;
+
+    //simulation suggests the more complex algorithm using r=.333 is only slightly faster.
+
+    recSize=Math.floor(r*k)+1;
+
+    randTruePrime_(s_q,recSize);
+    copyInt_(s_i2,0);
+    s_i2[Math.floor((k-2)/bpe)] |= (1<<((k-2)%bpe));   //s_i2=2^(k-2)
+    divide_(s_i2,s_q,s_i,s_rm);                        //s_i=floor((2^(k-1))/(2q))
+
+    z=bitSize(s_i);
+
+    for (;;) {
+      for (;;) {  //generate z-bit numbers until one falls in the range [0,s_i-1]
+        randBigInt_(s_R,z,0);
+        if (greater(s_i,s_R))
+          break;
+      }                //now s_R is in the range [0,s_i-1]
+      addInt_(s_R,1);  //now s_R is in the range [1,s_i]
+      add_(s_R,s_i);   //now s_R is in the range [s_i+1,2*s_i]
+
+      copy_(s_n,s_q);
+      mult_(s_n,s_R); 
+      multInt_(s_n,2);
+      addInt_(s_n,1);    //s_n=2*s_R*s_q+1
+      
+      copy_(s_r2,s_R);
+      multInt_(s_r2,2);  //s_r2=2*s_R
+
+      //check s_n for divisibility by small primes up to B
+      for (divisible=0,j=0; (j<primes.length) && (primes[j]<B); j++)
+        if (modInt(s_n,primes[j])==0 && !equalsInt(s_n,primes[j])) {
+          divisible=1;
+          break;
+        }      
+
+      if (!divisible)    //if it passes small primes check, then try a single Miller-Rabin base 2
+        if (!millerRabinInt(s_n,2)) //this line represents 75% of the total runtime for randTruePrime_ 
+          divisible=1;
+
+      if (!divisible) {  //if it passes that test, continue checking s_n
+        addInt_(s_n,-3);
+        for (j=s_n.length-1;(s_n[j]==0) && (j>0); j--);  //strip leading zeros
+        for (zz=0,w=s_n[j]; w; (w>>=1),zz++);
+        zz+=bpe*j;                             //zz=number of bits in s_n, ignoring leading zeros
+        for (;;) {  //generate z-bit numbers until one falls in the range [0,s_n-1]
+          randBigInt_(s_a,zz,0);
+          if (greater(s_n,s_a))
+            break;
+        }                //now s_a is in the range [0,s_n-1]
+        addInt_(s_n,3);  //now s_a is in the range [0,s_n-4]
+        addInt_(s_a,2);  //now s_a is in the range [2,s_n-2]
+        copy_(s_b,s_a);
+        copy_(s_n1,s_n);
+        addInt_(s_n1,-1);
+        powMod_(s_b,s_n1,s_n);   //s_b=s_a^(s_n-1) modulo s_n
+        addInt_(s_b,-1);
+        if (isZero(s_b)) {
+          copy_(s_b,s_a);
+          powMod_(s_b,s_r2,s_n);
+          addInt_(s_b,-1);
+          copy_(s_aa,s_n);
+          copy_(s_d,s_b);
+          GCD_(s_d,s_n);  //if s_b and s_n are relatively prime, then s_n is a prime
+          if (equalsInt(s_d,1)) {
+            copy_(ans,s_aa);
+            return;     //if we've made it this far, then s_n is absolutely guaranteed to be prime
+          }
+        }
+      }
+    }
+  }
+
+  //Return an n-bit random BigInt (n>=1).  If s=1, then the most significant of those n bits is set to 1.
+  function randBigInt(n,s) {
+    var a,b;
+    a=Math.floor((n-1)/bpe)+2; //# array elements to hold the BigInt with a leading 0 element
+    b=int2bigInt(0,0,a);
+    randBigInt_(b,n,s);
+    return b;
+  }
+
+  //Set b to an n-bit random BigInt.  If s=1, then the most significant of those n bits is set to 1.
+  //Array b must be big enough to hold the result. Must have n>=1
+  function randBigInt_(b,n,s) {
+    var i,a;
+    for (i=0;i<b.length;i++)
+      b[i]=0;
+    a=Math.floor((n-1)/bpe)+1; //# array elements to hold the BigInt
+    for (i=0;i<a;i++) {
+      b[i]=randomBitInt(bpe);
+    }
+    b[a-1] &= (2<<((n-1)%bpe))-1;
+    if (s==1)
+      b[a-1] |= (1<<((n-1)%bpe));
+  }
+
+  //Return the greatest common divisor of bigInts x and y (each with same number of elements).
+  function GCD(x,y) {
+    var xc,yc;
+    xc=dup(x);
+    yc=dup(y);
+    GCD_(xc,yc);
+    return xc;
+  }
+
+  //set x to the greatest common divisor of bigInts x and y (each with same number of elements).
+  //y is destroyed.
+  function GCD_(x,y) {
+    var i,xp,yp,A,B,C,D,q,sing,qp;
+    if (T.length!=x.length)
+      T=dup(x);
+
+    sing=1;
+    while (sing) { //while y has nonzero elements other than y[0]
+      sing=0;
+      for (i=1;i<y.length;i++) //check if y has nonzero elements other than 0
+        if (y[i]) {
+          sing=1;
+          break;
+        }
+      if (!sing) break; //quit when y all zero elements except possibly y[0]
+
+      for (i=x.length;!x[i] && i>=0;i--);  //find most significant element of x
+      xp=x[i];
+      yp=y[i];
+      A=1; B=0; C=0; D=1;
+      while ((yp+C) && (yp+D)) {
+        q =Math.floor((xp+A)/(yp+C));
+        qp=Math.floor((xp+B)/(yp+D));
+        if (q!=qp)
+          break;
+        t= A-q*C;   A=C;   C=t;    //  do (A,B,xp, C,D,yp) = (C,D,yp, A,B,xp) - q*(0,0,0, C,D,yp)      
+        t= B-q*D;   B=D;   D=t;
+        t=xp-q*yp; xp=yp; yp=t;
+      }
+      if (B) {
+        copy_(T,x);
+        linComb_(x,y,A,B); //x=A*x+B*y
+        linComb_(y,T,D,C); //y=D*y+C*T
+      } else {
+        mod_(x,y);
+        copy_(T,x);
+        copy_(x,y);
+        copy_(y,T);
+      } 
+    }
+    if (y[0]==0)
+      return;
+    t=modInt(x,y[0]);
+    copyInt_(x,y[0]);
+    y[0]=t;
+    while (y[0]) {
+      x[0]%=y[0];
+      t=x[0]; x[0]=y[0]; y[0]=t;
+    }
+  }
+
+  //do x=x**(-1) mod n, for bigInts x and n.
+  //If no inverse exists, it sets x to zero and returns 0, else it returns 1.
+  //The x array must be at least as large as the n array.
+  function inverseMod_(x,n) {
+    var k=1+2*Math.max(x.length,n.length);
+
+    if(!(x[0]&1)  && !(n[0]&1)) {  //if both inputs are even, then inverse doesn't exist
+      copyInt_(x,0);
+      return 0;
+    }
+
+    if (eg_u.length!=k) {
+      eg_u=new Array(k);
+      eg_v=new Array(k);
+      eg_A=new Array(k);
+      eg_B=new Array(k);
+      eg_C=new Array(k);
+      eg_D=new Array(k);
+    }
+
+    copy_(eg_u,x);
+    copy_(eg_v,n);
+    copyInt_(eg_A,1);
+    copyInt_(eg_B,0);
+    copyInt_(eg_C,0);
+    copyInt_(eg_D,1);
+    for (;;) {
+      while(!(eg_u[0]&1)) {  //while eg_u is even
+        halve_(eg_u);
+        if (!(eg_A[0]&1) && !(eg_B[0]&1)) { //if eg_A==eg_B==0 mod 2
+          halve_(eg_A);
+          halve_(eg_B);      
+        } else {
+          add_(eg_A,n);  halve_(eg_A);
+          sub_(eg_B,x);  halve_(eg_B);
+        }
+      }
+
+      while (!(eg_v[0]&1)) {  //while eg_v is even
+        halve_(eg_v);
+        if (!(eg_C[0]&1) && !(eg_D[0]&1)) { //if eg_C==eg_D==0 mod 2
+          halve_(eg_C);
+          halve_(eg_D);      
+        } else {
+          add_(eg_C,n);  halve_(eg_C);
+          sub_(eg_D,x);  halve_(eg_D);
+        }
+      }
+
+      if (!greater(eg_v,eg_u)) { //eg_v <= eg_u
+        sub_(eg_u,eg_v);
+        sub_(eg_A,eg_C);
+        sub_(eg_B,eg_D);
+      } else {                   //eg_v > eg_u
+        sub_(eg_v,eg_u);
+        sub_(eg_C,eg_A);
+        sub_(eg_D,eg_B);
+      }
+
+      if (equalsInt(eg_u,0)) {
+        while (negative(eg_C)) //make sure answer is nonnegative
+          add_(eg_C,n);
+        copy_(x,eg_C);
+
+        if (!equalsInt(eg_v,1)) { //if GCD_(x,n)!=1, then there is no inverse
+          copyInt_(x,0);
+          return 0;
+        }
+        return 1;
+      }
+    }
+  }
+
+  //return x**(-1) mod n, for integers x and n.  Return 0 if there is no inverse
+  function inverseModInt(x,n) {
+    var a=1,b=0,t;
+    for (;;) {
+      if (x==1) return a;
+      if (x==0) return 0;
+      b-=a*Math.floor(n/x);
+      n%=x;
+
+      if (n==1) return b; //to avoid negatives, change this b to n-b, and each -= to +=
+      if (n==0) return 0;
+      a-=b*Math.floor(x/n);
+      x%=n;
+    }
+  }
+
+  //this deprecated function is for backward compatibility only. 
+  function inverseModInt_(x,n) {
+     return inverseModInt(x,n);
+  }
+
+
+  //Given positive bigInts x and y, change the bigints v, a, and b to positive bigInts such that:
+  //     v = GCD_(x,y) = a*x-b*y
+  //The bigInts v, a, b, must have exactly as many elements as the larger of x and y.
+  function eGCD_(x,y,v,a,b) {
+    var g=0;
+    var k=Math.max(x.length,y.length);
+    if (eg_u.length!=k) {
+      eg_u=new Array(k);
+      eg_A=new Array(k);
+      eg_B=new Array(k);
+      eg_C=new Array(k);
+      eg_D=new Array(k);
+    }
+    while(!(x[0]&1)  && !(y[0]&1)) {  //while x and y both even
+      halve_(x);
+      halve_(y);
+      g++;
+    }
+    copy_(eg_u,x);
+    copy_(v,y);
+    copyInt_(eg_A,1);
+    copyInt_(eg_B,0);
+    copyInt_(eg_C,0);
+    copyInt_(eg_D,1);
+    for (;;) {
+      while(!(eg_u[0]&1)) {  //while u is even
+        halve_(eg_u);
+        if (!(eg_A[0]&1) && !(eg_B[0]&1)) { //if A==B==0 mod 2
+          halve_(eg_A);
+          halve_(eg_B);      
+        } else {
+          add_(eg_A,y);  halve_(eg_A);
+          sub_(eg_B,x);  halve_(eg_B);
+        }
+      }
+
+      while (!(v[0]&1)) {  //while v is even
+        halve_(v);
+        if (!(eg_C[0]&1) && !(eg_D[0]&1)) { //if C==D==0 mod 2
+          halve_(eg_C);
+          halve_(eg_D);      
+        } else {
+          add_(eg_C,y);  halve_(eg_C);
+          sub_(eg_D,x);  halve_(eg_D);
+        }
+      }
+
+      if (!greater(v,eg_u)) { //v<=u
+        sub_(eg_u,v);
+        sub_(eg_A,eg_C);
+        sub_(eg_B,eg_D);
+      } else {                //v>u
+        sub_(v,eg_u);
+        sub_(eg_C,eg_A);
+        sub_(eg_D,eg_B);
+      }
+      if (equalsInt(eg_u,0)) {
+        while (negative(eg_C)) {   //make sure a (C) is nonnegative
+          add_(eg_C,y);
+          sub_(eg_D,x);
+        }
+        multInt_(eg_D,-1);  ///make sure b (D) is nonnegative
+        copy_(a,eg_C);
+        copy_(b,eg_D);
+        leftShift_(v,g);
+        return;
+      }
+    }
+  }
+
+
+  //is bigInt x negative?
+  function negative(x) {
+    return ((x[x.length-1]>>(bpe-1))&1);
+  }
+
+
+  //is (x << (shift*bpe)) > y?
+  //x and y are nonnegative bigInts
+  //shift is a nonnegative integer
+  function greaterShift(x,y,shift) {
+    var i, kx=x.length, ky=y.length;
+    var k=((kx+shift)<ky) ? (kx+shift) : ky;
+    for (i=ky-1-shift; i<kx && i>=0; i++) 
+      if (x[i]>0)
+        return 1; //if there are nonzeros in x to the left of the first column of y, then x is bigger
+    for (i=kx-1+shift; i<ky; i++)
+      if (y[i]>0)
+        return 0; //if there are nonzeros in y to the left of the first column of x, then x is not bigger
+    for (i=k-1; i>=shift; i--)
+      if      (x[i-shift]>y[i]) return 1;
+      else if (x[i-shift]<y[i]) return 0;
+    return 0;
+  }
+
+  //is x > y? (x and y both nonnegative)
+  function greater(x,y) {
+    var i;
+    var k=(x.length<y.length) ? x.length : y.length;
+
+    for (i=x.length;i<y.length;i++)
+      if (y[i])
+        return 0;  //y has more digits
+
+    for (i=y.length;i<x.length;i++)
+      if (x[i])
+        return 1;  //x has more digits
+
+    for (i=k-1;i>=0;i--)
+      if (x[i]>y[i])
+        return 1;
+      else if (x[i]<y[i])
+        return 0;
+    return 0;
+  }
+
+  //divide x by y giving quotient q and remainder r.  (q=floor(x/y),  r=x mod y).  All 4 are bigints.
+  //x must have at least one leading zero element.
+  //y must be nonzero.
+  //q and r must be arrays that are exactly the same length as x. (Or q can have more).
+  //Must have x.length >= y.length >= 2.
+  function divide_(x,y,q,r) {
+    var kx, ky;
+    var i,j,y1,y2,c,a,b;
+    copy_(r,x);
+    for (ky=y.length;y[ky-1]==0;ky--); //ky is number of elements in y, not including leading zeros
+
+    //normalize: ensure the most significant element of y has its highest bit set  
+    b=y[ky-1];
+    for (a=0; b; a++)
+      b>>=1;  
+    a=bpe-a;  //a is how many bits to shift so that the high order bit of y is leftmost in its array element
+    leftShift_(y,a);  //multiply both by 1<<a now, then divide both by that at the end
+    leftShift_(r,a);
+
+    //Rob Visser discovered a bug: the following line was originally just before the normalization.
+    for (kx=r.length;r[kx-1]==0 && kx>ky;kx--); //kx is number of elements in normalized x, not including leading zeros
+
+    copyInt_(q,0);                      // q=0
+    while (!greaterShift(y,r,kx-ky)) {  // while (leftShift_(y,kx-ky) <= r) {
+      subShift_(r,y,kx-ky);             //   r=r-leftShift_(y,kx-ky)
+      q[kx-ky]++;                       //   q[kx-ky]++;
+    }                                   // }
+
+    for (i=kx-1; i>=ky; i--) {
+      if (r[i]==y[ky-1])
+        q[i-ky]=mask;
+      else
+        q[i-ky]=Math.floor((r[i]*radix+r[i-1])/y[ky-1]);
+
+      //The following for(;;) loop is equivalent to the commented while loop, 
+      //except that the uncommented version avoids overflow.
+      //The commented loop comes from HAC, which assumes r[-1]==y[-1]==0
+      //  while (q[i-ky]*(y[ky-1]*radix+y[ky-2]) > r[i]*radix*radix+r[i-1]*radix+r[i-2])
+      //    q[i-ky]--;    
+      for (;;) {
+        y2=(ky>1 ? y[ky-2] : 0)*q[i-ky];
+        c=y2;
+        y2=y2 & mask;
+        c = (c - y2) / radix;
+        y1=c+q[i-ky]*y[ky-1];
+        c=y1;
+        y1=y1 & mask;
+        c = (c - y1) / radix;
+
+        if (c==r[i] ? y1==r[i-1] ? y2>(i>1 ? r[i-2] : 0) : y1>r[i-1] : c>r[i]) 
+          q[i-ky]--;
+        else
+          break;
+      }
+
+      linCombShift_(r,y,-q[i-ky],i-ky);    //r=r-q[i-ky]*leftShift_(y,i-ky)
+      if (negative(r)) {
+        addShift_(r,y,i-ky);         //r=r+leftShift_(y,i-ky)
+        q[i-ky]--;
+      }
+    }
+
+    rightShift_(y,a);  //undo the normalization step
+    rightShift_(r,a);  //undo the normalization step
+  }
+
+  //do carries and borrows so each element of the bigInt x fits in bpe bits.
+  function carry_(x) {
+    var i,k,c,b;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i];
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+    }
+  }
+
+  //return x mod n for bigInt x and integer n.
+  function modInt(x,n) {
+    var i,c=0;
+    for (i=x.length-1; i>=0; i--)
+      c=(c*radix+x[i])%n;
+    return c;
+  }
+
+  //convert the integer t into a bigInt with at least the given number of bits.
+  //the returned array stores the bigInt in bpe-bit chunks, little endian (buff[0] is least significant word)
+  //Pad the array with leading zeros so that it has at least minSize elements.
+  //There will always be at least one leading 0 element.
+  function int2bigInt(t,bits,minSize) {   
+    var i,k, buff;
+    k=Math.ceil(bits/bpe)+1;
+    k=minSize>k ? minSize : k;
+    buff=new Array(k);
+    copyInt_(buff,t);
+    return buff;
+  }
+
+  //return the bigInt given a string representation in a given base.  
+  //Pad the array with leading zeros so that it has at least minSize elements.
+  //If base=-1, then it reads in a space-separated list of array elements in decimal.
+  //The array will always have at least one leading zero, unless base=-1.
+  function str2bigInt(s,base,minSize) {
+    var d, i, j, x, y, kk;
+    var k=s.length;
+    if (base==-1) { //comma-separated list of array elements in decimal
+      x=new Array(0);
+      for (;;) {
+        y=new Array(x.length+1);
+        for (i=0;i<x.length;i++)
+          y[i+1]=x[i];
+        y[0]=parseInt(s,10);
+        x=y;
+        d=s.indexOf(',',0);
+        if (d<1) 
+          break;
+        s=s.substring(d+1);
+        if (s.length==0)
+          break;
+      }
+      if (x.length<minSize) {
+        y=new Array(minSize);
+        copy_(y,x);
+        return y;
+      }
+      return x;
+    }
+
+    // log2(base)*k
+    var bb = base, p = 0;
+    var b = base == 1 ? k : 0;
+    while (bb > 1) {
+      if (bb & 1) p = 1;
+      b += k;
+      bb >>= 1;
+    }
+    b += p*k;
+
+    x=int2bigInt(0,b,0);
+    for (i=0;i<k;i++) {
+      d=digitsStr.indexOf(s.substring(i,i+1),0);
+      if (base<=36 && d>=36)  //convert lowercase to uppercase if base<=36
+        d-=26;
+      if (d>=base || d<0) {   //stop at first illegal character
+        break;
+      }
+      multInt_(x,base);
+      addInt_(x,d);
+    }
+
+    for (k=x.length;k>0 && !x[k-1];k--); //strip off leading zeros
+    k=minSize>k+1 ? minSize : k+1;
+    y=new Array(k);
+    kk=k<x.length ? k : x.length;
+    for (i=0;i<kk;i++)
+      y[i]=x[i];
+    for (;i<k;i++)
+      y[i]=0;
+    return y;
+  }
+
+  //is bigint x equal to integer y?
+  //y must have less than bpe bits
+  function equalsInt(x,y) {
+    var i;
+    if (x[0]!=y)
+      return 0;
+    for (i=1;i<x.length;i++)
+      if (x[i])
+        return 0;
+    return 1;
+  }
+
+  //are bigints x and y equal?
+  //this works even if x and y are different lengths and have arbitrarily many leading zeros
+  function equals(x,y) {
+    var i;
+    var k=x.length<y.length ? x.length : y.length;
+    for (i=0;i<k;i++)
+      if (x[i]!=y[i])
+        return 0;
+    if (x.length>y.length) {
+      for (;i<x.length;i++)
+        if (x[i])
+          return 0;
+    } else {
+      for (;i<y.length;i++)
+        if (y[i])
+          return 0;
+    }
+    return 1;
+  }
+
+  //is the bigInt x equal to zero?
+  function isZero(x) {
+    var i;
+    for (i=0;i<x.length;i++)
+      if (x[i])
+        return 0;
+    return 1;
+  }
+
+  //convert a bigInt into a string in a given base, from base 2 up to base 95.
+  //Base -1 prints the contents of the array representing the number.
+  function bigInt2str(x,base) {
+    var i,t,s="";
+
+    if (s6.length!=x.length) 
+      s6=dup(x);
+    else
+      copy_(s6,x);
+
+    if (base==-1) { //return the list of array contents
+      for (i=x.length-1;i>0;i--)
+        s+=x[i]+',';
+      s+=x[0];
+    }
+    else { //return it in the given base
+      while (!isZero(s6)) {
+        t=divInt_(s6,base);  //t=s6 % base; s6=floor(s6/base);
+        s=digitsStr.substring(t,t+1)+s;
+      }
+    }
+    if (s.length==0)
+      s="0";
+    return s;
+  }
+
+  //returns a duplicate of bigInt x
+  function dup(x) {
+    var i, buff;
+    buff=new Array(x.length);
+    copy_(buff,x);
+    return buff;
+  }
+
+  //do x=y on bigInts x and y.  x must be an array at least as big as y (not counting the leading zeros in y).
+  function copy_(x,y) {
+    var i;
+    var k=x.length<y.length ? x.length : y.length;
+    for (i=0;i<k;i++)
+      x[i]=y[i];
+    for (i=k;i<x.length;i++)
+      x[i]=0;
+  }
+
+  //do x=y on bigInt x and integer y.  
+  function copyInt_(x,n) {
+    var i,c;
+    for (c=n,i=0;i<x.length;i++) {
+      x[i]=c & mask;
+      c>>=bpe;
+    }
+  }
+
+  //do x=x+n where x is a bigInt and n is an integer.
+  //x must be large enough to hold the result.
+  function addInt_(x,n) {
+    var i,k,c,b;
+    x[0]+=n;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i];
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+      if (!c) return; //stop carrying as soon as the carry is zero
+    }
+  }
+
+  //right shift bigInt x by n bits.
+  function rightShift_(x,n) {
+    var i;
+    var k=Math.floor(n/bpe);
+    if (k) {
+      for (i=0;i<x.length-k;i++) //right shift x by k elements
+        x[i]=x[i+k];
+      for (;i<x.length;i++)
+        x[i]=0;
+      n%=bpe;
+    }
+    for (i=0;i<x.length-1;i++) {
+      x[i]=mask & ((x[i+1]<<(bpe-n)) | (x[i]>>n));
+    }
+    x[i]>>=n;
+  }
+
+  //do x=floor(|x|/2)*sgn(x) for bigInt x in 2's complement
+  function halve_(x) {
+    var i;
+    for (i=0;i<x.length-1;i++) {
+      x[i]=mask & ((x[i+1]<<(bpe-1)) | (x[i]>>1));
+    }
+    x[i]=(x[i]>>1) | (x[i] & (radix>>1));  //most significant bit stays the same
+  }
+
+  //left shift bigInt x by n bits.
+  function leftShift_(x,n) {
+    var i;
+    var k=Math.floor(n/bpe);
+    if (k) {
+      for (i=x.length; i>=k; i--) //left shift x by k elements
+        x[i]=x[i-k];
+      for (;i>=0;i--)
+        x[i]=0;  
+      n%=bpe;
+    }
+    if (!n)
+      return;
+    for (i=x.length-1;i>0;i--) {
+      x[i]=mask & ((x[i]<<n) | (x[i-1]>>(bpe-n)));
+    }
+    x[i]=mask & (x[i]<<n);
+  }
+
+  //do x=x*n where x is a bigInt and n is an integer.
+  //x must be large enough to hold the result.
+  function multInt_(x,n) {
+    var i,k,c,b;
+    if (!n)
+      return;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i]*n;
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+    }
+  }
+
+  //do x=floor(x/n) for bigInt x and integer n, and return the remainder
+  function divInt_(x,n) {
+    var i,r=0,s;
+    for (i=x.length-1;i>=0;i--) {
+      s=r*radix+x[i];
+      x[i]=Math.floor(s/n);
+      r=s%n;
+    }
+    return r;
+  }
+
+  //do the linear combination x=a*x+b*y for bigInts x and y, and integers a and b.
+  //x must be large enough to hold the answer.
+  function linComb_(x,y,a,b) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    kk=x.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=a*x[i]+b*y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;i<kk;i++) {
+      c+=a*x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do the linear combination x=a*x+b*(y<<(ys*bpe)) for bigInts x and y, and integers a, b and ys.
+  //x must be large enough to hold the answer.
+  function linCombShift_(x,y,b,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]+b*y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x+(y<<(ys*bpe)) for bigInts x and y, and integers a,b and ys.
+  //x must be large enough to hold the answer.
+  function addShift_(x,y,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]+y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x-(y<<(ys*bpe)) for bigInts x and y, and integers a,b and ys.
+  //x must be large enough to hold the answer.
+  function subShift_(x,y,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]-y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x-y for bigInts x and y.
+  //x must be large enough to hold the answer.
+  //negative answers will be 2s complement
+  function sub_(x,y) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=x[i]-y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<x.length;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x+y for bigInts x and y.
+  //x must be large enough to hold the answer.
+  function add_(x,y) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=x[i]+y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<x.length;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x*y for bigInts x and y.  This is faster when y<x.
+  function mult_(x,y) {
+    var i;
+    if (ss.length!=2*x.length)
+      ss=new Array(2*x.length);
+    copyInt_(ss,0);
+    for (i=0;i<y.length;i++)
+      if (y[i])
+        linCombShift_(ss,x,y[i],i);   //ss=1*ss+y[i]*(x<<(i*bpe))
+    copy_(x,ss);
+  }
+
+  //do x=x mod n for bigInts x and n.
+  function mod_(x,n) {
+    if (s4.length!=x.length)
+      s4=dup(x);
+    else
+      copy_(s4,x);
+    if (s5.length!=x.length)
+      s5=dup(x);  
+    divide_(s4,n,s5,x);  //x = remainder of s4 / n
+  }
+
+  //do x=x*y mod n for bigInts x,y,n.
+  //for greater speed, let y<x.
+  function multMod_(x,y,n) {
+    var i;
+    if (s0.length!=2*x.length)
+      s0=new Array(2*x.length);
+    copyInt_(s0,0);
+    for (i=0;i<y.length;i++)
+      if (y[i])
+        linCombShift_(s0,x,y[i],i);   //s0=1*s0+y[i]*(x<<(i*bpe))
+    mod_(s0,n);
+    copy_(x,s0);
+  }
+
+  //do x=x*x mod n for bigInts x,n.
+  function squareMod_(x,n) {
+    var i,j,d,c,kx,kn,k;
+    for (kx=x.length; kx>0 && !x[kx-1]; kx--);  //ignore leading zeros in x
+    k=kx>n.length ? 2*kx : 2*n.length; //k=# elements in the product, which is twice the elements in the larger of x and n
+    if (s0.length!=k) 
+      s0=new Array(k);
+    copyInt_(s0,0);
+    for (i=0;i<kx;i++) {
+      c=s0[2*i]+x[i]*x[i];
+      s0[2*i]=c & mask;
+      c = (c - s0[2*i]) / radix;
+      for (j=i+1;j<kx;j++) {
+        c=s0[i+j]+2*x[i]*x[j]+c;
+        s0[i+j]=(c & mask);
+        c = (c - s0[i+j]) / radix;
+      }
+      s0[i+kx]=c;
+    }
+    mod_(s0,n);
+    copy_(x,s0);
+  }
+
+  //return x with exactly k leading zero elements
+  function trim(x,k) {
+    var i,y;
+    for (i=x.length; i>0 && !x[i-1]; i--);
+    y=new Array(i+k);
+    copy_(y,x);
+    return y;
+  }
+
+  //do x=x**y mod n, where x,y,n are bigInts and ** is exponentiation.  0**0=1.
+  //this is faster when n is odd.  x usually needs to have as many elements as n.
+  function powMod_(x,y,n) {
+    var k1,k2,kn,np;
+    if(s7.length!=n.length)
+      s7=dup(n);
+
+    //for even modulus, use a simple square-and-multiply algorithm,
+    //rather than using the more complex Montgomery algorithm.
+    if ((n[0]&1)==0) {
+      copy_(s7,x);
+      copyInt_(x,1);
+      while(!equalsInt(y,0)) {
+        if (y[0]&1)
+          multMod_(x,s7,n);
+        divInt_(y,2);
+        squareMod_(s7,n); 
+      }
+      return;
+    }
+
+    //calculate np from n for the Montgomery multiplications
+    copyInt_(s7,0);
+    for (kn=n.length;kn>0 && !n[kn-1];kn--);
+    np=radix-inverseModInt(modInt(n,radix),radix);
+    s7[kn]=1;
+    multMod_(x ,s7,n);   // x = x * 2**(kn*bp) mod n
+
+    if (s3.length!=x.length)
+      s3=dup(x);
+    else
+      copy_(s3,x);
+
+    for (k1=y.length-1;k1>0 & !y[k1]; k1--);  //k1=first nonzero element of y
+    if (y[k1]==0) {  //anything to the 0th power is 1
+      copyInt_(x,1);
+      return;
+    }
+    for (k2=1<<(bpe-1);k2 && !(y[k1] & k2); k2>>=1);  //k2=position of first 1 bit in y[k1]
+    for (;;) {
+      if (!(k2>>=1)) {  //look at next bit of y
+        k1--;
+        if (k1<0) {
+          mont_(x,one,n,np);
+          return;
+        }
+        k2=1<<(bpe-1);
+      }    
+      mont_(x,x,n,np);
+
+      if (k2 & y[k1]) //if next bit is a 1
+        mont_(x,s3,n,np);
+    }
+  }
+
+
+  //do x=x*y*Ri mod n for bigInts x,y,n, 
+  //  where Ri = 2**(-kn*bpe) mod n, and kn is the 
+  //  number of elements in the n array, not 
+  //  counting leading zeros.  
+  //x array must have at least as many elemnts as the n array
+  //It's OK if x and y are the same variable.
+  //must have:
+  //  x,y < n
+  //  n is odd
+  //  np = -(n^(-1)) mod radix
+  function mont_(x,y,n,np) {
+    var i,j,c,ui,t,t2,ks;
+    var kn=n.length;
+    var ky=y.length;
+
+    if (sa.length!=kn)
+      sa=new Array(kn);
+      
+    copyInt_(sa,0);
+
+    for (;kn>0 && n[kn-1]==0;kn--); //ignore leading zeros of n
+    for (;ky>0 && y[ky-1]==0;ky--); //ignore leading zeros of y
+    ks=sa.length-1; //sa will never have more than this many nonzero elements.  
+
+    //the following loop consumes 95% of the runtime for randTruePrime_() and powMod_() for large numbers
+    for (i=0; i<kn; i++) {
+      t=sa[0]+x[i]*y[0];
+      ui=((t & mask) * np) & mask;  //the inner "& mask" was needed on Safari (but not MSIE) at one time
+      c=(t+ui*n[0]);
+      c = (c - (c & mask)) / radix;
+      t=x[i];
+      
+      //do sa=(sa+x[i]*y+ui*n)/b   where b=2**bpe.  Loop is unrolled 5-fold for speed
+      j=1;
+      for (;j<ky-4;) {
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<ky;)   {
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<kn-4;) {
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<kn;)   {
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<ks;)   {
+        c+=sa[j];                t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      sa[j-1]=c & mask;
+    }
+
+    if (!greater(n,sa))
+      sub_(sa,n);
+    copy_(x,sa);
+  }
+
+
+  // otr.js additions
+
+
+  // computes num / den mod n
+  function divMod(num, den, n) {
+    return multMod(num, inverseMod(den, n), n)
+  }
+
+  // computes one - two mod n
+  function subMod(one, two, n) {
+    one = mod(one, n)
+    two = mod(two, n)
+    if (greater(two, one)) one = add(one, n)
+    return sub(one, two)
+  }
+
+  // computes 2^m as a bigInt
+  function twoToThe(m) {
+    var b = Math.floor(m / bpe) + 2
+    var t = new Array(b)
+    for (var i = 0; i < b; i++) t[i] = 0
+    t[b - 2] = 1 << (m % bpe)
+    return t
+  }
+
+  // cache these results for faster lookup
+  var _num2bin = (function () {
+    var i = 0, _num2bin= {}
+    for (; i < 0x100; ++i) {
+      _num2bin[i] = String.fromCharCode(i)  // 0 -> "\00"
+    }
+    return _num2bin
+  }())
+
+  // serialize a bigInt to an ascii string
+  // padded up to pad length
+  function bigInt2bits(bi, pad) {
+    pad || (pad = 0)
+    bi = dup(bi)
+    var ba = ''
+    while (!isZero(bi)) {
+      ba = _num2bin[bi[0] & 0xff] + ba
+      rightShift_(bi, 8)
+    }
+    while (ba.length < pad) {
+      ba = '\x00' + ba
+    }
+    return ba
+  }
+
+  // converts a byte array to a bigInt
+  function ba2bigInt(data) {
+    var mpi = str2bigInt('0', 10, data.length)
+    data.forEach(function (d, i) {
+      if (i) leftShift_(mpi, 8)
+      mpi[0] |= d
+    })
+    return mpi
+  }
+
+  // returns a function that returns an array of n bytes
+  var randomBytes = (function () {
+
+    // in node
+    if ( typeof crypto !== 'undefined' &&
+      typeof crypto.randomBytes === 'function' ) {
+      return function (n) {
+        try {
+          var buf = crypto.randomBytes(n)
+        } catch (e) { throw e }
+        return Array.prototype.slice.call(buf, 0)
+      }
+    }
+
+    // in browser
+    else if ( typeof crypto !== 'undefined' &&
+      typeof crypto.getRandomValues === 'function' ) {
+      return function (n) {
+        var buf = new Uint8Array(n)
+        crypto.getRandomValues(buf)
+        return Array.prototype.slice.call(buf, 0)
+      }
+    }
+
+    // err
+    else {
+      throw new Error('Keys should not be generated without CSPRNG.')
+    }
+
+  }())
+
+  // Salsa 20 in webworker needs a 40 byte seed
+  function getSeed() {
+    return randomBytes(40)
+  }
+
+  // returns a single random byte
+  function randomByte() {
+    return randomBytes(1)[0]
+  }
+
+  // returns a k-bit random integer
+  function randomBitInt(k) {
+    if (k > 31) throw new Error("Too many bits.")
+    var i = 0, r = 0
+    var b = Math.floor(k / 8)
+    var mask = (1 << (k % 8)) - 1
+    if (mask) r = randomByte() & mask
+    for (; i < b; i++)
+      r = (256 * r) + randomByte()
+    return r
+  }
+
+  return {
+      str2bigInt    : str2bigInt
+    , bigInt2str    : bigInt2str
+    , int2bigInt    : int2bigInt
+    , multMod       : multMod
+    , powMod        : powMod
+    , inverseMod    : inverseMod
+    , randBigInt    : randBigInt
+    , randBigInt_   : randBigInt_
+    , equals        : equals
+    , equalsInt     : equalsInt
+    , sub           : sub
+    , mod           : mod
+    , modInt        : modInt
+    , mult          : mult
+    , divInt_       : divInt_
+    , rightShift_   : rightShift_
+    , dup           : dup
+    , greater       : greater
+    , add           : add
+    , isZero        : isZero
+    , bitSize       : bitSize
+    , millerRabin   : millerRabin
+    , divide_       : divide_
+    , trim          : trim
+    , primes        : primes
+    , findPrimes    : findPrimes
+    , getSeed       : getSeed
+    , divMod        : divMod
+    , subMod        : subMod
+    , twoToThe      : twoToThe
+    , bigInt2bits   : bigInt2bits
+    , ba2bigInt     : ba2bigInt
+  }
+
+}))
+/*!
+ * Source: build/lib/otr/build/dep/crypto.js, license: code.google.com/p/crypto-js/wiki/license, url: code.google.com/p/crypto-js */
+;(function (root, factory) {
+
+  if (typeof define === "function" && define.amd) {
+    define(factory)
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory()
+  } else {
+    root.CryptoJS = factory()
+  }
+
+}(this, function () {
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * CryptoJS core components.
+ */
+var CryptoJS = CryptoJS || (function (Math, undefined) {
+    /**
+     * CryptoJS namespace.
+     */
+    var C = {};
+
+    /**
+     * Library namespace.
+     */
+    var C_lib = C.lib = {};
+
+    /**
+     * Base object for prototypal inheritance.
+     */
+    var Base = C_lib.Base = (function () {
+        function F() {}
+
+        return {
+            /**
+             * Creates a new object that inherits from this object.
+             *
+             * @param {Object} overrides Properties to copy into the new object.
+             *
+             * @return {Object} The new object.
+             *
+             * @static
+             *
+             * @example
+             *
+             *     var MyType = CryptoJS.lib.Base.extend({
+             *         field: 'value',
+             *
+             *         method: function () {
+             *         }
+             *     });
+             */
+            extend: function (overrides) {
+                // Spawn
+                F.prototype = this;
+                var subtype = new F();
+
+                // Augment
+                if (overrides) {
+                    subtype.mixIn(overrides);
+                }
+
+                // Create default initializer
+                if (!subtype.hasOwnProperty('init')) {
+                    subtype.init = function () {
+                        subtype.$super.init.apply(this, arguments);
+                    };
+                }
+
+                // Initializer's prototype is the subtype object
+                subtype.init.prototype = subtype;
+
+                // Reference supertype
+                subtype.$super = this;
+
+                return subtype;
+            },
+
+            /**
+             * Extends this object and runs the init method.
+             * Arguments to create() will be passed to init().
+             *
+             * @return {Object} The new object.
+             *
+             * @static
+             *
+             * @example
+             *
+             *     var instance = MyType.create();
+             */
+            create: function () {
+                var instance = this.extend();
+                instance.init.apply(instance, arguments);
+
+                return instance;
+            },
+
+            /**
+             * Initializes a newly created object.
+             * Override this method to add some logic when your objects are created.
+             *
+             * @example
+             *
+             *     var MyType = CryptoJS.lib.Base.extend({
+             *         init: function () {
+             *             // ...
+             *         }
+             *     });
+             */
+            init: function () {
+            },
+
+            /**
+             * Copies properties into this object.
+             *
+             * @param {Object} properties The properties to mix in.
+             *
+             * @example
+             *
+             *     MyType.mixIn({
+             *         field: 'value'
+             *     });
+             */
+            mixIn: function (properties) {
+                for (var propertyName in properties) {
+                    if (properties.hasOwnProperty(propertyName)) {
+                        this[propertyName] = properties[propertyName];
+                    }
+                }
+
+                // IE won't copy toString using the loop above
+                if (properties.hasOwnProperty('toString')) {
+                    this.toString = properties.toString;
+                }
+            },
+
+            /**
+             * Creates a copy of this object.
+             *
+             * @return {Object} The clone.
+             *
+             * @example
+             *
+             *     var clone = instance.clone();
+             */
+            clone: function () {
+                return this.init.prototype.extend(this);
+            }
+        };
+    }());
+
+    /**
+     * An array of 32-bit words.
+     *
+     * @property {Array} words The array of 32-bit words.
+     * @property {number} sigBytes The number of significant bytes in this word array.
+     */
+    var WordArray = C_lib.WordArray = Base.extend({
+        /**
+         * Initializes a newly created word array.
+         *
+         * @param {Array} words (Optional) An array of 32-bit words.
+         * @param {number} sigBytes (Optional) The number of significant bytes in the words.
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.lib.WordArray.create();
+         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);
+         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);
+         */
+        init: function (words, sigBytes) {
+            words = this.words = words || [];
+
+            if (sigBytes != undefined) {
+                this.sigBytes = sigBytes;
+            } else {
+                this.sigBytes = words.length * 4;
+            }
+        },
+
+        /**
+         * Converts this word array to a string.
+         *
+         * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex
+         *
+         * @return {string} The stringified word array.
+         *
+         * @example
+         *
+         *     var string = wordArray + '';
+         *     var string = wordArray.toString();
+         *     var string = wordArray.toString(CryptoJS.enc.Utf8);
+         */
+        toString: function (encoder) {
+            return (encoder || Hex).stringify(this);
+        },
+
+        /**
+         * Concatenates a word array to this word array.
+         *
+         * @param {WordArray} wordArray The word array to append.
+         *
+         * @return {WordArray} This word array.
+         *
+         * @example
+         *
+         *     wordArray1.concat(wordArray2);
+         */
+        concat: function (wordArray) {
+            // Shortcuts
+            var thisWords = this.words;
+            var thatWords = wordArray.words;
+            var thisSigBytes = this.sigBytes;
+            var thatSigBytes = wordArray.sigBytes;
+
+            // Clamp excess bits
+            this.clamp();
+
+            // Concat
+            if (thisSigBytes % 4) {
+                // Copy one byte at a time
+                for (var i = 0; i < thatSigBytes; i++) {
+                    var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                    thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);
+                }
+            } else if (thatWords.length > 0xffff) {
+                // Copy one word at a time
+                for (var i = 0; i < thatSigBytes; i += 4) {
+                    thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];
+                }
+            } else {
+                // Copy all words at once
+                thisWords.push.apply(thisWords, thatWords);
+            }
+            this.sigBytes += thatSigBytes;
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Removes insignificant bits.
+         *
+         * @example
+         *
+         *     wordArray.clamp();
+         */
+        clamp: function () {
+            // Shortcuts
+            var words = this.words;
+            var sigBytes = this.sigBytes;
+
+            // Clamp
+            words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);
+            words.length = Math.ceil(sigBytes / 4);
+        },
+
+        /**
+         * Creates a copy of this word array.
+         *
+         * @return {WordArray} The clone.
+         *
+         * @example
+         *
+         *     var clone = wordArray.clone();
+         */
+        clone: function () {
+            var clone = Base.clone.call(this);
+            clone.words = this.words.slice(0);
+
+            return clone;
+        },
+
+        /**
+         * Creates a word array filled with random bytes.
+         *
+         * @param {number} nBytes The number of random bytes to generate.
+         *
+         * @return {WordArray} The random word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.lib.WordArray.random(16);
+         */
+        random: function (nBytes) {
+            var words = [];
+            for (var i = 0; i < nBytes; i += 4) {
+                words.push((Math.random() * 0x100000000) | 0);
+            }
+
+            return new WordArray.init(words, nBytes);
+        }
+    });
+
+    /**
+     * Encoder namespace.
+     */
+    var C_enc = C.enc = {};
+
+    /**
+     * Hex encoding strategy.
+     */
+    var Hex = C_enc.Hex = {
+        /**
+         * Converts a word array to a hex string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The hex string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var hexString = CryptoJS.enc.Hex.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+
+            // Convert
+            var hexChars = [];
+            for (var i = 0; i < sigBytes; i++) {
+                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                hexChars.push((bite >>> 4).toString(16));
+                hexChars.push((bite & 0x0f).toString(16));
+            }
+
+            return hexChars.join('');
+        },
+
+        /**
+         * Converts a hex string to a word array.
+         *
+         * @param {string} hexStr The hex string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Hex.parse(hexString);
+         */
+        parse: function (hexStr) {
+            // Shortcut
+            var hexStrLength = hexStr.length;
+
+            // Convert
+            var words = [];
+            for (var i = 0; i < hexStrLength; i += 2) {
+                words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);
+            }
+
+            return new WordArray.init(words, hexStrLength / 2);
+        }
+    };
+
+    /**
+     * Latin1 encoding strategy.
+     */
+    var Latin1 = C_enc.Latin1 = {
+        /**
+         * Converts a word array to a Latin1 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The Latin1 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+
+            // Convert
+            var latin1Chars = [];
+            for (var i = 0; i < sigBytes; i++) {
+                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                latin1Chars.push(String.fromCharCode(bite));
+            }
+
+            return latin1Chars.join('');
+        },
+
+        /**
+         * Converts a Latin1 string to a word array.
+         *
+         * @param {string} latin1Str The Latin1 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Latin1.parse(latin1String);
+         */
+        parse: function (latin1Str) {
+            // Shortcut
+            var latin1StrLength = latin1Str.length;
+
+            // Convert
+            var words = [];
+            for (var i = 0; i < latin1StrLength; i++) {
+                words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);
+            }
+
+            return new WordArray.init(words, latin1StrLength);
+        }
+    };
+
+    /**
+     * UTF-8 encoding strategy.
+     */
+    var Utf8 = C_enc.Utf8 = {
+        /**
+         * Converts a word array to a UTF-8 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The UTF-8 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            try {
+                return decodeURIComponent(escape(Latin1.stringify(wordArray)));
+            } catch (e) {
+                throw new Error('Malformed UTF-8 data');
+            }
+        },
+
+        /**
+         * Converts a UTF-8 string to a word array.
+         *
+         * @param {string} utf8Str The UTF-8 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Utf8.parse(utf8String);
+         */
+        parse: function (utf8Str) {
+            return Latin1.parse(unescape(encodeURIComponent(utf8Str)));
+        }
+    };
+
+    /**
+     * Abstract buffered block algorithm template.
+     *
+     * The property blockSize must be implemented in a concrete subtype.
+     *
+     * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0
+     */
+    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({
+        /**
+         * Resets this block algorithm's data buffer to its initial state.
+         *
+         * @example
+         *
+         *     bufferedBlockAlgorithm.reset();
+         */
+        reset: function () {
+            // Initial values
+            this._data = new WordArray.init();
+            this._nDataBytes = 0;
+        },
+
+        /**
+         * Adds new data to this block algorithm's buffer.
+         *
+         * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.
+         *
+         * @example
+         *
+         *     bufferedBlockAlgorithm._append('data');
+         *     bufferedBlockAlgorithm._append(wordArray);
+         */
+        _append: function (data) {
+            // Convert string to WordArray, else assume WordArray already
+            if (typeof data == 'string') {
+                data = Utf8.parse(data);
+            }
+
+            // Append
+            this._data.concat(data);
+            this._nDataBytes += data.sigBytes;
+        },
+
+        /**
+         * Processes available data blocks.
+         *
+         * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.
+         *
+         * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.
+         *
+         * @return {WordArray} The processed data.
+         *
+         * @example
+         *
+         *     var processedData = bufferedBlockAlgorithm._process();
+         *     var processedData = bufferedBlockAlgorithm._process(!!'flush');
+         */
+        _process: function (doFlush) {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+            var dataSigBytes = data.sigBytes;
+            var blockSize = this.blockSize;
+            var blockSizeBytes = blockSize * 4;
+
+            // Count blocks ready
+            var nBlocksReady = dataSigBytes / blockSizeBytes;
+            if (doFlush) {
+                // Round up to include partial blocks
+                nBlocksReady = Math.ceil(nBlocksReady);
+            } else {
+                // Round down to include only full blocks,
+                // less the number of blocks that must remain in the buffer
+                nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);
+            }
+
+            // Count words ready
+            var nWordsReady = nBlocksReady * blockSize;
+
+            // Count bytes ready
+            var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);
+
+            // Process blocks
+            if (nWordsReady) {
+                for (var offset = 0; offset < nWordsReady; offset += blockSize) {
+                    // Perform concrete-algorithm logic
+                    this._doProcessBlock(dataWords, offset);
+                }
+
+                // Remove processed words
+                var processedWords = dataWords.splice(0, nWordsReady);
+                data.sigBytes -= nBytesReady;
+            }
+
+            // Return processed words
+            return new WordArray.init(processedWords, nBytesReady);
+        },
+
+        /**
+         * Creates a copy of this object.
+         *
+         * @return {Object} The clone.
+         *
+         * @example
+         *
+         *     var clone = bufferedBlockAlgorithm.clone();
+         */
+        clone: function () {
+            var clone = Base.clone.call(this);
+            clone._data = this._data.clone();
+
+            return clone;
+        },
+
+        _minBufferSize: 0
+    });
+
+    /**
+     * Abstract hasher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)
+     */
+    var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({
+        /**
+         * Configuration options.
+         */
+        cfg: Base.extend(),
+
+        /**
+         * Initializes a newly created hasher.
+         *
+         * @param {Object} cfg (Optional) The configuration options to use for this hash computation.
+         *
+         * @example
+         *
+         *     var hasher = CryptoJS.algo.SHA256.create();
+         */
+        init: function (cfg) {
+            // Apply config defaults
+            this.cfg = this.cfg.extend(cfg);
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this hasher to its initial state.
+         *
+         * @example
+         *
+         *     hasher.reset();
+         */
+        reset: function () {
+            // Reset data buffer
+            BufferedBlockAlgorithm.reset.call(this);
+
+            // Perform concrete-hasher logic
+            this._doReset();
+        },
+
+        /**
+         * Updates this hasher with a message.
+         *
+         * @param {WordArray|string} messageUpdate The message to append.
+         *
+         * @return {Hasher} This hasher.
+         *
+         * @example
+         *
+         *     hasher.update('message');
+         *     hasher.update(wordArray);
+         */
+        update: function (messageUpdate) {
+            // Append
+            this._append(messageUpdate);
+
+            // Update the hash
+            this._process();
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Finalizes the hash computation.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} messageUpdate (Optional) A final message update.
+         *
+         * @return {WordArray} The hash.
+         *
+         * @example
+         *
+         *     var hash = hasher.finalize();
+         *     var hash = hasher.finalize('message');
+         *     var hash = hasher.finalize(wordArray);
+         */
+        finalize: function (messageUpdate) {
+            // Final message update
+            if (messageUpdate) {
+                this._append(messageUpdate);
+            }
+
+            // Perform concrete-hasher logic
+            var hash = this._doFinalize();
+
+            return hash;
+        },
+
+        blockSize: 512/32,
+
+        /**
+         * Creates a shortcut function to a hasher's object interface.
+         *
+         * @param {Hasher} hasher The hasher to create a helper for.
+         *
+         * @return {Function} The shortcut function.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);
+         */
+        _createHelper: function (hasher) {
+            return function (message, cfg) {
+                return new hasher.init(cfg).finalize(message);
+            };
+        },
+
+        /**
+         * Creates a shortcut function to the HMAC's object interface.
+         *
+         * @param {Hasher} hasher The hasher to use in this HMAC helper.
+         *
+         * @return {Function} The shortcut function.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);
+         */
+        _createHmacHelper: function (hasher) {
+            return function (message, key) {
+                return new C_algo.HMAC.init(hasher, key).finalize(message);
+            };
+        }
+    });
+
+    /**
+     * Algorithm namespace.
+     */
+    var C_algo = C.algo = {};
+
+    return C;
+}(Math));
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var C_enc = C.enc;
+
+    /**
+     * Base64 encoding strategy.
+     */
+    var Base64 = C_enc.Base64 = {
+        /**
+         * Converts a word array to a Base64 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The Base64 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var base64String = CryptoJS.enc.Base64.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+            var map = this._map;
+
+            // Clamp excess bits
+            wordArray.clamp();
+
+            // Convert
+            var base64Chars = [];
+            for (var i = 0; i < sigBytes; i += 3) {
+                var byte1 = (words[i >>> 2]       >>> (24 - (i % 4) * 8))       & 0xff;
+                var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;
+                var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;
+
+                var triplet = (byte1 << 16) | (byte2 << 8) | byte3;
+
+                for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {
+                    base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));
+                }
+            }
+
+            // Add padding
+            var paddingChar = map.charAt(64);
+            if (paddingChar) {
+                while (base64Chars.length % 4) {
+                    base64Chars.push(paddingChar);
+                }
+            }
+
+            return base64Chars.join('');
+        },
+
+        /**
+         * Converts a Base64 string to a word array.
+         *
+         * @param {string} base64Str The Base64 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Base64.parse(base64String);
+         */
+        parse: function (base64Str) {
+            // Shortcuts
+            var base64StrLength = base64Str.length;
+            var map = this._map;
+
+            // Ignore padding
+            var paddingChar = map.charAt(64);
+            if (paddingChar) {
+                var paddingIndex = base64Str.indexOf(paddingChar);
+                if (paddingIndex != -1) {
+                    base64StrLength = paddingIndex;
+                }
+            }
+
+            // Convert
+            var words = [];
+            var nBytes = 0;
+            for (var i = 0; i < base64StrLength; i++) {
+                if (i % 4) {
+                    var bits1 = map.indexOf(base64Str.charAt(i - 1)) << ((i % 4) * 2);
+                    var bits2 = map.indexOf(base64Str.charAt(i)) >>> (6 - (i % 4) * 2);
+                    words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);
+                    nBytes++;
+                }
+            }
+
+            return WordArray.create(words, nBytes);
+        },
+
+        _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='
+    };
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * Cipher core components.
+ */
+CryptoJS.lib.Cipher || (function (undefined) {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var Base = C_lib.Base;
+    var WordArray = C_lib.WordArray;
+    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm;
+    var C_enc = C.enc;
+    var Utf8 = C_enc.Utf8;
+    var Base64 = C_enc.Base64;
+    var C_algo = C.algo;
+    var EvpKDF = C_algo.EvpKDF;
+
+    /**
+     * Abstract base cipher template.
+     *
+     * @property {number} keySize This cipher's key size. Default: 4 (128 bits)
+     * @property {number} ivSize This cipher's IV size. Default: 4 (128 bits)
+     * @property {number} _ENC_XFORM_MODE A constant representing encryption mode.
+     * @property {number} _DEC_XFORM_MODE A constant representing decryption mode.
+     */
+    var Cipher = C_lib.Cipher = BufferedBlockAlgorithm.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {WordArray} iv The IV to use for this operation.
+         */
+        cfg: Base.extend(),
+
+        /**
+         * Creates this cipher in encryption mode.
+         *
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {Cipher} A cipher instance.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.createEncryptor(keyWordArray, { iv: ivWordArray });
+         */
+        createEncryptor: function (key, cfg) {
+            return this.create(this._ENC_XFORM_MODE, key, cfg);
+        },
+
+        /**
+         * Creates this cipher in decryption mode.
+         *
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {Cipher} A cipher instance.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.createDecryptor(keyWordArray, { iv: ivWordArray });
+         */
+        createDecryptor: function (key, cfg) {
+            return this.create(this._DEC_XFORM_MODE, key, cfg);
+        },
+
+        /**
+         * Initializes a newly created cipher.
+         *
+         * @param {number} xformMode Either the encryption or decryption transormation mode constant.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.create(CryptoJS.algo.AES._ENC_XFORM_MODE, keyWordArray, { iv: ivWordArray });
+         */
+        init: function (xformMode, key, cfg) {
+            // Apply config defaults
+            this.cfg = this.cfg.extend(cfg);
+
+            // Store transform mode and key
+            this._xformMode = xformMode;
+            this._key = key;
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this cipher to its initial state.
+         *
+         * @example
+         *
+         *     cipher.reset();
+         */
+        reset: function () {
+            // Reset data buffer
+            BufferedBlockAlgorithm.reset.call(this);
+
+            // Perform concrete-cipher logic
+            this._doReset();
+        },
+
+        /**
+         * Adds data to be encrypted or decrypted.
+         *
+         * @param {WordArray|string} dataUpdate The data to encrypt or decrypt.
+         *
+         * @return {WordArray} The data after processing.
+         *
+         * @example
+         *
+         *     var encrypted = cipher.process('data');
+         *     var encrypted = cipher.process(wordArray);
+         */
+        process: function (dataUpdate) {
+            // Append
+            this._append(dataUpdate);
+
+            // Process available blocks
+            return this._process();
+        },
+
+        /**
+         * Finalizes the encryption or decryption process.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} dataUpdate The final data to encrypt or decrypt.
+         *
+         * @return {WordArray} The data after final processing.
+         *
+         * @example
+         *
+         *     var encrypted = cipher.finalize();
+         *     var encrypted = cipher.finalize('data');
+         *     var encrypted = cipher.finalize(wordArray);
+         */
+        finalize: function (dataUpdate) {
+            // Final data update
+            if (dataUpdate) {
+                this._append(dataUpdate);
+            }
+
+            // Perform concrete-cipher logic
+            var finalProcessedData = this._doFinalize();
+
+            return finalProcessedData;
+        },
+
+        keySize: 128/32,
+
+        ivSize: 128/32,
+
+        _ENC_XFORM_MODE: 1,
+
+        _DEC_XFORM_MODE: 2,
+
+        /**
+         * Creates shortcut functions to a cipher's object interface.
+         *
+         * @param {Cipher} cipher The cipher to create a helper for.
+         *
+         * @return {Object} An object with encrypt and decrypt shortcut functions.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var AES = CryptoJS.lib.Cipher._createHelper(CryptoJS.algo.AES);
+         */
+        _createHelper: (function () {
+            function selectCipherStrategy(key) {
+                if (typeof key == 'string') {
+                    return PasswordBasedCipher;
+                } else {
+                    return SerializableCipher;
+                }
+            }
+
+            return function (cipher) {
+                return {
+                    encrypt: function (message, key, cfg) {
+                        return selectCipherStrategy(key).encrypt(cipher, message, key, cfg);
+                    },
+
+                    decrypt: function (ciphertext, key, cfg) {
+                        return selectCipherStrategy(key).decrypt(cipher, ciphertext, key, cfg);
+                    }
+                };
+            };
+        }())
+    });
+
+    /**
+     * Abstract base stream cipher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 1 (32 bits)
+     */
+    var StreamCipher = C_lib.StreamCipher = Cipher.extend({
+        _doFinalize: function () {
+            // Process partial blocks
+            var finalProcessedBlocks = this._process(!!'flush');
+
+            return finalProcessedBlocks;
+        },
+
+        blockSize: 1
+    });
+
+    /**
+     * Mode namespace.
+     */
+    var C_mode = C.mode = {};
+
+    /**
+     * Abstract base block cipher mode template.
+     */
+    var BlockCipherMode = C_lib.BlockCipherMode = Base.extend({
+        /**
+         * Creates this mode for encryption.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.createEncryptor(cipher, iv.words);
+         */
+        createEncryptor: function (cipher, iv) {
+            return this.Encryptor.create(cipher, iv);
+        },
+
+        /**
+         * Creates this mode for decryption.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.createDecryptor(cipher, iv.words);
+         */
+        createDecryptor: function (cipher, iv) {
+            return this.Decryptor.create(cipher, iv);
+        },
+
+        /**
+         * Initializes a newly created mode.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.Encryptor.create(cipher, iv.words);
+         */
+        init: function (cipher, iv) {
+            this._cipher = cipher;
+            this._iv = iv;
+        }
+    });
+
+    /**
+     * Cipher Block Chaining mode.
+     */
+    var CBC = C_mode.CBC = (function () {
+        /**
+         * Abstract base CBC mode.
+         */
+        var CBC = BlockCipherMode.extend();
+
+        /**
+         * CBC encryptor.
+         */
+        CBC.Encryptor = CBC.extend({
+            /**
+             * Processes the data block at offset.
+             *
+             * @param {Array} words The data words to operate on.
+             * @param {number} offset The offset where the block starts.
+             *
+             * @example
+             *
+             *     mode.processBlock(data.words, offset);
+             */
+            processBlock: function (words, offset) {
+                // Shortcuts
+                var cipher = this._cipher;
+                var blockSize = cipher.blockSize;
+
+                // XOR and encrypt
+                xorBlock.call(this, words, offset, blockSize);
+                cipher.encryptBlock(words, offset);
+
+                // Remember this block to use with next block
+                this._prevBlock = words.slice(offset, offset + blockSize);
+            }
+        });
+
+        /**
+         * CBC decryptor.
+         */
+        CBC.Decryptor = CBC.extend({
+            /**
+             * Processes the data block at offset.
+             *
+             * @param {Array} words The data words to operate on.
+             * @param {number} offset The offset where the block starts.
+             *
+             * @example
+             *
+             *     mode.processBlock(data.words, offset);
+             */
+            processBlock: function (words, offset) {
+                // Shortcuts
+                var cipher = this._cipher;
+                var blockSize = cipher.blockSize;
+
+                // Remember this block to use with next block
+                var thisBlock = words.slice(offset, offset + blockSize);
+
+                // Decrypt and XOR
+                cipher.decryptBlock(words, offset);
+                xorBlock.call(this, words, offset, blockSize);
+
+                // This block becomes the previous block
+                this._prevBlock = thisBlock;
+            }
+        });
+
+        function xorBlock(words, offset, blockSize) {
+            // Shortcut
+            var iv = this._iv;
+
+            // Choose mixing block
+            if (iv) {
+                var block = iv;
+
+                // Remove IV for subsequent blocks
+                this._iv = undefined;
+            } else {
+                var block = this._prevBlock;
+            }
+
+            // XOR blocks
+            for (var i = 0; i < blockSize; i++) {
+                words[offset + i] ^= block[i];
+            }
+        }
+
+        return CBC;
+    }());
+
+    /**
+     * Padding namespace.
+     */
+    var C_pad = C.pad = {};
+
+    /**
+     * PKCS #5/7 padding strategy.
+     */
+    var Pkcs7 = C_pad.Pkcs7 = {
+        /**
+         * Pads data using the algorithm defined in PKCS #5/7.
+         *
+         * @param {WordArray} data The data to pad.
+         * @param {number} blockSize The multiple that the data should be padded to.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     CryptoJS.pad.Pkcs7.pad(wordArray, 4);
+         */
+        pad: function (data, blockSize) {
+            // Shortcut
+            var blockSizeBytes = blockSize * 4;
+
+            // Count padding bytes
+            var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;
+
+            // Create padding word
+            var paddingWord = (nPaddingBytes << 24) | (nPaddingBytes << 16) | (nPaddingBytes << 8) | nPaddingBytes;
+
+            // Create padding
+            var paddingWords = [];
+            for (var i = 0; i < nPaddingBytes; i += 4) {
+                paddingWords.push(paddingWord);
+            }
+            var padding = WordArray.create(paddingWords, nPaddingBytes);
+
+            // Add padding
+            data.concat(padding);
+        },
+
+        /**
+         * Unpads data that had been padded using the algorithm defined in PKCS #5/7.
+         *
+         * @param {WordArray} data The data to unpad.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     CryptoJS.pad.Pkcs7.unpad(wordArray);
+         */
+        unpad: function (data) {
+            // Get number of padding bytes from last byte
+            var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;
+
+            // Remove padding
+            data.sigBytes -= nPaddingBytes;
+        }
+    };
+
+    /**
+     * Abstract base block cipher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 4 (128 bits)
+     */
+    var BlockCipher = C_lib.BlockCipher = Cipher.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {Mode} mode The block mode to use. Default: CBC
+         * @property {Padding} padding The padding strategy to use. Default: Pkcs7
+         */
+        cfg: Cipher.cfg.extend({
+            mode: CBC,
+            padding: Pkcs7
+        }),
+
+        reset: function () {
+            // Reset cipher
+            Cipher.reset.call(this);
+
+            // Shortcuts
+            var cfg = this.cfg;
+            var iv = cfg.iv;
+            var mode = cfg.mode;
+
+            // Reset block mode
+            if (this._xformMode == this._ENC_XFORM_MODE) {
+                var modeCreator = mode.createEncryptor;
+            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
+                var modeCreator = mode.createDecryptor;
+
+                // Keep at least one block in the buffer for unpadding
+                this._minBufferSize = 1;
+            }
+            this._mode = modeCreator.call(mode, this, iv && iv.words);
+        },
+
+        _doProcessBlock: function (words, offset) {
+            this._mode.processBlock(words, offset);
+        },
+
+        _doFinalize: function () {
+            // Shortcut
+            var padding = this.cfg.padding;
+
+            // Finalize
+            if (this._xformMode == this._ENC_XFORM_MODE) {
+                // Pad data
+                padding.pad(this._data, this.blockSize);
+
+                // Process final blocks
+                var finalProcessedBlocks = this._process(!!'flush');
+            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
+                // Process final blocks
+                var finalProcessedBlocks = this._process(!!'flush');
+
+                // Unpad data
+                padding.unpad(finalProcessedBlocks);
+            }
+
+            return finalProcessedBlocks;
+        },
+
+        blockSize: 128/32
+    });
+
+    /**
+     * A collection of cipher parameters.
+     *
+     * @property {WordArray} ciphertext The raw ciphertext.
+     * @property {WordArray} key The key to this ciphertext.
+     * @property {WordArray} iv The IV used in the ciphering operation.
+     * @property {WordArray} salt The salt used with a key derivation function.
+     * @property {Cipher} algorithm The cipher algorithm.
+     * @property {Mode} mode The block mode used in the ciphering operation.
+     * @property {Padding} padding The padding scheme used in the ciphering operation.
+     * @property {number} blockSize The block size of the cipher.
+     * @property {Format} formatter The default formatting strategy to convert this cipher params object to a string.
+     */
+    var CipherParams = C_lib.CipherParams = Base.extend({
+        /**
+         * Initializes a newly created cipher params object.
+         *
+         * @param {Object} cipherParams An object with any of the possible cipher parameters.
+         *
+         * @example
+         *
+         *     var cipherParams = CryptoJS.lib.CipherParams.create({
+         *         ciphertext: ciphertextWordArray,
+         *         key: keyWordArray,
+         *         iv: ivWordArray,
+         *         salt: saltWordArray,
+         *         algorithm: CryptoJS.algo.AES,
+         *         mode: CryptoJS.mode.CBC,
+         *         padding: CryptoJS.pad.PKCS7,
+         *         blockSize: 4,
+         *         formatter: CryptoJS.format.OpenSSL
+         *     });
+         */
+        init: function (cipherParams) {
+            this.mixIn(cipherParams);
+        },
+
+        /**
+         * Converts this cipher params object to a string.
+         *
+         * @param {Format} formatter (Optional) The formatting strategy to use.
+         *
+         * @return {string} The stringified cipher params.
+         *
+         * @throws Error If neither the formatter nor the default formatter is set.
+         *
+         * @example
+         *
+         *     var string = cipherParams + '';
+         *     var string = cipherParams.toString();
+         *     var string = cipherParams.toString(CryptoJS.format.OpenSSL);
+         */
+        toString: function (formatter) {
+            return (formatter || this.formatter).stringify(this);
+        }
+    });
+
+    /**
+     * Format namespace.
+     */
+    var C_format = C.format = {};
+
+    /**
+     * OpenSSL formatting strategy.
+     */
+    var OpenSSLFormatter = C_format.OpenSSL = {
+        /**
+         * Converts a cipher params object to an OpenSSL-compatible string.
+         *
+         * @param {CipherParams} cipherParams The cipher params object.
+         *
+         * @return {string} The OpenSSL-compatible string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var openSSLString = CryptoJS.format.OpenSSL.stringify(cipherParams);
+         */
+        stringify: function (cipherParams) {
+            // Shortcuts
+            var ciphertext = cipherParams.ciphertext;
+            var salt = cipherParams.salt;
+
+            // Format
+            if (salt) {
+                var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);
+            } else {
+                var wordArray = ciphertext;
+            }
+
+            return wordArray.toString(Base64);
+        },
+
+        /**
+         * Converts an OpenSSL-compatible string to a cipher params object.
+         *
+         * @param {string} openSSLStr The OpenSSL-compatible string.
+         *
+         * @return {CipherParams} The cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipherParams = CryptoJS.format.OpenSSL.parse(openSSLString);
+         */
+        parse: function (openSSLStr) {
+            // Parse base64
+            var ciphertext = Base64.parse(openSSLStr);
+
+            // Shortcut
+            var ciphertextWords = ciphertext.words;
+
+            // Test for salt
+            if (ciphertextWords[0] == 0x53616c74 && ciphertextWords[1] == 0x65645f5f) {
+                // Extract salt
+                var salt = WordArray.create(ciphertextWords.slice(2, 4));
+
+                // Remove salt from ciphertext
+                ciphertextWords.splice(0, 4);
+                ciphertext.sigBytes -= 16;
+            }
+
+            return CipherParams.create({ ciphertext: ciphertext, salt: salt });
+        }
+    };
+
+    /**
+     * A cipher wrapper that returns ciphertext as a serializable cipher params object.
+     */
+    var SerializableCipher = C_lib.SerializableCipher = Base.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {Formatter} format The formatting strategy to convert cipher param objects to and from a string. Default: OpenSSL
+         */
+        cfg: Base.extend({
+            format: OpenSSLFormatter
+        }),
+
+        /**
+         * Encrypts a message.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {WordArray|string} message The message to encrypt.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {CipherParams} A cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key);
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv });
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         */
+        encrypt: function (cipher, message, key, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Encrypt
+            var encryptor = cipher.createEncryptor(key, cfg);
+            var ciphertext = encryptor.finalize(message);
+
+            // Shortcut
+            var cipherCfg = encryptor.cfg;
+
+            // Create and return serializable cipher params
+            return CipherParams.create({
+                ciphertext: ciphertext,
+                key: key,
+                iv: cipherCfg.iv,
+                algorithm: cipher,
+                mode: cipherCfg.mode,
+                padding: cipherCfg.padding,
+                blockSize: cipher.blockSize,
+                formatter: cfg.format
+            });
+        },
+
+        /**
+         * Decrypts serialized ciphertext.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {WordArray} The plaintext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         */
+        decrypt: function (cipher, ciphertext, key, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Convert string to CipherParams
+            ciphertext = this._parse(ciphertext, cfg.format);
+
+            // Decrypt
+            var plaintext = cipher.createDecryptor(key, cfg).finalize(ciphertext.ciphertext);
+
+            return plaintext;
+        },
+
+        /**
+         * Converts serialized ciphertext to CipherParams,
+         * else assumed CipherParams already and returns ciphertext unchanged.
+         *
+         * @param {CipherParams|string} ciphertext The ciphertext.
+         * @param {Formatter} format The formatting strategy to use to parse serialized ciphertext.
+         *
+         * @return {CipherParams} The unserialized ciphertext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher._parse(ciphertextStringOrParams, format);
+         */
+        _parse: function (ciphertext, format) {
+            if (typeof ciphertext == 'string') {
+                return format.parse(ciphertext, this);
+            } else {
+                return ciphertext;
+            }
+        }
+    });
+
+    /**
+     * Key derivation function namespace.
+     */
+    var C_kdf = C.kdf = {};
+
+    /**
+     * OpenSSL key derivation function.
+     */
+    var OpenSSLKdf = C_kdf.OpenSSL = {
+        /**
+         * Derives a key and IV from a password.
+         *
+         * @param {string} password The password to derive from.
+         * @param {number} keySize The size in words of the key to generate.
+         * @param {number} ivSize The size in words of the IV to generate.
+         * @param {WordArray|string} salt (Optional) A 64-bit salt to use. If omitted, a salt will be generated randomly.
+         *
+         * @return {CipherParams} A cipher params object with the key, IV, and salt.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32);
+         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32, 'saltsalt');
+         */
+        execute: function (password, keySize, ivSize, salt) {
+            // Generate random salt
+            if (!salt) {
+                salt = WordArray.random(64/8);
+            }
+
+            // Derive key and IV
+            var key = EvpKDF.create({ keySize: keySize + ivSize }).compute(password, salt);
+
+            // Separate key and IV
+            var iv = WordArray.create(key.words.slice(keySize), ivSize * 4);
+            key.sigBytes = keySize * 4;
+
+            // Return params
+            return CipherParams.create({ key: key, iv: iv, salt: salt });
+        }
+    };
+
+    /**
+     * A serializable cipher wrapper that derives the key from a password,
+     * and returns ciphertext as a serializable cipher params object.
+     */
+    var PasswordBasedCipher = C_lib.PasswordBasedCipher = SerializableCipher.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {KDF} kdf The key derivation function to use to generate a key and IV from a password. Default: OpenSSL
+         */
+        cfg: SerializableCipher.cfg.extend({
+            kdf: OpenSSLKdf
+        }),
+
+        /**
+         * Encrypts a message using a password.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {WordArray|string} message The message to encrypt.
+         * @param {string} password The password.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {CipherParams} A cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password');
+         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password', { format: CryptoJS.format.OpenSSL });
+         */
+        encrypt: function (cipher, message, password, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Derive key and other params
+            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize);
+
+            // Add IV to config
+            cfg.iv = derivedParams.iv;
+
+            // Encrypt
+            var ciphertext = SerializableCipher.encrypt.call(this, cipher, message, derivedParams.key, cfg);
+
+            // Mix in derived params
+            ciphertext.mixIn(derivedParams);
+
+            return ciphertext;
+        },
+
+        /**
+         * Decrypts serialized ciphertext using a password.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
+         * @param {string} password The password.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {WordArray} The plaintext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, 'password', { format: CryptoJS.format.OpenSSL });
+         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, 'password', { format: CryptoJS.format.OpenSSL });
+         */
+        decrypt: function (cipher, ciphertext, password, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Convert string to CipherParams
+            ciphertext = this._parse(ciphertext, cfg.format);
+
+            // Derive key and other params
+            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize, ciphertext.salt);
+
+            // Add IV to config
+            cfg.iv = derivedParams.iv;
+
+            // Decrypt
+            var plaintext = SerializableCipher.decrypt.call(this, cipher, ciphertext, derivedParams.key, cfg);
+
+            return plaintext;
+        }
+    });
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var BlockCipher = C_lib.BlockCipher;
+    var C_algo = C.algo;
+
+    // Lookup tables
+    var SBOX = [];
+    var INV_SBOX = [];
+    var SUB_MIX_0 = [];
+    var SUB_MIX_1 = [];
+    var SUB_MIX_2 = [];
+    var SUB_MIX_3 = [];
+    var INV_SUB_MIX_0 = [];
+    var INV_SUB_MIX_1 = [];
+    var INV_SUB_MIX_2 = [];
+    var INV_SUB_MIX_3 = [];
+
+    // Compute lookup tables
+    (function () {
+        // Compute double table
+        var d = [];
+        for (var i = 0; i < 256; i++) {
+            if (i < 128) {
+                d[i] = i << 1;
+            } else {
+                d[i] = (i << 1) ^ 0x11b;
+            }
+        }
+
+        // Walk GF(2^8)
+        var x = 0;
+        var xi = 0;
+        for (var i = 0; i < 256; i++) {
+            // Compute sbox
+            var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4);
+            sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63;
+            SBOX[x] = sx;
+            INV_SBOX[sx] = x;
+
+            // Compute multiplication
+            var x2 = d[x];
+            var x4 = d[x2];
+            var x8 = d[x4];
+
+            // Compute sub bytes, mix columns tables
+            var t = (d[sx] * 0x101) ^ (sx * 0x1010100);
+            SUB_MIX_0[x] = (t << 24) | (t >>> 8);
+            SUB_MIX_1[x] = (t << 16) | (t >>> 16);
+            SUB_MIX_2[x] = (t << 8)  | (t >>> 24);
+            SUB_MIX_3[x] = t;
+
+            // Compute inv sub bytes, inv mix columns tables
+            var t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100);
+            INV_SUB_MIX_0[sx] = (t << 24) | (t >>> 8);
+            INV_SUB_MIX_1[sx] = (t << 16) | (t >>> 16);
+            INV_SUB_MIX_2[sx] = (t << 8)  | (t >>> 24);
+            INV_SUB_MIX_3[sx] = t;
+
+            // Compute next counter
+            if (!x) {
+                x = xi = 1;
+            } else {
+                x = x2 ^ d[d[d[x8 ^ x2]]];
+                xi ^= d[d[xi]];
+            }
+        }
+    }());
+
+    // Precomputed Rcon lookup
+    var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36];
+
+    /**
+     * AES block cipher algorithm.
+     */
+    var AES = C_algo.AES = BlockCipher.extend({
+        _doReset: function () {
+            // Shortcuts
+            var key = this._key;
+            var keyWords = key.words;
+            var keySize = key.sigBytes / 4;
+
+            // Compute number of rounds
+            var nRounds = this._nRounds = keySize + 6
+
+            // Compute number of key schedule rows
+            var ksRows = (nRounds + 1) * 4;
+
+            // Compute key schedule
+            var keySchedule = this._keySchedule = [];
+            for (var ksRow = 0; ksRow < ksRows; ksRow++) {
+                if (ksRow < keySize) {
+                    keySchedule[ksRow] = keyWords[ksRow];
+                } else {
+                    var t = keySchedule[ksRow - 1];
+
+                    if (!(ksRow % keySize)) {
+                        // Rot word
+                        t = (t << 8) | (t >>> 24);
+
+                        // Sub word
+                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
+
+                        // Mix Rcon
+                        t ^= RCON[(ksRow / keySize) | 0] << 24;
+                    } else if (keySize > 6 && ksRow % keySize == 4) {
+                        // Sub word
+                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
+                    }
+
+                    keySchedule[ksRow] = keySchedule[ksRow - keySize] ^ t;
+                }
+            }
+
+            // Compute inv key schedule
+            var invKeySchedule = this._invKeySchedule = [];
+            for (var invKsRow = 0; invKsRow < ksRows; invKsRow++) {
+                var ksRow = ksRows - invKsRow;
+
+                if (invKsRow % 4) {
+                    var t = keySchedule[ksRow];
+                } else {
+                    var t = keySchedule[ksRow - 4];
+                }
+
+                if (invKsRow < 4 || ksRow <= 4) {
+                    invKeySchedule[invKsRow] = t;
+                } else {
+                    invKeySchedule[invKsRow] = INV_SUB_MIX_0[SBOX[t >>> 24]] ^ INV_SUB_MIX_1[SBOX[(t >>> 16) & 0xff]] ^
+                                               INV_SUB_MIX_2[SBOX[(t >>> 8) & 0xff]] ^ INV_SUB_MIX_3[SBOX[t & 0xff]];
+                }
+            }
+        },
+
+        encryptBlock: function (M, offset) {
+            this._doCryptBlock(M, offset, this._keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX);
+        },
+
+        decryptBlock: function (M, offset) {
+            // Swap 2nd and 4th rows
+            var t = M[offset + 1];
+            M[offset + 1] = M[offset + 3];
+            M[offset + 3] = t;
+
+            this._doCryptBlock(M, offset, this._invKeySchedule, INV_SUB_MIX_0, INV_SUB_MIX_1, INV_SUB_MIX_2, INV_SUB_MIX_3, INV_SBOX);
+
+            // Inv swap 2nd and 4th rows
+            var t = M[offset + 1];
+            M[offset + 1] = M[offset + 3];
+            M[offset + 3] = t;
+        },
+
+        _doCryptBlock: function (M, offset, keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX) {
+            // Shortcut
+            var nRounds = this._nRounds;
+
+            // Get input, add round key
+            var s0 = M[offset]     ^ keySchedule[0];
+            var s1 = M[offset + 1] ^ keySchedule[1];
+            var s2 = M[offset + 2] ^ keySchedule[2];
+            var s3 = M[offset + 3] ^ keySchedule[3];
+
+            // Key schedule row counter
+            var ksRow = 4;
+
+            // Rounds
+            for (var round = 1; round < nRounds; round++) {
+                // Shift rows, sub bytes, mix columns, add round key
+                var t0 = SUB_MIX_0[s0 >>> 24] ^ SUB_MIX_1[(s1 >>> 16) & 0xff] ^ SUB_MIX_2[(s2 >>> 8) & 0xff] ^ SUB_MIX_3[s3 & 0xff] ^ keySchedule[ksRow++];
+                var t1 = SUB_MIX_0[s1 >>> 24] ^ SUB_MIX_1[(s2 >>> 16) & 0xff] ^ SUB_MIX_2[(s3 >>> 8) & 0xff] ^ SUB_MIX_3[s0 & 0xff] ^ keySchedule[ksRow++];
+                var t2 = SUB_MIX_0[s2 >>> 24] ^ SUB_MIX_1[(s3 >>> 16) & 0xff] ^ SUB_MIX_2[(s0 >>> 8) & 0xff] ^ SUB_MIX_3[s1 & 0xff] ^ keySchedule[ksRow++];
+                var t3 = SUB_MIX_0[s3 >>> 24] ^ SUB_MIX_1[(s0 >>> 16) & 0xff] ^ SUB_MIX_2[(s1 >>> 8) & 0xff] ^ SUB_MIX_3[s2 & 0xff] ^ keySchedule[ksRow++];
+
+                // Update state
+                s0 = t0;
+                s1 = t1;
+                s2 = t2;
+                s3 = t3;
+            }
+
+            // Shift rows, sub bytes, add round key
+            var t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++];
+            var t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++];
+            var t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++];
+            var t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++];
+
+            // Set output
+            M[offset]     = t0;
+            M[offset + 1] = t1;
+            M[offset + 2] = t2;
+            M[offset + 3] = t3;
+        },
+
+        keySize: 256/32
+    });
+
+    /**
+     * Shortcut functions to the cipher's object interface.
+     *
+     * @example
+     *
+     *     var ciphertext = CryptoJS.AES.encrypt(message, key, cfg);
+     *     var plaintext  = CryptoJS.AES.decrypt(ciphertext, key, cfg);
+     */
+    C.AES = BlockCipher._createHelper(AES);
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var Hasher = C_lib.Hasher;
+    var C_algo = C.algo;
+
+    // Reusable object
+    var W = [];
+
+    /**
+     * SHA-1 hash algorithm.
+     */
+    var SHA1 = C_algo.SHA1 = Hasher.extend({
+        _doReset: function () {
+            this._hash = new WordArray.init([
+                0x67452301, 0xefcdab89,
+                0x98badcfe, 0x10325476,
+                0xc3d2e1f0
+            ]);
+        },
+
+        _doProcessBlock: function (M, offset) {
+            // Shortcut
+            var H = this._hash.words;
+
+            // Working variables
+            var a = H[0];
+            var b = H[1];
+            var c = H[2];
+            var d = H[3];
+            var e = H[4];
+
+            // Computation
+            for (var i = 0; i < 80; i++) {
+                if (i < 16) {
+                    W[i] = M[offset + i] | 0;
+                } else {
+                    var n = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];
+                    W[i] = (n << 1) | (n >>> 31);
+                }
+
+                var t = ((a << 5) | (a >>> 27)) + e + W[i];
+                if (i < 20) {
+                    t += ((b & c) | (~b & d)) + 0x5a827999;
+                } else if (i < 40) {
+                    t += (b ^ c ^ d) + 0x6ed9eba1;
+                } else if (i < 60) {
+                    t += ((b & c) | (b & d) | (c & d)) - 0x70e44324;
+                } else /* if (i < 80) */ {
+                    t += (b ^ c ^ d) - 0x359d3e2a;
+                }
+
+                e = d;
+                d = c;
+                c = (b << 30) | (b >>> 2);
+                b = a;
+                a = t;
+            }
+
+            // Intermediate hash value
+            H[0] = (H[0] + a) | 0;
+            H[1] = (H[1] + b) | 0;
+            H[2] = (H[2] + c) | 0;
+            H[3] = (H[3] + d) | 0;
+            H[4] = (H[4] + e) | 0;
+        },
+
+        _doFinalize: function () {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+
+            var nBitsTotal = this._nDataBytes * 8;
+            var nBitsLeft = data.sigBytes * 8;
+
+            // Add padding
+            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
+            data.sigBytes = dataWords.length * 4;
+
+            // Hash final blocks
+            this._process();
+
+            // Return final computed hash
+            return this._hash;
+        },
+
+        clone: function () {
+            var clone = Hasher.clone.call(this);
+            clone._hash = this._hash.clone();
+
+            return clone;
+        }
+    });
+
+    /**
+     * Shortcut function to the hasher's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     *
+     * @return {WordArray} The hash.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hash = CryptoJS.SHA1('message');
+     *     var hash = CryptoJS.SHA1(wordArray);
+     */
+    C.SHA1 = Hasher._createHelper(SHA1);
+
+    /**
+     * Shortcut function to the HMAC's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     * @param {WordArray|string} key The secret key.
+     *
+     * @return {WordArray} The HMAC.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hmac = CryptoJS.HmacSHA1(message, key);
+     */
+    C.HmacSHA1 = Hasher._createHmacHelper(SHA1);
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function (Math) {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var Hasher = C_lib.Hasher;
+    var C_algo = C.algo;
+
+    // Initialization and round constants tables
+    var H = [];
+    var K = [];
+
+    // Compute constants
+    (function () {
+        function isPrime(n) {
+            var sqrtN = Math.sqrt(n);
+            for (var factor = 2; factor <= sqrtN; factor++) {
+                if (!(n % factor)) {
+                    return false;
+                }
+            }
+
+            return true;
+        }
+
+        function getFractionalBits(n) {
+            return ((n - (n | 0)) * 0x100000000) | 0;
+        }
+
+        var n = 2;
+        var nPrime = 0;
+        while (nPrime < 64) {
+            if (isPrime(n)) {
+                if (nPrime < 8) {
+                    H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));
+                }
+                K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));
+
+                nPrime++;
+            }
+
+            n++;
+        }
+    }());
+
+    // Reusable object
+    var W = [];
+
+    /**
+     * SHA-256 hash algorithm.
+     */
+    var SHA256 = C_algo.SHA256 = Hasher.extend({
+        _doReset: function () {
+            this._hash = new WordArray.init(H.slice(0));
+        },
+
+        _doProcessBlock: function (M, offset) {
+            // Shortcut
+            var H = this._hash.words;
+
+            // Working variables
+            var a = H[0];
+            var b = H[1];
+            var c = H[2];
+            var d = H[3];
+            var e = H[4];
+            var f = H[5];
+            var g = H[6];
+            var h = H[7];
+
+            // Computation
+            for (var i = 0; i < 64; i++) {
+                if (i < 16) {
+                    W[i] = M[offset + i] | 0;
+                } else {
+                    var gamma0x = W[i - 15];
+                    var gamma0  = ((gamma0x << 25) | (gamma0x >>> 7))  ^
+                                  ((gamma0x << 14) | (gamma0x >>> 18)) ^
+                                   (gamma0x >>> 3);
+
+                    var gamma1x = W[i - 2];
+                    var gamma1  = ((gamma1x << 15) | (gamma1x >>> 17)) ^
+                                  ((gamma1x << 13) | (gamma1x >>> 19)) ^
+                                   (gamma1x >>> 10);
+
+                    W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];
+                }
+
+                var ch  = (e & f) ^ (~e & g);
+                var maj = (a & b) ^ (a & c) ^ (b & c);
+
+                var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));
+                var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7)  | (e >>> 25));
+
+                var t1 = h + sigma1 + ch + K[i] + W[i];
+                var t2 = sigma0 + maj;
+
+                h = g;
+                g = f;
+                f = e;
+                e = (d + t1) | 0;
+                d = c;
+                c = b;
+                b = a;
+                a = (t1 + t2) | 0;
+            }
+
+            // Intermediate hash value
+            H[0] = (H[0] + a) | 0;
+            H[1] = (H[1] + b) | 0;
+            H[2] = (H[2] + c) | 0;
+            H[3] = (H[3] + d) | 0;
+            H[4] = (H[4] + e) | 0;
+            H[5] = (H[5] + f) | 0;
+            H[6] = (H[6] + g) | 0;
+            H[7] = (H[7] + h) | 0;
+        },
+
+        _doFinalize: function () {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+
+            var nBitsTotal = this._nDataBytes * 8;
+            var nBitsLeft = data.sigBytes * 8;
+
+            // Add padding
+            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
+            data.sigBytes = dataWords.length * 4;
+
+            // Hash final blocks
+            this._process();
+
+            // Return final computed hash
+            return this._hash;
+        },
+
+        clone: function () {
+            var clone = Hasher.clone.call(this);
+            clone._hash = this._hash.clone();
+
+            return clone;
+        }
+    });
+
+    /**
+     * Shortcut function to the hasher's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     *
+     * @return {WordArray} The hash.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hash = CryptoJS.SHA256('message');
+     *     var hash = CryptoJS.SHA256(wordArray);
+     */
+    C.SHA256 = Hasher._createHelper(SHA256);
+
+    /**
+     * Shortcut function to the HMAC's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     * @param {WordArray|string} key The secret key.
+     *
+     * @return {WordArray} The HMAC.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hmac = CryptoJS.HmacSHA256(message, key);
+     */
+    C.HmacSHA256 = Hasher._createHmacHelper(SHA256);
+}(Math));
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var Base = C_lib.Base;
+    var C_enc = C.enc;
+    var Utf8 = C_enc.Utf8;
+    var C_algo = C.algo;
+
+    /**
+     * HMAC algorithm.
+     */
+    var HMAC = C_algo.HMAC = Base.extend({
+        /**
+         * Initializes a newly created HMAC.
+         *
+         * @param {Hasher} hasher The hash algorithm to use.
+         * @param {WordArray|string} key The secret key.
+         *
+         * @example
+         *
+         *     var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);
+         */
+        init: function (hasher, key) {
+            // Init hasher
+            hasher = this._hasher = new hasher.init();
+
+            // Convert string to WordArray, else assume WordArray already
+            if (typeof key == 'string') {
+                key = Utf8.parse(key);
+            }
+
+            // Shortcuts
+            var hasherBlockSize = hasher.blockSize;
+            var hasherBlockSizeBytes = hasherBlockSize * 4;
+
+            // Allow arbitrary length keys
+            if (key.sigBytes > hasherBlockSizeBytes) {
+                key = hasher.finalize(key);
+            }
+
+            // Clamp excess bits
+            key.clamp();
+
+            // Clone key for inner and outer pads
+            var oKey = this._oKey = key.clone();
+            var iKey = this._iKey = key.clone();
+
+            // Shortcuts
+            var oKeyWords = oKey.words;
+            var iKeyWords = iKey.words;
+
+            // XOR keys with pad constants
+            for (var i = 0; i < hasherBlockSize; i++) {
+                oKeyWords[i] ^= 0x5c5c5c5c;
+                iKeyWords[i] ^= 0x36363636;
+            }
+            oKey.sigBytes = iKey.sigBytes = hasherBlockSizeBytes;
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this HMAC to its initial state.
+         *
+         * @example
+         *
+         *     hmacHasher.reset();
+         */
+        reset: function () {
+            // Shortcut
+            var hasher = this._hasher;
+
+            // Reset
+            hasher.reset();
+            hasher.update(this._iKey);
+        },
+
+        /**
+         * Updates this HMAC with a message.
+         *
+         * @param {WordArray|string} messageUpdate The message to append.
+         *
+         * @return {HMAC} This HMAC instance.
+         *
+         * @example
+         *
+         *     hmacHasher.update('message');
+         *     hmacHasher.update(wordArray);
+         */
+        update: function (messageUpdate) {
+            this._hasher.update(messageUpdate);
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Finalizes the HMAC computation.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} messageUpdate (Optional) A final message update.
+         *
+         * @return {WordArray} The HMAC.
+         *
+         * @example
+         *
+         *     var hmac = hmacHasher.finalize();
+         *     var hmac = hmacHasher.finalize('message');
+         *     var hmac = hmacHasher.finalize(wordArray);
+         */
+        finalize: function (messageUpdate) {
+            // Shortcut
+            var hasher = this._hasher;
+
+            // Compute HMAC
+            var innerHash = hasher.finalize(messageUpdate);
+            hasher.reset();
+            var hmac = hasher.finalize(this._oKey.clone().concat(innerHash));
+
+            return hmac;
+        }
+    });
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * A noop padding strategy.
+ */
+CryptoJS.pad.NoPadding = {
+    pad: function () {
+    },
+
+    unpad: function () {
+    }
+};
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * Counter block mode.
+ */
+CryptoJS.mode.CTR = (function () {
+    var CTR = CryptoJS.lib.BlockCipherMode.extend();
+
+    var Encryptor = CTR.Encryptor = CTR.extend({
+        processBlock: function (words, offset) {
+            // Shortcuts
+            var cipher = this._cipher
+            var blockSize = cipher.blockSize;
+            var iv = this._iv;
+            var counter = this._counter;
+
+            // Generate keystream
+            if (iv) {
+                counter = this._counter = iv.slice(0);
+
+                // Remove IV for subsequent blocks
+                this._iv = undefined;
+            }
+            var keystream = counter.slice(0);
+            cipher.encryptBlock(keystream, 0);
+
+            // Increment counter
+            counter[blockSize - 1] = (counter[blockSize - 1] + 1) | 0
+
+            // Encrypt
+            for (var i = 0; i < blockSize; i++) {
+                words[offset + i] ^= keystream[i];
+            }
+        }
+    });
+
+    CTR.Decryptor = Encryptor;
+
+    return CTR;
+}());
+
+
+  return CryptoJS
+
+}));
+/*!
+ * Source: build/lib/otr/build/dep/eventemitter.js, license: MIT, url: http://git.io/ee */
+/*!
+ * EventEmitter v4.2.3 - git.io/ee
+ * Oliver Caldwell
+ * MIT license
+ * @preserve
+ */
+
+(function () {
+	'use strict';
+
+	/**
+	 * Class for managing events.
+	 * Can be extended to provide event functionality in other classes.
+	 *
+	 * @class EventEmitter Manages event registering and emitting.
+	 */
+	function EventEmitter() {}
+
+	// Shortcuts to improve speed and size
+
+	// Easy access to the prototype
+	var proto = EventEmitter.prototype;
+
+	/**
+	 * Finds the index of the listener for the event in it's storage array.
+	 *
+	 * @param {Function[]} listeners Array of listeners to search through.
+	 * @param {Function} listener Method to look for.
+	 * @return {Number} Index of the specified listener, -1 if not found
+	 * @api private
+	 */
+	function indexOfListener(listeners, listener) {
+		var i = listeners.length;
+		while (i--) {
+			if (listeners[i].listener === listener) {
+				return i;
+			}
+		}
+
+		return -1;
+	}
+
+	/**
+	 * Alias a method while keeping the context correct, to allow for overwriting of target method.
+	 *
+	 * @param {String} name The name of the target method.
+	 * @return {Function} The aliased method
+	 * @api private
+	 */
+	function alias(name) {
+		return function aliasClosure() {
+			return this[name].apply(this, arguments);
+		};
+	}
+
+	/**
+	 * Returns the listener array for the specified event.
+	 * Will initialise the event object and listener arrays if required.
+	 * Will return an object if you use a regex search. The object contains keys for each matched event. So /ba[rz]/ might return an object containing bar and baz. But only if you have either defined them with defineEvent or added some listeners to them.
+	 * Each property in the object response is an array of listener functions.
+	 *
+	 * @param {String|RegExp} evt Name of the event to return the listeners from.
+	 * @return {Function[]|Object} All listener functions for the event.
+	 */
+	proto.getListeners = function getListeners(evt) {
+		var events = this._getEvents();
+		var response;
+		var key;
+
+		// Return a concatenated array of all matching events if
+		// the selector is a regular expression.
+		if (typeof evt === 'object') {
+			response = {};
+			for (key in events) {
+				if (events.hasOwnProperty(key) && evt.test(key)) {
+					response[key] = events[key];
+				}
+			}
+		}
+		else {
+			response = events[evt] || (events[evt] = []);
+		}
+
+		return response;
+	};
+
+	/**
+	 * Takes a list of listener objects and flattens it into a list of listener functions.
+	 *
+	 * @param {Object[]} listeners Raw listener objects.
+	 * @return {Function[]} Just the listener functions.
+	 */
+	proto.flattenListeners = function flattenListeners(listeners) {
+		var flatListeners = [];
+		var i;
+
+		for (i = 0; i < listeners.length; i += 1) {
+			flatListeners.push(listeners[i].listener);
+		}
+
+		return flatListeners;
+	};
+
+	/**
+	 * Fetches the requested listeners via getListeners but will always return the results inside an object. This is mainly for internal use but others may find it useful.
+	 *
+	 * @param {String|RegExp} evt Name of the event to return the listeners from.
+	 * @return {Object} All listener functions for an event in an object.
+	 */
+	proto.getListenersAsObject = function getListenersAsObject(evt) {
+		var listeners = this.getListeners(evt);
+		var response;
+
+		if (listeners instanceof Array) {
+			response = {};
+			response[evt] = listeners;
+		}
+
+		return response || listeners;
+	};
+
+	/**
+	 * Adds a listener function to the specified event.
+	 * The listener will not be added if it is a duplicate.
+	 * If the listener returns true then it will be removed after it is called.
+	 * If you pass a regular expression as the event name then the listener will be added to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to attach the listener to.
+	 * @param {Function} listener Method to be called when the event is emitted. If the function returns true then it will be removed after calling.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addListener = function addListener(evt, listener) {
+		var listeners = this.getListenersAsObject(evt);
+		var listenerIsWrapped = typeof listener === 'object';
+		var key;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key) && indexOfListener(listeners[key], listener) === -1) {
+				listeners[key].push(listenerIsWrapped ? listener : {
+					listener: listener,
+					once: false
+				});
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of addListener
+	 */
+	proto.on = alias('addListener');
+
+	/**
+	 * Semi-alias of addListener. It will add a listener that will be
+	 * automatically removed after it's first execution.
+	 *
+	 * @param {String|RegExp} evt Name of the event to attach the listener to.
+	 * @param {Function} listener Method to be called when the event is emitted. If the function returns true then it will be removed after calling.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addOnceListener = function addOnceListener(evt, listener) {
+		return this.addListener(evt, {
+			listener: listener,
+			once: true
+		});
+	};
+
+	/**
+	 * Alias of addOnceListener.
+	 */
+	proto.once = alias('addOnceListener');
+
+	/**
+	 * Defines an event name. This is required if you want to use a regex to add a listener to multiple events at once. If you don't do this then how do you expect it to know what event to add to? Should it just add to every possible match for a regex? No. That is scary and bad.
+	 * You need to tell it what event names should be matched by a regex.
+	 *
+	 * @param {String} evt Name of the event to create.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.defineEvent = function defineEvent(evt) {
+		this.getListeners(evt);
+		return this;
+	};
+
+	/**
+	 * Uses defineEvent to define multiple events.
+	 *
+	 * @param {String[]} evts An array of event names to define.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.defineEvents = function defineEvents(evts) {
+		for (var i = 0; i < evts.length; i += 1) {
+			this.defineEvent(evts[i]);
+		}
+		return this;
+	};
+
+	/**
+	 * Removes a listener function from the specified event.
+	 * When passed a regular expression as the event name, it will remove the listener from all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to remove the listener from.
+	 * @param {Function} listener Method to remove from the event.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeListener = function removeListener(evt, listener) {
+		var listeners = this.getListenersAsObject(evt);
+		var index;
+		var key;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key)) {
+				index = indexOfListener(listeners[key], listener);
+
+				if (index !== -1) {
+					listeners[key].splice(index, 1);
+				}
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of removeListener
+	 */
+	proto.off = alias('removeListener');
+
+	/**
+	 * Adds listeners in bulk using the manipulateListeners method.
+	 * If you pass an object as the second argument you can add to multiple events at once. The object should contain key value pairs of events and listeners or listener arrays. You can also pass it an event name and an array of listeners to be added.
+	 * You can also pass it a regular expression to add the array of listeners to all events that match it.
+	 * Yeah, this function does quite a bit. That's probably a bad thing.
+	 *
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to add to multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to add.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addListeners = function addListeners(evt, listeners) {
+		// Pass through to manipulateListeners
+		return this.manipulateListeners(false, evt, listeners);
+	};
+
+	/**
+	 * Removes listeners in bulk using the manipulateListeners method.
+	 * If you pass an object as the second argument you can remove from multiple events at once. The object should contain key value pairs of events and listeners or listener arrays.
+	 * You can also pass it an event name and an array of listeners to be removed.
+	 * You can also pass it a regular expression to remove the listeners from all events that match it.
+	 *
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to remove from multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to remove.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeListeners = function removeListeners(evt, listeners) {
+		// Pass through to manipulateListeners
+		return this.manipulateListeners(true, evt, listeners);
+	};
+
+	/**
+	 * Edits listeners in bulk. The addListeners and removeListeners methods both use this to do their job. You should really use those instead, this is a little lower level.
+	 * The first argument will determine if the listeners are removed (true) or added (false).
+	 * If you pass an object as the second argument you can add/remove from multiple events at once. The object should contain key value pairs of events and listeners or listener arrays.
+	 * You can also pass it an event name and an array of listeners to be added/removed.
+	 * You can also pass it a regular expression to manipulate the listeners of all events that match it.
+	 *
+	 * @param {Boolean} remove True if you want to remove listeners, false if you want to add.
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to add/remove from multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to add/remove.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.manipulateListeners = function manipulateListeners(remove, evt, listeners) {
+		var i;
+		var value;
+		var single = remove ? this.removeListener : this.addListener;
+		var multiple = remove ? this.removeListeners : this.addListeners;
+
+		// If evt is an object then pass each of it's properties to this method
+		if (typeof evt === 'object' && !(evt instanceof RegExp)) {
+			for (i in evt) {
+				if (evt.hasOwnProperty(i) && (value = evt[i])) {
+					// Pass the single listener straight through to the singular method
+					if (typeof value === 'function') {
+						single.call(this, i, value);
+					}
+					else {
+						// Otherwise pass back to the multiple function
+						multiple.call(this, i, value);
+					}
+				}
+			}
+		}
+		else {
+			// So evt must be a string
+			// And listeners must be an array of listeners
+			// Loop over it and pass each one to the multiple method
+			i = listeners.length;
+			while (i--) {
+				single.call(this, evt, listeners[i]);
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Removes all listeners from a specified event.
+	 * If you do not specify an event then all listeners will be removed.
+	 * That means every event will be emptied.
+	 * You can also pass a regex to remove all events that match it.
+	 *
+	 * @param {String|RegExp} [evt] Optional name of the event to remove all listeners for. Will remove from every event if not passed.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeEvent = function removeEvent(evt) {
+		var type = typeof evt;
+		var events = this._getEvents();
+		var key;
+
+		// Remove different things depending on the state of evt
+		if (type === 'string') {
+			// Remove all listeners for the specified event
+			delete events[evt];
+		}
+		else if (type === 'object') {
+			// Remove all events matching the regex.
+			for (key in events) {
+				if (events.hasOwnProperty(key) && evt.test(key)) {
+					delete events[key];
+				}
+			}
+		}
+		else {
+			// Remove all listeners in all events
+			delete this._events;
+		}
+
+		return this;
+	};
+
+	/**
+	 * Emits an event of your choice.
+	 * When emitted, every listener attached to that event will be executed.
+	 * If you pass the optional argument array then those arguments will be passed to every listener upon execution.
+	 * Because it uses `apply`, your array of arguments will be passed as if you wrote them out separately.
+	 * So they will not arrive within the array on the other side, they will be separate.
+	 * You can also pass a regular expression to emit to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to emit and execute listeners for.
+	 * @param {Array} [args] Optional array of arguments to be passed to each listener.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.emitEvent = function emitEvent(evt, args) {
+		var listeners = this.getListenersAsObject(evt);
+		var listener;
+		var i;
+		var key;
+		var response;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key)) {
+				i = listeners[key].length;
+
+				while (i--) {
+					// If the listener returns true then it shall be removed from the event
+					// The function is executed either with a basic call or an apply if there is an args array
+					listener = listeners[key][i];
+
+					if (listener.once === true) {
+						this.removeListener(evt, listener.listener);
+					}
+
+					response = listener.listener.apply(this, args || []);
+
+					if (response === this._getOnceReturnValue()) {
+						this.removeListener(evt, listener.listener);
+					}
+				}
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of emitEvent
+	 */
+	proto.trigger = alias('emitEvent');
+
+	/**
+	 * Subtly different from emitEvent in that it will pass its arguments on to the listeners, as opposed to taking a single array of arguments to pass on.
+	 * As with emitEvent, you can pass a regex in place of the event name to emit to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to emit and execute listeners for.
+	 * @param {...*} Optional additional arguments to be passed to each listener.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.emit = function emit(evt) {
+		var args = Array.prototype.slice.call(arguments, 1);
+		return this.emitEvent(evt, args);
+	};
+
+	/**
+	 * Sets the current value to check against when executing listeners. If a
+	 * listeners return value matches the one set here then it will be removed
+	 * after execution. This value defaults to true.
+	 *
+	 * @param {*} value The new value to check for when executing listeners.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.setOnceReturnValue = function setOnceReturnValue(value) {
+		this._onceReturnValue = value;
+		return this;
+	};
+
+	/**
+	 * Fetches the current value to check against when executing listeners. If
+	 * the listeners return value matches this one then it should be removed
+	 * automatically. It will return true by default.
+	 *
+	 * @return {*|Boolean} The current value to check for or the default, true.
+	 * @api private
+	 */
+	proto._getOnceReturnValue = function _getOnceReturnValue() {
+		if (this.hasOwnProperty('_onceReturnValue')) {
+			return this._onceReturnValue;
+		}
+		else {
+			return true;
+		}
+	};
+
+	/**
+	 * Fetches the events object and creates one if required.
+	 *
+	 * @return {Object} The events storage object.
+	 * @api private
+	 */
+	proto._getEvents = function _getEvents() {
+		return this._events || (this._events = {});
+	};
+
+	// Expose the class either via AMD, CommonJS or the global object
+	if (typeof define === 'function' && define.amd) {
+		define(function () {
+			return EventEmitter;
+		});
+	}
+	else if (typeof module === 'object' && module.exports){
+		module.exports = EventEmitter;
+	}
+	else {
+		this.EventEmitter = EventEmitter;
+	}
+}.call(this));
+
+/*!
+ * Source: build/lib/otr/build/otr.js, license: MPL v2.0, url: https://arlolra.github.io/otr/ */
+/*!
+
+  otr.js v0.2.13 - 2014-09-07
+  (c) 2014 - Arlo Breault <arlolra at gmail.com>
+  Freely distributed under the MPL v2.0 license.
+
+  This file is concatenated for the browser.
+  Please see: https://github.com/arlolra/otr
+
+*/
+
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define([
+        "bigint"
+      , "crypto"
+      , "eventemitter"
+    ], function (BigInt, CryptoJS, EventEmitter) {
+      var root = {
+          BigInt: BigInt
+        , CryptoJS: CryptoJS
+        , EventEmitter: EventEmitter
+        , OTR: {}
+        , DSA: {}
+      }
+      return factory.call(root)
+    })
+  } else {
+    root.OTR = {}
+    root.DSA = {}
+    factory.call(root)
+  }
+
+}(this, function () {
+
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CONST = {
+
+    // diffie-heilman
+      N : 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7EDEE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB9ED529077096966D670C354E4ABC9804F1746C08CA237327FFFFFFFFFFFFFFFF'
+    , G : '2'
+
+    // otr message states
+    , MSGSTATE_PLAINTEXT : 0
+    , MSGSTATE_ENCRYPTED : 1
+    , MSGSTATE_FINISHED  : 2
+
+    // otr auth states
+    , AUTHSTATE_NONE               : 0
+    , AUTHSTATE_AWAITING_DHKEY     : 1
+    , AUTHSTATE_AWAITING_REVEALSIG : 2
+    , AUTHSTATE_AWAITING_SIG       : 3
+
+    // whitespace tags
+    , WHITESPACE_TAG    : '\x20\x09\x20\x20\x09\x09\x09\x09\x20\x09\x20\x09\x20\x09\x20\x20'
+    , WHITESPACE_TAG_V2 : '\x20\x20\x09\x09\x20\x20\x09\x20'
+    , WHITESPACE_TAG_V3 : '\x20\x20\x09\x09\x20\x20\x09\x09'
+
+    // otr tags
+    , OTR_TAG       : '?OTR'
+    , OTR_VERSION_1 : '\x00\x01'
+    , OTR_VERSION_2 : '\x00\x02'
+    , OTR_VERSION_3 : '\x00\x03'
+
+    // smp machine states
+    , SMPSTATE_EXPECT0 : 0
+    , SMPSTATE_EXPECT1 : 1
+    , SMPSTATE_EXPECT2 : 2
+    , SMPSTATE_EXPECT3 : 3
+    , SMPSTATE_EXPECT4 : 4
+
+    // unstandard status codes
+    , STATUS_SEND_QUERY  : 0
+    , STATUS_AKE_INIT    : 1
+    , STATUS_AKE_SUCCESS : 2
+    , STATUS_END_OTR     : 3
+
+  }
+
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = CONST
+  } else {
+    root.OTR.CONST = CONST
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var HLP = {}, CryptoJS, BigInt
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = HLP = {}
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+  } else {
+    if (root.OTR) root.OTR.HLP = HLP
+    if (root.DSA) root.DSA.HLP = HLP
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+  }
+
+  // data types (byte lengths)
+  var DTS = {
+      BYTE  : 1
+    , SHORT : 2
+    , INT   : 4
+    , CTR   : 8
+    , MAC   : 20
+    , SIG   : 40
+  }
+
+  // otr message wrapper begin and end
+  var WRAPPER_BEGIN = "?OTR"
+    , WRAPPER_END   = "."
+
+  var TWO = BigInt.str2bigInt('2', 10)
+
+  HLP.debug = function (msg) {
+    // used as HLP.debug.call(ctx, msg)
+    if ( this.debug &&
+         typeof this.debug !== 'function' &&
+         typeof console !== 'undefined'
+    ) console.log(msg)
+  }
+
+  HLP.extend = function (child, parent) {
+    for (var key in parent) {
+      if (Object.hasOwnProperty.call(parent, key))
+        child[key] = parent[key]
+    }
+    function Ctor() { this.constructor = child }
+    Ctor.prototype = parent.prototype
+    child.prototype = new Ctor()
+    child.__super__ = parent.prototype
+  }
+
+  // assumes 32-bit
+  function intCompare(x, y) {
+    var z = ~(x ^ y)
+    z &= z >> 16
+    z &= z >> 8
+    z &= z >> 4
+    z &= z >> 2
+    z &= z >> 1
+    return z & 1
+  }
+
+  // constant-time string comparison
+  HLP.compare = function (str1, str2) {
+    if (str1.length !== str2.length)
+      return false
+    var i = 0, result = 0
+    for (; i < str1.length; i++)
+      result |= str1[i].charCodeAt(0) ^ str2[i].charCodeAt(0)
+    return intCompare(result, 0)
+  }
+
+  HLP.randomExponent = function () {
+    return BigInt.randBigInt(1536)
+  }
+
+  HLP.smpHash = function (version, fmpi, smpi) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packBytes(version, DTS.BYTE)))
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(fmpi)))
+    if (smpi) sha256.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(smpi)))
+    var hash = sha256.finalize()
+    return HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+  }
+
+  HLP.makeMac = function (aesctr, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var mac = CryptoJS.HmacSHA256(CryptoJS.enc.Latin1.parse(aesctr), pass)
+    return HLP.mask(mac.toString(CryptoJS.enc.Latin1), 0, 160)
+  }
+
+  HLP.make1Mac = function (aesctr, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var mac = CryptoJS.HmacSHA1(CryptoJS.enc.Latin1.parse(aesctr), pass)
+    return mac.toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.encryptAes = function (msg, c, iv) {
+    var opts = {
+        mode: CryptoJS.mode.CTR
+      , iv: CryptoJS.enc.Latin1.parse(iv)
+      , padding: CryptoJS.pad.NoPadding
+    }
+    var aesctr = CryptoJS.AES.encrypt(
+        msg
+      , CryptoJS.enc.Latin1.parse(c)
+      , opts
+    )
+    var aesctr_decoded = CryptoJS.enc.Base64.parse(aesctr.toString())
+    return CryptoJS.enc.Latin1.stringify(aesctr_decoded)
+  }
+
+  HLP.decryptAes = function (msg, c, iv) {
+    msg = CryptoJS.enc.Latin1.parse(msg)
+    var opts = {
+        mode: CryptoJS.mode.CTR
+      , iv: CryptoJS.enc.Latin1.parse(iv)
+      , padding: CryptoJS.pad.NoPadding
+    }
+    return CryptoJS.AES.decrypt(
+        CryptoJS.enc.Base64.stringify(msg)
+      , CryptoJS.enc.Latin1.parse(c)
+      , opts
+    )
+  }
+
+  HLP.multPowMod = function (a, b, c, d, e) {
+    return BigInt.multMod(BigInt.powMod(a, b, e), BigInt.powMod(c, d, e), e)
+  }
+
+  HLP.ZKP = function (v, c, d, e) {
+    return BigInt.equals(c, HLP.smpHash(v, d, e))
+  }
+
+  // greater than, or equal
+  HLP.GTOE = function (a, b) {
+    return (BigInt.equals(a, b) || BigInt.greater(a, b))
+  }
+
+  HLP.between = function (x, a, b) {
+    return (BigInt.greater(x, a) && BigInt.greater(b, x))
+  }
+
+  HLP.checkGroup = function (g, N_MINUS_2) {
+    return HLP.GTOE(g, TWO) && HLP.GTOE(N_MINUS_2, g)
+  }
+
+  HLP.h1 = function (b, secbytes) {
+    var sha1 = CryptoJS.algo.SHA1.create()
+    sha1.update(CryptoJS.enc.Latin1.parse(b))
+    sha1.update(CryptoJS.enc.Latin1.parse(secbytes))
+    return (sha1.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.h2 = function (b, secbytes) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(b))
+    sha256.update(CryptoJS.enc.Latin1.parse(secbytes))
+    return (sha256.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.mask = function (bytes, start, n) {
+    return bytes.substr(start / 8, n / 8)
+  }
+
+  var _toString = String.fromCharCode;
+  HLP.packBytes = function (val, bytes) {
+    val = val.toString(16)
+    var nex, res = ''  // big-endian, unsigned long
+    for (; bytes > 0; bytes--) {
+      nex = val.length ? val.substr(-2, 2) : '0'
+      val = val.substr(0, val.length - 2)
+      res = _toString(parseInt(nex, 16)) + res
+    }
+    return res
+  }
+
+  HLP.packINT = function (d) {
+    return HLP.packBytes(d, DTS.INT)
+  }
+
+  HLP.packCtr = function (d) {
+    return HLP.padCtr(HLP.packBytes(d, DTS.CTR))
+  }
+
+  HLP.padCtr = function (ctr) {
+    return ctr + '\x00\x00\x00\x00\x00\x00\x00\x00'
+  }
+
+  HLP.unpackCtr = function (d) {
+    d = HLP.toByteArray(d.substring(0, 8))
+    return HLP.unpack(d)
+  }
+
+  HLP.unpack = function (arr) {
+    var val = 0, i = 0, len = arr.length
+    for (; i < len; i++) {
+      val = (val * 256) + arr[i]
+    }
+    return val
+  }
+
+  HLP.packData = function (d) {
+    return HLP.packINT(d.length) + d
+  }
+
+  HLP.bits2bigInt = function (bits) {
+    bits = HLP.toByteArray(bits)
+    return BigInt.ba2bigInt(bits)
+  }
+
+  HLP.packMPI = function (mpi) {
+    return HLP.packData(BigInt.bigInt2bits(BigInt.trim(mpi, 0)))
+  }
+
+  HLP.packSHORT = function (short) {
+    return HLP.packBytes(short, DTS.SHORT)
+  }
+
+  HLP.unpackSHORT = function (short) {
+    short = HLP.toByteArray(short)
+    return HLP.unpack(short)
+  }
+
+  HLP.packTLV = function (type, value) {
+    return HLP.packSHORT(type) + HLP.packSHORT(value.length) + value
+  }
+
+  HLP.readLen = function (msg) {
+    msg = HLP.toByteArray(msg.substring(0, 4))
+    return HLP.unpack(msg)
+  }
+
+  HLP.readData = function (data) {
+    var n = HLP.unpack(data.splice(0, 4))
+    return [n, data]
+  }
+
+  HLP.readMPI = function (data) {
+    data = HLP.toByteArray(data)
+    data = HLP.readData(data)
+    return BigInt.ba2bigInt(data[1])
+  }
+
+  HLP.packMPIs = function (arr) {
+    return arr.reduce(function (prv, cur) {
+      return prv + HLP.packMPI(cur)
+    }, '')
+  }
+
+  HLP.unpackMPIs = function (num, mpis) {
+    var i = 0, arr = []
+    for (; i < num; i++) arr.push('MPI')
+    return (HLP.splitype(arr, mpis)).map(function (m) {
+      return HLP.readMPI(m)
+    })
+  }
+
+  HLP.wrapMsg = function (msg, fs, v3, our_it, their_it) {
+    msg = CryptoJS.enc.Base64.stringify(CryptoJS.enc.Latin1.parse(msg))
+    msg = WRAPPER_BEGIN + ":" + msg + WRAPPER_END
+
+    var its
+    if (v3) {
+      its = '|'
+      its += (HLP.readLen(our_it)).toString(16)
+      its += '|'
+      its += (HLP.readLen(their_it)).toString(16)
+    }
+
+    if (!fs) return [null, msg]
+
+    var n = Math.ceil(msg.length / fs)
+    if (n > 65535) return ['Too many fragments']
+    if (n == 1) return [null, msg]
+
+    var k, bi, ei, frag, mf, mfs = []
+    for (k = 1; k <= n; k++) {
+      bi = (k - 1) * fs
+      ei = k * fs
+      frag = msg.slice(bi, ei)
+      mf = WRAPPER_BEGIN
+      if (v3) mf += its
+      mf += ',' + k + ','
+      mf += n + ','
+      mf += frag + ','
+      mfs.push(mf)
+    }
+
+    return [null, mfs]
+  }
+
+  HLP.splitype = function splitype(arr, msg) {
+    var data = []
+    arr.forEach(function (a) {
+      var str
+      switch (a) {
+        case 'PUBKEY':
+          str = splitype(['SHORT', 'MPI', 'MPI', 'MPI', 'MPI'], msg).join('')
+          break
+        case 'DATA':  // falls through
+        case 'MPI':
+          str = msg.substring(0, HLP.readLen(msg) + 4)
+          break
+        default:
+          str = msg.substring(0, DTS[a])
+      }
+      data.push(str)
+      msg = msg.substring(str.length)
+    })
+    return data
+  }
+
+  // https://github.com/msgpack/msgpack-javascript/blob/master/msgpack.js
+
+  var _bin2num = (function () {
+    var i = 0, _bin2num = {}
+    for (; i < 0x100; ++i) {
+      _bin2num[String.fromCharCode(i)] = i  // "\00" -> 0x00
+    }
+    for (i = 0x80; i < 0x100; ++i) {  // [Webkit][Gecko]
+      _bin2num[String.fromCharCode(0xf700 + i)] = i  // "\f780" -> 0x80
+    }
+    return _bin2num
+  }())
+
+  HLP.toByteArray = function (data) {
+    var rv = []
+      , ary = data.split("")
+      , i = -1
+      , iz = ary.length
+      , remain = iz % 8
+
+    while (remain--) {
+      ++i
+      rv[i] = _bin2num[ary[i]]
+    }
+    remain = iz >> 3
+    while (remain--) {
+      rv.push(_bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]])
+    }
+    return rv
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, Worker, WWPath, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = DSA
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    WWPath = require('path').join(__dirname, '/dsa-webworker.js')
+    HLP = require('./helpers.js')
+  } else {
+    // copy over and expose internals
+    Object.keys(root.DSA).forEach(function (k) {
+      DSA[k] = root.DSA[k]
+    })
+    root.DSA = DSA
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    Worker = root.Worker
+    WWPath = 'dsa-webworker.js'
+    HLP = DSA.HLP
+  }
+
+  var ZERO = BigInt.str2bigInt('0', 10)
+    , ONE = BigInt.str2bigInt('1', 10)
+    , TWO = BigInt.str2bigInt('2', 10)
+    , KEY_TYPE = '\x00\x00'
+
+  var DEBUG = false
+  function timer() {
+    var start = (new Date()).getTime()
+    return function (s) {
+      if (!DEBUG || typeof console === 'undefined') return
+      var t = (new Date()).getTime()
+      console.log(s + ': ' + (t - start))
+      start = t
+    }
+  }
+
+  function makeRandom(min, max) {
+    var c = BigInt.randBigInt(BigInt.bitSize(max))
+    if (!HLP.between(c, min, max)) return makeRandom(min, max)
+    return c
+  }
+
+  // altered BigInt.randProbPrime()
+  // n rounds of Miller Rabin (after trial division with small primes)
+  var rpprb = []
+  function isProbPrime(k, n) {
+    var i, B = 30000, l = BigInt.bitSize(k)
+    var primes = BigInt.primes
+
+    if (primes.length === 0)
+      primes = BigInt.findPrimes(B)
+
+    if (rpprb.length != k.length)
+      rpprb = BigInt.dup(k)
+
+    // check ans for divisibility by small primes up to B
+    for (i = 0; (i < primes.length) && (primes[i] <= B); i++)
+      if (BigInt.modInt(k, primes[i]) === 0 && !BigInt.equalsInt(k, primes[i]))
+        return 0
+
+    // do n rounds of Miller Rabin, with random bases less than k
+    for (i = 0; i < n; i++) {
+      BigInt.randBigInt_(rpprb, l, 0)
+      while(!BigInt.greater(k, rpprb))  // pick a random rpprb that's < k
+        BigInt.randBigInt_(rpprb, l, 0)
+      if (!BigInt.millerRabin(k, rpprb))
+        return 0
+    }
+
+    return 1
+  }
+
+  var bit_lengths = {
+      '1024': { N: 160, repeat: 40 }  // 40x should give 2^-80 confidence
+    , '2048': { N: 224, repeat: 56 }
+  }
+
+  var primes = {}
+
+  // follows go lang http://golang.org/src/pkg/crypto/dsa/dsa.go
+  // fips version was removed in 0c99af0df3e7
+  function generatePrimes(bit_length) {
+
+    var t = timer()  // for debugging
+
+    // number of MR tests to perform
+    var repeat = bit_lengths[bit_length].repeat
+
+    var N = bit_lengths[bit_length].N
+
+    var LM1 = BigInt.twoToThe(bit_length - 1)
+    var bl4 = 4 * bit_length
+    var brk = false
+
+    var q, p, rem, counter
+    for (;;) {
+
+      q = BigInt.randBigInt(N, 1)
+      q[0] |= 1
+
+      if (!isProbPrime(q, repeat)) continue
+      t('q')
+
+      for (counter = 0; counter < bl4; counter++) {
+        p = BigInt.randBigInt(bit_length, 1)
+        p[0] |= 1
+
+        rem = BigInt.mod(p, q)
+        rem = BigInt.sub(rem, ONE)
+        p = BigInt.sub(p, rem)
+
+        if (BigInt.greater(LM1, p)) continue
+        if (!isProbPrime(p, repeat)) continue
+
+        t('p')
+        primes[bit_length] = { p: p, q: q }
+        brk = true
+        break
+      }
+
+      if (brk) break
+    }
+
+    var h = BigInt.dup(TWO)
+    var pm1 = BigInt.sub(p, ONE)
+    var e = BigInt.multMod(pm1, BigInt.inverseMod(q, p), p)
+
+    var g
+    for (;;) {
+      g = BigInt.powMod(h, e, p)
+      if (BigInt.equals(g, ONE)) {
+        h = BigInt.add(h, ONE)
+        continue
+      }
+      primes[bit_length].g = g
+      t('g')
+      return
+    }
+
+    throw new Error('Unreachable!')
+  }
+
+  function DSA(obj, opts) {
+    if (!(this instanceof DSA)) return new DSA(obj, opts)
+
+    // options
+    opts = opts || {}
+
+    // inherit
+    if (obj) {
+      var self = this
+      ;['p', 'q', 'g', 'y', 'x'].forEach(function (prop) {
+        self[prop] = obj[prop]
+      })
+      this.type = obj.type || KEY_TYPE
+      return
+    }
+
+    // default to 1024
+    var bit_length = parseInt(opts.bit_length ? opts.bit_length : 1024, 10)
+
+    if (!bit_lengths[bit_length])
+      throw new Error('Unsupported bit length.')
+
+    // set primes
+    if (!primes[bit_length])
+      generatePrimes(bit_length)
+
+    this.p = primes[bit_length].p
+    this.q = primes[bit_length].q
+    this.g = primes[bit_length].g
+
+    // key type
+    this.type = KEY_TYPE
+
+    // private key
+    this.x = makeRandom(ZERO, this.q)
+
+    // public keys (p, q, g, y)
+    this.y = BigInt.powMod(this.g, this.x, this.p)
+
+    // nocache?
+    if (opts.nocache) primes[bit_length] = null
+  }
+
+  DSA.prototype = {
+
+    constructor: DSA,
+
+    packPublic: function () {
+      var str = this.type
+      str += HLP.packMPI(this.p)
+      str += HLP.packMPI(this.q)
+      str += HLP.packMPI(this.g)
+      str += HLP.packMPI(this.y)
+      return str
+    },
+
+    packPrivate: function () {
+      var str = this.packPublic() + HLP.packMPI(this.x)
+      str = CryptoJS.enc.Latin1.parse(str)
+      return str.toString(CryptoJS.enc.Base64)
+    },
+
+    // http://www.imperialviolet.org/2013/06/15/suddendeathentropy.html
+    generateNonce: function (m) {
+      var priv = BigInt.bigInt2bits(BigInt.trim(this.x, 0))
+      var rand = BigInt.bigInt2bits(BigInt.randBigInt(256))
+
+      var sha256 = CryptoJS.algo.SHA256.create()
+      sha256.update(CryptoJS.enc.Latin1.parse(priv))
+      sha256.update(m)
+      sha256.update(CryptoJS.enc.Latin1.parse(rand))
+
+      var hash = sha256.finalize()
+      hash = HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+      BigInt.rightShift_(hash, 256 - BigInt.bitSize(this.q))
+
+      return HLP.between(hash, ZERO, this.q) ? hash : this.generateNonce(m)
+    },
+
+    sign: function (m) {
+      m = CryptoJS.enc.Latin1.parse(m)
+      var b = BigInt.str2bigInt(m.toString(CryptoJS.enc.Hex), 16)
+      var k, r = ZERO, s = ZERO
+      while (BigInt.isZero(s) || BigInt.isZero(r)) {
+        k = this.generateNonce(m)
+        r = BigInt.mod(BigInt.powMod(this.g, k, this.p), this.q)
+        if (BigInt.isZero(r)) continue
+        s = BigInt.inverseMod(k, this.q)
+        s = BigInt.mult(s, BigInt.add(b, BigInt.mult(this.x, r)))
+        s = BigInt.mod(s, this.q)
+      }
+      return [r, s]
+    },
+
+    fingerprint: function () {
+      var pk = this.packPublic()
+      if (this.type === KEY_TYPE) pk = pk.substring(2)
+      pk = CryptoJS.enc.Latin1.parse(pk)
+      return CryptoJS.SHA1(pk).toString(CryptoJS.enc.Hex)
+    }
+
+  }
+
+  DSA.parsePublic = function (str, priv) {
+    var fields = ['SHORT', 'MPI', 'MPI', 'MPI', 'MPI']
+    if (priv) fields.push('MPI')
+    str = HLP.splitype(fields, str)
+    var obj = {
+        type: str[0]
+      , p: HLP.readMPI(str[1])
+      , q: HLP.readMPI(str[2])
+      , g: HLP.readMPI(str[3])
+      , y: HLP.readMPI(str[4])
+    }
+    if (priv) obj.x = HLP.readMPI(str[5])
+    return new DSA(obj)
+  }
+
+  function tokenizeStr(str) {
+    var start, end
+
+    start = str.indexOf("(")
+    end = str.lastIndexOf(")")
+
+    if (start < 0 || end < 0)
+      throw new Error("Malformed S-Expression")
+
+    str = str.substring(start + 1, end)
+
+    var splt = str.search(/\s/)
+    var obj = {
+        type: str.substring(0, splt)
+      , val: []
+    }
+
+    str = str.substring(splt + 1, end)
+    start = str.indexOf("(")
+
+    if (start < 0) obj.val.push(str)
+    else {
+
+      var i, len, ss, es
+      while (start > -1) {
+        i = start + 1
+        len = str.length
+        for (ss = 1, es = 0; i < len && es < ss; i++) {
+          if (str[i] === "(") ss++
+          if (str[i] === ")") es++
+        }
+        obj.val.push(tokenizeStr(str.substring(start, ++i)))
+        str = str.substring(++i)
+        start = str.indexOf("(")
+      }
+
+    }
+    return obj
+  }
+
+  function parseLibotr(obj) {
+    if (!obj.type) throw new Error("Parse error.")
+
+    var o, val
+    if (obj.type === "privkeys") {
+      o = []
+      obj.val.forEach(function (i) {
+        o.push(parseLibotr(i))
+      })
+      return o
+    }
+
+    o = {}
+    obj.val.forEach(function (i) {
+
+      val = i.val[0]
+      if (typeof val === "string") {
+
+        if (val.indexOf("#") === 0) {
+          val = val.substring(1, val.lastIndexOf("#"))
+          val = BigInt.str2bigInt(val, 16)
+        }
+
+      } else {
+        val = parseLibotr(i)
+      }
+
+      o[i.type] = val
+    })
+
+    return o
+  }
+
+  DSA.parsePrivate = function (str, libotr) {
+    if (!libotr) {
+      str = CryptoJS.enc.Base64.parse(str)
+      str = str.toString(CryptoJS.enc.Latin1)
+      return DSA.parsePublic(str, true)
+    }
+    // only returning the first key found
+    return parseLibotr(tokenizeStr(str))[0]["private-key"].dsa
+  }
+
+  DSA.verify = function (key, m, r, s) {
+    if (!HLP.between(r, ZERO, key.q) || !HLP.between(s, ZERO, key.q))
+      return false
+
+    var hm = CryptoJS.enc.Latin1.parse(m)  // CryptoJS.SHA1(m)
+    hm = BigInt.str2bigInt(hm.toString(CryptoJS.enc.Hex), 16)
+
+    var w = BigInt.inverseMod(s, key.q)
+    var u1 = BigInt.multMod(hm, w, key.q)
+    var u2 = BigInt.multMod(r, w, key.q)
+
+    u1 = BigInt.powMod(key.g, u1, key.p)
+    u2 = BigInt.powMod(key.y, u2, key.p)
+
+    var v = BigInt.mod(BigInt.multMod(u1, u2, key.p), key.q)
+
+    return BigInt.equals(v, r)
+  }
+
+  DSA.createInWebWorker = function (options, cb) {
+    var opts = {
+        path: WWPath
+      , seed: BigInt.getSeed
+    }
+    if (options && typeof options === 'object')
+      Object.keys(options).forEach(function (k) {
+        opts[k] = options[k]
+      })
+
+    // load optional dep. in node
+    if (typeof module !== 'undefined' && module.exports)
+      Worker = require('webworker-threads').Worker
+
+    var worker = new Worker(opts.path)
+    worker.onmessage = function (e) {
+      var data = e.data
+      switch (data.type) {
+        case "debug":
+          if (!DEBUG || typeof console === 'undefined') return
+          console.log(data.val)
+          break;
+        case "data":
+          worker.terminate()
+          cb(DSA.parsePrivate(data.val))
+          break;
+        default:
+          throw new Error("Unrecognized type.")
+      }
+    }
+    worker.postMessage({
+        seed: opts.seed()
+      , imports: opts.imports
+      , debug: DEBUG
+    })
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var Parse = {}, CryptoJS, CONST, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = Parse
+    CryptoJS = require('../vendor/crypto.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+  } else {
+    root.OTR.Parse = Parse
+    CryptoJS = root.CryptoJS
+    CONST = root.OTR.CONST
+    HLP = root.OTR.HLP
+  }
+
+  // whitespace tags
+  var tags = {}
+  tags[CONST.WHITESPACE_TAG_V2] = CONST.OTR_VERSION_2
+  tags[CONST.WHITESPACE_TAG_V3] = CONST.OTR_VERSION_3
+
+  Parse.parseMsg = function (otr, msg) {
+
+    var ver = []
+
+    // is this otr?
+    var start = msg.indexOf(CONST.OTR_TAG)
+    if (!~start) {
+
+      // restart fragments
+      this.initFragment(otr)
+
+      // whitespace tags
+      ind = msg.indexOf(CONST.WHITESPACE_TAG)
+
+      if (~ind) {
+
+        msg = msg.split('')
+        msg.splice(ind, 16)
+
+        var tag, len = msg.length
+        for (; ind < len;) {
+          tag = msg.slice(ind, ind + 8).join('')
+          if (Object.hasOwnProperty.call(tags, tag)) {
+            msg.splice(ind, 8)
+            ver.push(tags[tag])
+            continue
+          }
+          ind += 8
+        }
+
+        msg = msg.join('')
+
+      }
+
+      return { msg: msg, ver: ver }
+    }
+
+    var ind = start + CONST.OTR_TAG.length
+    var com = msg[ind]
+
+    // message fragment
+    if (com === ',' || com === '|') {
+      return this.msgFragment(otr, msg.substring(ind + 1), (com === '|'))
+    }
+
+    this.initFragment(otr)
+
+    // query message
+    if (~['?', 'v'].indexOf(com)) {
+
+      // version 1
+      if (msg[ind] === '?') {
+        ver.push(CONST.OTR_VERSION_1)
+        ind += 1
+      }
+
+      // other versions
+      var vers = {
+          '2': CONST.OTR_VERSION_2
+        , '3': CONST.OTR_VERSION_3
+      }
+      var qs = msg.substring(ind + 1)
+      var qi = qs.indexOf('?')
+
+      if (qi >= 1) {
+        qs = qs.substring(0, qi).split('')
+        if (msg[ind] === 'v') {
+          qs.forEach(function (q) {
+            if (Object.hasOwnProperty.call(vers, q)) ver.push(vers[q])
+          })
+        }
+      }
+
+      return { cls: 'query', ver: ver }
+    }
+
+    // otr message
+    if (com === ':') {
+
+      ind += 1
+
+      var info = msg.substring(ind, ind + 4)
+      if (info.length < 4) return { msg: msg }
+      info = CryptoJS.enc.Base64.parse(info).toString(CryptoJS.enc.Latin1)
+
+      var version = info.substring(0, 2)
+      var type = info.substring(2)
+
+      // supporting otr versions 2 and 3
+      if (!otr['ALLOW_V' + HLP.unpackSHORT(version)]) return { msg: msg }
+
+      ind += 4
+
+      var end = msg.substring(ind).indexOf('.')
+      if (!~end) return { msg: msg }
+
+      msg = CryptoJS.enc.Base64.parse(msg.substring(ind, ind + end))
+      msg = CryptoJS.enc.Latin1.stringify(msg)
+
+      // instance tags
+      var instance_tags
+      if (version === CONST.OTR_VERSION_3) {
+        instance_tags = msg.substring(0, 8)
+        msg = msg.substring(8)
+      }
+
+      var cls
+      if (~['\x02', '\x0a', '\x11', '\x12'].indexOf(type)) {
+        cls = 'ake'
+      } else if (type === '\x03') {
+        cls = 'data'
+      }
+
+      return {
+          version: version
+        , type: type
+        , msg: msg
+        , cls: cls
+        , instance_tags: instance_tags
+      }
+    }
+
+    // error message
+    if (msg.substring(ind, ind + 7) === ' Error:') {
+      if (otr.ERROR_START_AKE) {
+        otr.sendQueryMsg()
+      }
+      return { msg: msg.substring(ind + 7), cls: 'error' }
+    }
+
+    return { msg: msg }
+  }
+
+  Parse.initFragment = function (otr) {
+    otr.fragment = { s: '', j: 0, k: 0 }
+  }
+
+  Parse.msgFragment = function (otr, msg, v3) {
+
+    msg = msg.split(',')
+
+    // instance tags
+    if (v3) {
+      var its = msg.shift().split('|')
+      var their_it = HLP.packINT(parseInt(its[0], 16))
+      var our_it = HLP.packINT(parseInt(its[1], 16))
+      if (otr.checkInstanceTags(their_it + our_it)) return  // ignore
+    }
+
+    if (msg.length < 4 ||
+      isNaN(parseInt(msg[0], 10)) ||
+      isNaN(parseInt(msg[1], 10))
+    ) return
+
+    var k = parseInt(msg[0], 10)
+    var n = parseInt(msg[1], 10)
+    msg = msg[2]
+
+    if (n < k || n === 0 || k === 0) {
+      this.initFragment(otr)
+      return
+    }
+
+    if (k === 1) {
+      this.initFragment(otr)
+      otr.fragment = { k: 1, n: n, s: msg }
+    } else if (n === otr.fragment.n && k === (otr.fragment.k + 1)) {
+      otr.fragment.s += msg
+      otr.fragment.k += 1
+    } else {
+      this.initFragment(otr)
+    }
+
+    if (n === k) {
+      msg = otr.fragment.s
+      this.initFragment(otr)
+      return this.parseMsg(otr, msg)
+    }
+
+    return
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, CONST, HLP, DSA
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = AKE
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+    DSA = require('./dsa.js')
+  } else {
+    root.OTR.AKE = AKE
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    CONST = root.OTR.CONST
+    HLP = root.OTR.HLP
+    DSA = root.DSA
+  }
+
+  // diffie-hellman modulus
+  // see group 5, RFC 3526
+  var N = BigInt.str2bigInt(CONST.N, 16)
+  var N_MINUS_2 = BigInt.sub(N, BigInt.str2bigInt('2', 10))
+
+  function hMac(gx, gy, pk, kid, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var hmac = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, pass)
+    hmac.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(gx)))
+    hmac.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(gy)))
+    hmac.update(CryptoJS.enc.Latin1.parse(pk))
+    hmac.update(CryptoJS.enc.Latin1.parse(kid))
+    return (hmac.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  // AKE constructor
+  function AKE(otr) {
+    if (!(this instanceof AKE)) return new AKE(otr)
+
+    // otr instance
+    this.otr = otr
+
+    // our keys
+    this.our_dh = otr.our_old_dh
+    this.our_keyid = otr.our_keyid - 1
+
+    // their keys
+    this.their_y = null
+    this.their_keyid = null
+    this.their_priv_pk = null
+
+    // state
+    this.ssid = null
+    this.transmittedRS = false
+    this.r = null
+
+    // bind methods
+    var self = this
+    ;['sendMsg'].forEach(function (meth) {
+      self[meth] = self[meth].bind(self)
+    })
+  }
+
+  AKE.prototype = {
+
+    constructor: AKE,
+
+    createKeys: function(g) {
+      var s = BigInt.powMod(g, this.our_dh.privateKey, N)
+      var secbytes = HLP.packMPI(s)
+      this.ssid = HLP.mask(HLP.h2('\x00', secbytes), 0, 64)  // first 64-bits
+      var tmp = HLP.h2('\x01', secbytes)
+      this.c = HLP.mask(tmp, 0, 128)  // first 128-bits
+      this.c_prime = HLP.mask(tmp, 128, 128)  // second 128-bits
+      this.m1 = HLP.h2('\x02', secbytes)
+      this.m2 = HLP.h2('\x03', secbytes)
+      this.m1_prime = HLP.h2('\x04', secbytes)
+      this.m2_prime = HLP.h2('\x05', secbytes)
+    },
+
+    verifySignMac: function (mac, aesctr, m2, c, their_y, our_dh_pk, m1, ctr) {
+      // verify mac
+      var vmac = HLP.makeMac(aesctr, m2)
+      if (!HLP.compare(mac, vmac))
+        return ['MACs do not match.']
+
+      // decrypt x
+      var x = HLP.decryptAes(aesctr.substring(4), c, ctr)
+      x = HLP.splitype(['PUBKEY', 'INT', 'SIG'], x.toString(CryptoJS.enc.Latin1))
+
+      var m = hMac(their_y, our_dh_pk, x[0], x[1], m1)
+      var pub = DSA.parsePublic(x[0])
+
+      var r = HLP.bits2bigInt(x[2].substring(0, 20))
+      var s = HLP.bits2bigInt(x[2].substring(20))
+
+      // verify sign m
+      if (!DSA.verify(pub, m, r, s)) return ['Cannot verify signature of m.']
+
+      return [null, HLP.readLen(x[1]), pub]
+    },
+
+    makeM: function (their_y, m1, c, m2) {
+      var pk = this.otr.priv.packPublic()
+      var kid = HLP.packINT(this.our_keyid)
+      var m = hMac(this.our_dh.publicKey, their_y, pk, kid, m1)
+      m = this.otr.priv.sign(m)
+      var msg = pk + kid
+      msg += BigInt.bigInt2bits(m[0], 20)  // pad to 20 bytes
+      msg += BigInt.bigInt2bits(m[1], 20)
+      msg = CryptoJS.enc.Latin1.parse(msg)
+      var aesctr = HLP.packData(HLP.encryptAes(msg, c, HLP.packCtr(0)))
+      var mac = HLP.makeMac(aesctr, m2)
+      return aesctr + mac
+    },
+
+    akeSuccess: function (version) {
+      HLP.debug.call(this.otr, 'success')
+
+      if (BigInt.equals(this.their_y, this.our_dh.publicKey))
+        return this.otr.error('equal keys - we have a problem.')
+
+      this.otr.our_old_dh = this.our_dh
+      this.otr.their_priv_pk = this.their_priv_pk
+
+      if (!(
+        (this.their_keyid === this.otr.their_keyid &&
+         BigInt.equals(this.their_y, this.otr.their_y)) ||
+        (this.their_keyid === (this.otr.their_keyid - 1) &&
+         BigInt.equals(this.their_y, this.otr.their_old_y))
+      )) {
+
+        this.otr.their_y = this.their_y
+        this.otr.their_old_y = null
+        this.otr.their_keyid = this.their_keyid
+
+        // rotate keys
+        this.otr.sessKeys[0] = [ new this.otr.DHSession(
+            this.otr.our_dh
+          , this.otr.their_y
+        ), null ]
+        this.otr.sessKeys[1] = [ new this.otr.DHSession(
+            this.otr.our_old_dh
+          , this.otr.their_y
+        ), null ]
+
+      }
+
+      // ake info
+      this.otr.ssid = this.ssid
+      this.otr.transmittedRS = this.transmittedRS
+      this.otr_version = version
+
+      // go encrypted
+      this.otr.authstate = CONST.AUTHSTATE_NONE
+      this.otr.msgstate = CONST.MSGSTATE_ENCRYPTED
+
+      // null out values
+      this.r = null
+      this.myhashed = null
+      this.dhcommit = null
+      this.encrypted = null
+      this.hashed = null
+
+      this.otr.trigger('status', [CONST.STATUS_AKE_SUCCESS])
+
+      // send stored msgs
+      this.otr.sendStored()
+    },
+
+    handleAKE: function (msg) {
+      var send, vsm, type
+      var version = msg.version
+
+      switch (msg.type) {
+
+        case '\x02':
+          HLP.debug.call(this.otr, 'd-h key message')
+
+          msg = HLP.splitype(['DATA', 'DATA'], msg.msg)
+
+          if (this.otr.authstate === CONST.AUTHSTATE_AWAITING_DHKEY) {
+            var ourHash = HLP.readMPI(this.myhashed)
+            var theirHash = HLP.readMPI(msg[1])
+            if (BigInt.greater(ourHash, theirHash)) {
+              type = '\x02'
+              send = this.dhcommit
+              break  // ignore
+            } else {
+              // forget
+              this.our_dh = this.otr.dh()
+              this.otr.authstate = CONST.AUTHSTATE_NONE
+              this.r = null
+              this.myhashed = null
+            }
+          } else if (
+            this.otr.authstate === CONST.AUTHSTATE_AWAITING_SIG
+          ) this.our_dh = this.otr.dh()
+
+          this.otr.authstate = CONST.AUTHSTATE_AWAITING_REVEALSIG
+
+          this.encrypted = msg[0].substring(4)
+          this.hashed = msg[1].substring(4)
+
+          type = '\x0a'
+          send = HLP.packMPI(this.our_dh.publicKey)
+          break
+
+        case '\x0a':
+          HLP.debug.call(this.otr, 'reveal signature message')
+
+          msg = HLP.splitype(['MPI'], msg.msg)
+
+          if (this.otr.authstate !== CONST.AUTHSTATE_AWAITING_DHKEY) {
+            if (this.otr.authstate === CONST.AUTHSTATE_AWAITING_SIG) {
+              if (!BigInt.equals(this.their_y, HLP.readMPI(msg[0]))) return
+            } else {
+              return  // ignore
+            }
+          }
+
+          this.otr.authstate = CONST.AUTHSTATE_AWAITING_SIG
+
+          this.their_y = HLP.readMPI(msg[0])
+
+          // verify gy is legal 2 <= gy <= N-2
+          if (!HLP.checkGroup(this.their_y, N_MINUS_2))
+            return this.otr.error('Illegal g^y.')
+
+          this.createKeys(this.their_y)
+
+          type = '\x11'
+          send = HLP.packMPI(this.r)
+          send += this.makeM(this.their_y, this.m1, this.c, this.m2)
+
+          this.m1 = null
+          this.m2 = null
+          this.c = null
+          break
+
+        case '\x11':
+          HLP.debug.call(this.otr, 'signature message')
+
+          if (this.otr.authstate !== CONST.AUTHSTATE_AWAITING_REVEALSIG)
+            return  // ignore
+
+          msg = HLP.splitype(['DATA', 'DATA', 'MAC'], msg.msg)
+
+          this.r = HLP.readMPI(msg[0])
+
+          // decrypt their_y
+          var key = CryptoJS.enc.Hex.parse(BigInt.bigInt2str(this.r, 16))
+          key = CryptoJS.enc.Latin1.stringify(key)
+
+          var gxmpi = HLP.decryptAes(this.encrypted, key, HLP.packCtr(0))
+          gxmpi = gxmpi.toString(CryptoJS.enc.Latin1)
+
+          this.their_y = HLP.readMPI(gxmpi)
+
+          // verify hash
+          var hash = CryptoJS.SHA256(CryptoJS.enc.Latin1.parse(gxmpi))
+
+          if (!HLP.compare(this.hashed, hash.toString(CryptoJS.enc.Latin1)))
+            return this.otr.error('Hashed g^x does not match.')
+
+          // verify gx is legal 2 <= g^x <= N-2
+          if (!HLP.checkGroup(this.their_y, N_MINUS_2))
+            return this.otr.error('Illegal g^x.')
+
+          this.createKeys(this.their_y)
+
+          vsm = this.verifySignMac(
+              msg[2]
+            , msg[1]
+            , this.m2
+            , this.c
+            , this.their_y
+            , this.our_dh.publicKey
+            , this.m1
+            , HLP.packCtr(0)
+          )
+          if (vsm[0]) return this.otr.error(vsm[0])
+
+          // store their key
+          this.their_keyid = vsm[1]
+          this.their_priv_pk = vsm[2]
+
+          send = this.makeM(
+              this.their_y
+            , this.m1_prime
+            , this.c_prime
+            , this.m2_prime
+          )
+
+          this.m1 = null
+          this.m2 = null
+          this.m1_prime = null
+          this.m2_prime = null
+          this.c = null
+          this.c_prime = null
+
+          this.sendMsg(version, '\x12', send)
+          this.akeSuccess(version)
+          return
+
+        case '\x12':
+          HLP.debug.call(this.otr, 'data message')
+
+          if (this.otr.authstate !== CONST.AUTHSTATE_AWAITING_SIG)
+            return  // ignore
+
+          msg = HLP.splitype(['DATA', 'MAC'], msg.msg)
+
+          vsm = this.verifySignMac(
+              msg[1]
+            , msg[0]
+            , this.m2_prime
+            , this.c_prime
+            , this.their_y
+            , this.our_dh.publicKey
+            , this.m1_prime
+            , HLP.packCtr(0)
+          )
+          if (vsm[0]) return this.otr.error(vsm[0])
+
+          // store their key
+          this.their_keyid = vsm[1]
+          this.their_priv_pk = vsm[2]
+
+          this.m1_prime = null
+          this.m2_prime = null
+          this.c_prime = null
+
+          this.transmittedRS = true
+          this.akeSuccess(version)
+          return
+
+        default:
+          return  // ignore
+
+      }
+
+      this.sendMsg(version, type, send)
+    },
+
+    sendMsg: function (version, type, msg) {
+      var send = version + type
+      var v3 = (version === CONST.OTR_VERSION_3)
+
+      // instance tags for v3
+      if (v3) {
+        HLP.debug.call(this.otr, 'instance tags')
+        send += this.otr.our_instance_tag
+        send += this.otr.their_instance_tag
+      }
+
+      send += msg
+
+      // fragment message if necessary
+      send = HLP.wrapMsg(
+          send
+        , this.otr.fragment_size
+        , v3
+        , this.otr.our_instance_tag
+        , this.otr.their_instance_tag
+      )
+      if (send[0]) return this.otr.error(send[0])
+
+      this.otr.io(send[1])
+    },
+
+    initiateAKE: function (version) {
+      HLP.debug.call(this.otr, 'd-h commit message')
+
+      this.otr.trigger('status', [CONST.STATUS_AKE_INIT])
+
+      this.otr.authstate = CONST.AUTHSTATE_AWAITING_DHKEY
+
+      var gxmpi = HLP.packMPI(this.our_dh.publicKey)
+      gxmpi = CryptoJS.enc.Latin1.parse(gxmpi)
+
+      this.r = BigInt.randBigInt(128)
+      var key = CryptoJS.enc.Hex.parse(BigInt.bigInt2str(this.r, 16))
+      key = CryptoJS.enc.Latin1.stringify(key)
+
+      this.myhashed = CryptoJS.SHA256(gxmpi)
+      this.myhashed = HLP.packData(this.myhashed.toString(CryptoJS.enc.Latin1))
+
+      this.dhcommit = HLP.packData(HLP.encryptAes(gxmpi, key, HLP.packCtr(0)))
+      this.dhcommit += this.myhashed
+
+      this.sendMsg(version, '\x02', this.dhcommit)
+    }
+
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt,  EventEmitter, CONST, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = SM
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    EventEmitter = require('../vendor/eventemitter.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+  } else {
+    root.OTR.SM = SM
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    EventEmitter = root.EventEmitter
+    CONST = root.OTR.CONST
+    HLP = root.OTR.HLP
+  }
+
+  // diffie-hellman modulus and generator
+  // see group 5, RFC 3526
+  var G = BigInt.str2bigInt(CONST.G, 10)
+  var N = BigInt.str2bigInt(CONST.N, 16)
+  var N_MINUS_2 = BigInt.sub(N, BigInt.str2bigInt('2', 10))
+
+  // to calculate D's for zero-knowledge proofs
+  var Q = BigInt.sub(N, BigInt.str2bigInt('1', 10))
+  BigInt.divInt_(Q, 2)  // meh
+
+  function SM(reqs) {
+    if (!(this instanceof SM)) return new SM(reqs)
+
+    this.version = 1
+
+    this.our_fp = reqs.our_fp
+    this.their_fp = reqs.their_fp
+    this.ssid = reqs.ssid
+
+    this.debug = !!reqs.debug
+
+    // initial state
+    this.init()
+  }
+
+  // inherit from EE
+  HLP.extend(SM, EventEmitter)
+
+  // set the initial values
+  // also used when aborting
+  SM.prototype.init = function () {
+    this.smpstate = CONST.SMPSTATE_EXPECT1
+    this.secret = null
+  }
+
+  SM.prototype.makeSecret = function (our, secret) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packBytes(this.version, 1)))
+    sha256.update(CryptoJS.enc.Hex.parse(our ? this.our_fp : this.their_fp))
+    sha256.update(CryptoJS.enc.Hex.parse(our ? this.their_fp : this.our_fp))
+    sha256.update(CryptoJS.enc.Latin1.parse(this.ssid))
+    sha256.update(CryptoJS.enc.Latin1.parse(secret))
+    var hash = sha256.finalize()
+    this.secret = HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+  }
+
+  SM.prototype.makeG2s = function () {
+    this.a2 = HLP.randomExponent()
+    this.a3 = HLP.randomExponent()
+    this.g2a = BigInt.powMod(G, this.a2, N)
+    this.g3a = BigInt.powMod(G, this.a3, N)
+    if ( !HLP.checkGroup(this.g2a, N_MINUS_2) ||
+         !HLP.checkGroup(this.g3a, N_MINUS_2)
+    ) this.makeG2s()
+  }
+
+  SM.prototype.computeGs = function (g2a, g3a) {
+    this.g2 = BigInt.powMod(g2a, this.a2, N)
+    this.g3 = BigInt.powMod(g3a, this.a3, N)
+  }
+
+  SM.prototype.computePQ = function (r) {
+    this.p = BigInt.powMod(this.g3, r, N)
+    this.q = HLP.multPowMod(G, r, this.g2, this.secret, N)
+  }
+
+  SM.prototype.computeR = function () {
+    this.r = BigInt.powMod(this.QoQ, this.a3, N)
+  }
+
+  SM.prototype.computeRab = function (r) {
+    return BigInt.powMod(r, this.a3, N)
+  }
+
+  SM.prototype.computeC = function (v, r) {
+    return HLP.smpHash(v, BigInt.powMod(G, r, N))
+  }
+
+  SM.prototype.computeD = function (r, a, c) {
+    return BigInt.subMod(r, BigInt.multMod(a, c, Q), Q)
+  }
+
+  // the bulk of the work
+  SM.prototype.handleSM = function (msg) {
+    var send, r2, r3, r7, t1, t2, t3, t4, rab, tmp2, cR, d7, ms, trust
+
+    var expectStates = {
+        2: CONST.SMPSTATE_EXPECT1
+      , 3: CONST.SMPSTATE_EXPECT2
+      , 4: CONST.SMPSTATE_EXPECT3
+      , 5: CONST.SMPSTATE_EXPECT4
+      , 7: CONST.SMPSTATE_EXPECT1
+    }
+
+    if (msg.type === 6) {
+      this.init()
+      this.trigger('abort')
+      return
+    }
+
+    // abort! there was an error
+    if (this.smpstate !== expectStates[msg.type])
+      return this.abort()
+
+    switch (this.smpstate) {
+
+      case CONST.SMPSTATE_EXPECT1:
+        HLP.debug.call(this, 'smp tlv 2')
+
+        // user specified question
+        var ind, question
+        if (msg.type === 7) {
+          ind = msg.msg.indexOf('\x00')
+          question = msg.msg.substring(0, ind)
+          msg.msg = msg.msg.substring(ind + 1)
+        }
+
+        // 0:g2a, 1:c2, 2:d2, 3:g3a, 4:c3, 5:d3
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 6) return this.abort()
+        msg = HLP.unpackMPIs(6, msg.msg.substring(4))
+
+        if ( !HLP.checkGroup(msg[0], N_MINUS_2) ||
+             !HLP.checkGroup(msg[3], N_MINUS_2)
+        ) return this.abort()
+
+        // verify znp's
+        if (!HLP.ZKP(1, msg[1], HLP.multPowMod(G, msg[2], msg[0], msg[1], N)))
+          return this.abort()
+
+        if (!HLP.ZKP(2, msg[4], HLP.multPowMod(G, msg[5], msg[3], msg[4], N)))
+          return this.abort()
+
+        this.g3ao = msg[3]  // save for later
+
+        this.makeG2s()
+
+        // zero-knowledge proof that the exponents
+        // associated with g2a & g3a are known
+        r2 = HLP.randomExponent()
+        r3 = HLP.randomExponent()
+        this.c2 = this.computeC(3, r2)
+        this.c3 = this.computeC(4, r3)
+        this.d2 = this.computeD(r2, this.a2, this.c2)
+        this.d3 = this.computeD(r3, this.a3, this.c3)
+
+        this.computeGs(msg[0], msg[3])
+
+        this.smpstate = CONST.SMPSTATE_EXPECT0
+
+        // assume utf8 question
+        question = CryptoJS.enc.Latin1
+          .parse(question)
+          .toString(CryptoJS.enc.Utf8)
+
+        // invoke question
+        this.trigger('question', [question])
+        return
+
+      case CONST.SMPSTATE_EXPECT2:
+        HLP.debug.call(this, 'smp tlv 3')
+
+        // 0:g2a, 1:c2, 2:d2, 3:g3a, 4:c3, 5:d3, 6:p, 7:q, 8:cP, 9:d5, 10:d6
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 11) return this.abort()
+        msg = HLP.unpackMPIs(11, msg.msg.substring(4))
+
+        if ( !HLP.checkGroup(msg[0], N_MINUS_2) ||
+             !HLP.checkGroup(msg[3], N_MINUS_2) ||
+             !HLP.checkGroup(msg[6], N_MINUS_2) ||
+             !HLP.checkGroup(msg[7], N_MINUS_2)
+        ) return this.abort()
+
+        // verify znp of c3 / c3
+        if (!HLP.ZKP(3, msg[1], HLP.multPowMod(G, msg[2], msg[0], msg[1], N)))
+          return this.abort()
+
+        if (!HLP.ZKP(4, msg[4], HLP.multPowMod(G, msg[5], msg[3], msg[4], N)))
+          return this.abort()
+
+        this.g3ao = msg[3]  // save for later
+
+        this.computeGs(msg[0], msg[3])
+
+        // verify znp of cP
+        t1 = HLP.multPowMod(this.g3, msg[9], msg[6], msg[8], N)
+        t2 = HLP.multPowMod(G, msg[9], this.g2, msg[10], N)
+        t2 = BigInt.multMod(t2, BigInt.powMod(msg[7], msg[8], N), N)
+
+        if (!HLP.ZKP(5, msg[8], t1, t2))
+          return this.abort()
+
+        var r4 = HLP.randomExponent()
+        this.computePQ(r4)
+
+        // zero-knowledge proof that P & Q
+        // were generated according to the protocol
+        var r5 = HLP.randomExponent()
+        var r6 = HLP.randomExponent()
+        var tmp = HLP.multPowMod(G, r5, this.g2, r6, N)
+        var cP = HLP.smpHash(6, BigInt.powMod(this.g3, r5, N), tmp)
+        var d5 = this.computeD(r5, r4, cP)
+        var d6 = this.computeD(r6, this.secret, cP)
+
+        // store these
+        this.QoQ = BigInt.divMod(this.q, msg[7], N)
+        this.PoP = BigInt.divMod(this.p, msg[6], N)
+
+        this.computeR()
+
+        // zero-knowledge proof that R
+        // was generated according to the protocol
+        r7 = HLP.randomExponent()
+        tmp2 = BigInt.powMod(this.QoQ, r7, N)
+        cR = HLP.smpHash(7, BigInt.powMod(G, r7, N), tmp2)
+        d7 = this.computeD(r7, this.a3, cR)
+
+        this.smpstate = CONST.SMPSTATE_EXPECT4
+
+        send = HLP.packINT(8) + HLP.packMPIs([
+            this.p
+          , this.q
+          , cP
+          , d5
+          , d6
+          , this.r
+          , cR
+          , d7
+        ])
+
+        // TLV
+        send = HLP.packTLV(4, send)
+        break
+
+      case CONST.SMPSTATE_EXPECT3:
+        HLP.debug.call(this, 'smp tlv 4')
+
+        // 0:p, 1:q, 2:cP, 3:d5, 4:d6, 5:r, 6:cR, 7:d7
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 8) return this.abort()
+        msg = HLP.unpackMPIs(8, msg.msg.substring(4))
+
+        if ( !HLP.checkGroup(msg[0], N_MINUS_2) ||
+             !HLP.checkGroup(msg[1], N_MINUS_2) ||
+             !HLP.checkGroup(msg[5], N_MINUS_2)
+        ) return this.abort()
+
+        // verify znp of cP
+        t1 = HLP.multPowMod(this.g3, msg[3], msg[0], msg[2], N)
+        t2 = HLP.multPowMod(G, msg[3], this.g2, msg[4], N)
+        t2 = BigInt.multMod(t2, BigInt.powMod(msg[1], msg[2], N), N)
+
+        if (!HLP.ZKP(6, msg[2], t1, t2))
+          return this.abort()
+
+        // verify znp of cR
+        t3 = HLP.multPowMod(G, msg[7], this.g3ao, msg[6], N)
+        this.QoQ = BigInt.divMod(msg[1], this.q, N)  // save Q over Q
+        t4 = HLP.multPowMod(this.QoQ, msg[7], msg[5], msg[6], N)
+
+        if (!HLP.ZKP(7, msg[6], t3, t4))
+          return this.abort()
+
+        this.computeR()
+
+        // zero-knowledge proof that R
+        // was generated according to the protocol
+        r7 = HLP.randomExponent()
+        tmp2 = BigInt.powMod(this.QoQ, r7, N)
+        cR = HLP.smpHash(8, BigInt.powMod(G, r7, N), tmp2)
+        d7 = this.computeD(r7, this.a3, cR)
+
+        send = HLP.packINT(3) + HLP.packMPIs([ this.r, cR, d7 ])
+        send = HLP.packTLV(5, send)
+
+        rab = this.computeRab(msg[5])
+        trust = !!BigInt.equals(rab, BigInt.divMod(msg[0], this.p, N))
+
+        this.trigger('trust', [trust, 'answered'])
+        this.init()
+        break
+
+      case CONST.SMPSTATE_EXPECT4:
+        HLP.debug.call(this, 'smp tlv 5')
+
+        // 0:r, 1:cR, 2:d7
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 3) return this.abort()
+        msg = HLP.unpackMPIs(3, msg.msg.substring(4))
+
+        if (!HLP.checkGroup(msg[0], N_MINUS_2)) return this.abort()
+
+        // verify znp of cR
+        t3 = HLP.multPowMod(G, msg[2], this.g3ao, msg[1], N)
+        t4 = HLP.multPowMod(this.QoQ, msg[2], msg[0], msg[1], N)
+        if (!HLP.ZKP(8, msg[1], t3, t4))
+          return this.abort()
+
+        rab = this.computeRab(msg[0])
+        trust = !!BigInt.equals(rab, this.PoP)
+
+        this.trigger('trust', [trust, 'asked'])
+        this.init()
+        return
+
+    }
+
+    this.sendMsg(send)
+  }
+
+  // send a message
+  SM.prototype.sendMsg = function (send) {
+    this.trigger('send', [this.ssid, '\x00' + send])
+  }
+
+  SM.prototype.rcvSecret = function (secret, question) {
+    HLP.debug.call(this, 'receive secret')
+
+    var fn, our = false
+    if (this.smpstate === CONST.SMPSTATE_EXPECT0) {
+      fn = this.answer
+    } else {
+      fn = this.initiate
+      our = true
+    }
+
+    this.makeSecret(our, secret)
+    fn.call(this, question)
+  }
+
+  SM.prototype.answer = function () {
+    HLP.debug.call(this, 'smp answer')
+
+    var r4 = HLP.randomExponent()
+    this.computePQ(r4)
+
+    // zero-knowledge proof that P & Q
+    // were generated according to the protocol
+    var r5 = HLP.randomExponent()
+    var r6 = HLP.randomExponent()
+    var tmp = HLP.multPowMod(G, r5, this.g2, r6, N)
+    var cP = HLP.smpHash(5, BigInt.powMod(this.g3, r5, N), tmp)
+    var d5 = this.computeD(r5, r4, cP)
+    var d6 = this.computeD(r6, this.secret, cP)
+
+    this.smpstate = CONST.SMPSTATE_EXPECT3
+
+    var send = HLP.packINT(11) + HLP.packMPIs([
+        this.g2a
+      , this.c2
+      , this.d2
+      , this.g3a
+      , this.c3
+      , this.d3
+      , this.p
+      , this.q
+      , cP
+      , d5
+      , d6
+    ])
+
+    this.sendMsg(HLP.packTLV(3, send))
+  }
+
+  SM.prototype.initiate = function (question) {
+    HLP.debug.call(this, 'smp initiate')
+
+    if (this.smpstate !== CONST.SMPSTATE_EXPECT1)
+      this.abort()  // abort + restart
+
+    this.makeG2s()
+
+    // zero-knowledge proof that the exponents
+    // associated with g2a & g3a are known
+    var r2 = HLP.randomExponent()
+    var r3 = HLP.randomExponent()
+    this.c2 = this.computeC(1, r2)
+    this.c3 = this.computeC(2, r3)
+    this.d2 = this.computeD(r2, this.a2, this.c2)
+    this.d3 = this.computeD(r3, this.a3, this.c3)
+
+    // set the next expected state
+    this.smpstate = CONST.SMPSTATE_EXPECT2
+
+    var send = ''
+    var type = 2
+
+    if (question) {
+      send += question
+      send += '\x00'
+      type = 7
+    }
+
+    send += HLP.packINT(6) + HLP.packMPIs([
+        this.g2a
+      , this.c2
+      , this.d2
+      , this.g3a
+      , this.c3
+      , this.d3
+    ])
+
+    this.sendMsg(HLP.packTLV(type, send))
+  }
+
+  SM.prototype.abort = function () {
+    this.init()
+    this.sendMsg(HLP.packTLV(6, ''))
+    this.trigger('abort')
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, EventEmitter, Worker, SMWPath
+    , CONST, HLP, Parse, AKE, SM, DSA
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = OTR
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    EventEmitter = require('../vendor/eventemitter.js')
+    SMWPath = require('path').join(__dirname, '/sm-webworker.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+    Parse = require('./parse.js')
+    AKE = require('./ake.js')
+    SM = require('./sm.js')
+    DSA = require('./dsa.js')
+    // expose CONST for consistency with docs
+    OTR.CONST = CONST
+  } else {
+    // copy over and expose internals
+    Object.keys(root.OTR).forEach(function (k) {
+      OTR[k] = root.OTR[k]
+    })
+    root.OTR = OTR
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    EventEmitter = root.EventEmitter
+    Worker = root.Worker
+    SMWPath = 'sm-webworker.js'
+    CONST = OTR.CONST
+    HLP = OTR.HLP
+    Parse = OTR.Parse
+    AKE = OTR.AKE
+    SM = OTR.SM
+    DSA = root.DSA
+  }
+
+  // diffie-hellman modulus and generator
+  // see group 5, RFC 3526
+  var G = BigInt.str2bigInt(CONST.G, 10)
+  var N = BigInt.str2bigInt(CONST.N, 16)
+
+  // JavaScript integers
+  var MAX_INT = Math.pow(2, 53) - 1  // doubles
+  var MAX_UINT = Math.pow(2, 31) - 1  // bitwise operators
+
+  // an internal callback
+  function OTRCB(cb) {
+    this.cb = cb
+  }
+
+  // OTR contructor
+  function OTR(options) {
+    if (!(this instanceof OTR)) return new OTR(options)
+
+    // options
+    options = options || {}
+
+    // private keys
+    if (options.priv && !(options.priv instanceof DSA))
+      throw new Error('Requires long-lived DSA key.')
+
+    this.priv = options.priv ? options.priv : new DSA()
+
+    this.fragment_size = options.fragment_size || 0
+    if (this.fragment_size < 0)
+      throw new Error('Fragment size must be a positive integer.')
+
+    this.send_interval = options.send_interval || 0
+    if (this.send_interval < 0)
+      throw new Error('Send interval must be a positive integer.')
+
+    this.outgoing = []
+
+    // instance tag
+    this.our_instance_tag = options.instance_tag || OTR.makeInstanceTag()
+
+    // debug
+    this.debug = !!options.debug
+
+    // smp in webworker options
+    // this is still experimental and undocumented
+    this.smw = options.smw
+
+    // init vals
+    this.init()
+
+    // bind methods
+    var self = this
+    ;['sendMsg', 'receiveMsg'].forEach(function (meth) {
+      self[meth] = self[meth].bind(self)
+    })
+
+    EventEmitter.call(this)
+  }
+
+  // inherit from EE
+  HLP.extend(OTR, EventEmitter)
+
+  // add to prototype
+  OTR.prototype.init = function () {
+
+    this.msgstate = CONST.MSGSTATE_PLAINTEXT
+    this.authstate = CONST.AUTHSTATE_NONE
+
+    this.ALLOW_V2 = true
+    this.ALLOW_V3 = true
+
+    this.REQUIRE_ENCRYPTION = false
+    this.SEND_WHITESPACE_TAG = false
+    this.WHITESPACE_START_AKE = false
+    this.ERROR_START_AKE = false
+
+    Parse.initFragment(this)
+
+    // their keys
+    this.their_y = null
+    this.their_old_y = null
+    this.their_keyid = 0
+    this.their_priv_pk = null
+    this.their_instance_tag = '\x00\x00\x00\x00'
+
+    // our keys
+    this.our_dh = this.dh()
+    this.our_old_dh = this.dh()
+    this.our_keyid = 2
+
+    // session keys
+    this.sessKeys = [ new Array(2), new Array(2) ]
+
+    // saved
+    this.storedMgs = []
+    this.oldMacKeys = []
+
+    // smp
+    this.sm = null  // initialized after AKE
+
+    // when ake is complete
+    // save their keys and the session
+    this._akeInit()
+
+    // receive plaintext message since switching to plaintext
+    // used to decide when to stop sending pt tags when SEND_WHITESPACE_TAG
+    this.receivedPlaintext = false
+
+  }
+
+  OTR.prototype._akeInit = function () {
+    this.ake = new AKE(this)
+    this.transmittedRS = false
+    this.ssid = null
+  }
+
+  // smp over webworker
+  OTR.prototype._SMW = function (otr, reqs) {
+    this.otr = otr
+    var opts = {
+        path: SMWPath
+      , seed: BigInt.getSeed
+    }
+    if (typeof otr.smw === 'object')
+      Object.keys(otr.smw).forEach(function (k) {
+        opts[k] = otr.smw[k]
+      })
+
+    // load optional dep. in node
+    if (typeof module !== 'undefined' && module.exports)
+      Worker = require('webworker-threads').Worker
+
+    this.worker = new Worker(opts.path)
+    var self = this
+    this.worker.onmessage = function (e) {
+      var d = e.data
+      if (!d) return
+      self.trigger(d.method, d.args)
+    }
+    this.worker.postMessage({
+        type: 'seed'
+      , seed: opts.seed()
+      , imports: opts.imports
+    })
+    this.worker.postMessage({
+        type: 'init'
+      , reqs: reqs
+    })
+  }
+
+  // inherit from EE
+  HLP.extend(OTR.prototype._SMW, EventEmitter)
+
+  // shim sm methods
+  ;['handleSM', 'rcvSecret', 'abort'].forEach(function (m) {
+    OTR.prototype._SMW.prototype[m] = function () {
+      this.worker.postMessage({
+          type: 'method'
+        , method: m
+        , args: Array.prototype.slice.call(arguments, 0)
+      })
+    }
+  })
+
+  OTR.prototype._smInit = function () {
+    var reqs = {
+        ssid: this.ssid
+      , our_fp: this.priv.fingerprint()
+      , their_fp: this.their_priv_pk.fingerprint()
+      , debug: this.debug
+    }
+    if (this.smw) {
+      if (this.sm) this.sm.worker.terminate()  // destroy prev webworker
+      this.sm = new this._SMW(this, reqs)
+    } else {
+      this.sm = new SM(reqs)
+    }
+    var self = this
+    ;['trust', 'abort', 'question'].forEach(function (e) {
+      self.sm.on(e, function () {
+        self.trigger('smp', [e].concat(Array.prototype.slice.call(arguments)))
+      })
+    })
+    this.sm.on('send', function (ssid, send) {
+      if (self.ssid === ssid) {
+        send = self.prepareMsg(send)
+        self.io(send)
+      }
+    })
+  }
+
+  OTR.prototype.io = function (msg, meta) {
+
+    // buffer
+    msg = ([].concat(msg)).map(function(m){
+       return { msg: m, meta: meta }
+    })
+    this.outgoing = this.outgoing.concat(msg)
+
+    var self = this
+    ;(function send(first) {
+      if (!first) {
+        if (!self.outgoing.length) return
+        var elem = self.outgoing.shift(), cb = null
+        if (elem.meta instanceof OTRCB) {
+          cb = elem.meta.cb
+          elem.meta = null
+        }
+        self.trigger('io', [elem.msg, elem.meta])
+        if (cb) cb()
+      }
+      setTimeout(send, first ? 0 : self.send_interval)
+    }(true))
+
+  }
+
+  OTR.prototype.dh = function dh() {
+    var keys = { privateKey: BigInt.randBigInt(320) }
+    keys.publicKey = BigInt.powMod(G, keys.privateKey, N)
+    return keys
+  }
+
+  // session constructor
+  OTR.prototype.DHSession = function DHSession(our_dh, their_y) {
+    if (!(this instanceof DHSession)) return new DHSession(our_dh, their_y)
+
+    // shared secret
+    var s = BigInt.powMod(their_y, our_dh.privateKey, N)
+    var secbytes = HLP.packMPI(s)
+
+    // session id
+    this.id = HLP.mask(HLP.h2('\x00', secbytes), 0, 64)  // first 64-bits
+
+    // are we the high or low end of the connection?
+    var sq = BigInt.greater(our_dh.publicKey, their_y)
+    var sendbyte = sq ? '\x01' : '\x02'
+    var rcvbyte  = sq ? '\x02' : '\x01'
+
+    // sending and receiving keys
+    this.sendenc = HLP.mask(HLP.h1(sendbyte, secbytes), 0, 128)  // f16 bytes
+    this.sendmac = CryptoJS.SHA1(CryptoJS.enc.Latin1.parse(this.sendenc))
+    this.sendmac = this.sendmac.toString(CryptoJS.enc.Latin1)
+
+    this.rcvenc = HLP.mask(HLP.h1(rcvbyte, secbytes), 0, 128)
+    this.rcvmac = CryptoJS.SHA1(CryptoJS.enc.Latin1.parse(this.rcvenc))
+    this.rcvmac = this.rcvmac.toString(CryptoJS.enc.Latin1)
+    this.rcvmacused = false
+
+    // extra symmetric key
+    this.extra_symkey = HLP.h2('\xff', secbytes)
+
+    // counters
+    this.send_counter = 0
+    this.rcv_counter = 0
+  }
+
+  OTR.prototype.rotateOurKeys = function () {
+
+    // reveal old mac keys
+    var self = this
+    this.sessKeys[1].forEach(function (sk) {
+      if (sk && sk.rcvmacused) self.oldMacKeys.push(sk.rcvmac)
+    })
+
+    // rotate our keys
+    this.our_old_dh = this.our_dh
+    this.our_dh = this.dh()
+    this.our_keyid += 1
+
+    this.sessKeys[1][0] = this.sessKeys[0][0]
+    this.sessKeys[1][1] = this.sessKeys[0][1]
+    this.sessKeys[0] = [
+        this.their_y ?
+            new this.DHSession(this.our_dh, this.their_y) : null
+      , this.their_old_y ?
+            new this.DHSession(this.our_dh, this.their_old_y) : null
+    ]
+
+  }
+
+  OTR.prototype.rotateTheirKeys = function (their_y) {
+
+    // increment their keyid
+    this.their_keyid += 1
+
+    // reveal old mac keys
+    var self = this
+    this.sessKeys.forEach(function (sk) {
+      if (sk[1] && sk[1].rcvmacused) self.oldMacKeys.push(sk[1].rcvmac)
+    })
+
+    // rotate their keys / session
+    this.their_old_y = this.their_y
+    this.sessKeys[0][1] = this.sessKeys[0][0]
+    this.sessKeys[1][1] = this.sessKeys[1][0]
+
+    // new keys / sessions
+    this.their_y = their_y
+    this.sessKeys[0][0] = new this.DHSession(this.our_dh, this.their_y)
+    this.sessKeys[1][0] = new this.DHSession(this.our_old_dh, this.their_y)
+
+  }
+
+  OTR.prototype.prepareMsg = function (msg, esk) {
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED || this.their_keyid === 0)
+      return this.notify('Not ready to encrypt.')
+
+    var sessKeys = this.sessKeys[1][0]
+
+    if (sessKeys.send_counter >= MAX_INT)
+      return this.notify('Should have rekeyed by now.')
+
+    sessKeys.send_counter += 1
+
+    var ctr = HLP.packCtr(sessKeys.send_counter)
+
+    var send = this.ake.otr_version + '\x03'  // version and type
+    var v3 = (this.ake.otr_version === CONST.OTR_VERSION_3)
+
+    if (v3) {
+      send += this.our_instance_tag
+      send += this.their_instance_tag
+    }
+
+    send += '\x00'  // flag
+    send += HLP.packINT(this.our_keyid - 1)
+    send += HLP.packINT(this.their_keyid)
+    send += HLP.packMPI(this.our_dh.publicKey)
+    send += ctr.substring(0, 8)
+
+    if (Math.ceil(msg.length / 8) >= MAX_UINT)  // * 16 / 128
+      return this.notify('Message is too long.')
+
+    var aes = HLP.encryptAes(
+        CryptoJS.enc.Latin1.parse(msg)
+      , sessKeys.sendenc
+      , ctr
+    )
+
+    send += HLP.packData(aes)
+    send += HLP.make1Mac(send, sessKeys.sendmac)
+    send += HLP.packData(this.oldMacKeys.splice(0).join(''))
+
+    send = HLP.wrapMsg(
+        send
+      , this.fragment_size
+      , v3
+      , this.our_instance_tag
+      , this.their_instance_tag
+    )
+    if (send[0]) return this.notify(send[0])
+
+    // emit extra symmetric key
+    if (esk) this.trigger('file', ['send', sessKeys.extra_symkey, esk])
+
+    return send[1]
+  }
+
+  OTR.prototype.handleDataMsg = function (msg) {
+    var vt = msg.version + msg.type
+
+    if (this.ake.otr_version === CONST.OTR_VERSION_3)
+      vt += msg.instance_tags
+
+    var types = ['BYTE', 'INT', 'INT', 'MPI', 'CTR', 'DATA', 'MAC', 'DATA']
+    msg = HLP.splitype(types, msg.msg)
+
+    // ignore flag
+    var ign = (msg[0] === '\x01')
+
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED || msg.length !== 8) {
+      if (!ign) this.error('Received an unreadable encrypted message.')
+      return
+    }
+
+    var our_keyid = this.our_keyid - HLP.readLen(msg[2])
+    var their_keyid = this.their_keyid - HLP.readLen(msg[1])
+
+    if (our_keyid < 0 || our_keyid > 1) {
+      if (!ign) this.error('Not of our latest keys.')
+      return
+    }
+
+    if (their_keyid < 0 || their_keyid > 1) {
+      if (!ign) this.error('Not of your latest keys.')
+      return
+    }
+
+    var their_y = their_keyid ? this.their_old_y : this.their_y
+
+    if (their_keyid === 1 && !their_y) {
+      if (!ign) this.error('Do not have that key.')
+      return
+    }
+
+    var sessKeys = this.sessKeys[our_keyid][their_keyid]
+
+    var ctr = HLP.unpackCtr(msg[4])
+    if (ctr <= sessKeys.rcv_counter) {
+      if (!ign) this.error('Counter in message is not larger.')
+      return
+    }
+    sessKeys.rcv_counter = ctr
+
+    // verify mac
+    vt += msg.slice(0, 6).join('')
+    var vmac = HLP.make1Mac(vt, sessKeys.rcvmac)
+
+    if (!HLP.compare(msg[6], vmac)) {
+      if (!ign) this.error('MACs do not match.')
+      return
+    }
+    sessKeys.rcvmacused = true
+
+    var out = HLP.decryptAes(
+        msg[5].substring(4)
+      , sessKeys.rcvenc
+      , HLP.padCtr(msg[4])
+    )
+    out = out.toString(CryptoJS.enc.Latin1)
+
+    if (!our_keyid) this.rotateOurKeys()
+    if (!their_keyid) this.rotateTheirKeys(HLP.readMPI(msg[3]))
+
+    // parse TLVs
+    var ind = out.indexOf('\x00')
+    if (~ind) {
+      this.handleTLVs(out.substring(ind + 1), sessKeys)
+      out = out.substring(0, ind)
+    }
+
+    out = CryptoJS.enc.Latin1.parse(out)
+    return out.toString(CryptoJS.enc.Utf8)
+  }
+
+  OTR.prototype.handleTLVs = function (tlvs, sessKeys) {
+    var type, len, msg
+    for (; tlvs.length; ) {
+      type = HLP.unpackSHORT(tlvs.substr(0, 2))
+      len = HLP.unpackSHORT(tlvs.substr(2, 2))
+
+      msg = tlvs.substr(4, len)
+
+      // TODO: handle pathological cases better
+      if (msg.length < len) break
+
+      switch (type) {
+        case 1:
+          // Disconnected
+          this.msgstate = CONST.MSGSTATE_FINISHED
+          this.trigger('status', [CONST.STATUS_END_OTR])
+          break
+        case 2: case 3: case 4:
+        case 5: case 6: case 7:
+          // SMP
+          if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED) {
+            if (this.sm) this.sm.abort()
+            return
+          }
+          if (!this.sm) this._smInit()
+          this.sm.handleSM({ msg: msg, type: type })
+          break
+        case 8:
+          // utf8 filenames
+          msg = msg.substring(4) // remove 4-byte indication
+          msg = CryptoJS.enc.Latin1.parse(msg)
+          msg = msg.toString(CryptoJS.enc.Utf8)
+
+          // Extra Symkey
+          this.trigger('file', ['receive', sessKeys.extra_symkey, msg])
+          break
+      }
+
+      tlvs = tlvs.substring(4 + len)
+    }
+  }
+
+  OTR.prototype.smpSecret = function (secret, question) {
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED)
+      return this.notify('Must be encrypted for SMP.')
+
+    if (typeof secret !== 'string' || secret.length < 1)
+      return this.notify('Secret is required.')
+
+    if (!this.sm) this._smInit()
+
+    // utf8 inputs
+    secret = CryptoJS.enc.Utf8.parse(secret).toString(CryptoJS.enc.Latin1)
+    question = CryptoJS.enc.Utf8.parse(question).toString(CryptoJS.enc.Latin1)
+
+    this.sm.rcvSecret(secret, question)
+  }
+
+  OTR.prototype.sendQueryMsg = function () {
+    var versions = {}
+      , msg = CONST.OTR_TAG
+
+    if (this.ALLOW_V2) versions['2'] = true
+    if (this.ALLOW_V3) versions['3'] = true
+
+    // but we don't allow v1
+    // if (versions['1']) msg += '?'
+
+    var vs = Object.keys(versions)
+    if (vs.length) {
+      msg += 'v'
+      vs.forEach(function (v) {
+        if (v !== '1') msg += v
+      })
+      msg += '?'
+    }
+
+    this.io(msg)
+    this.trigger('status', [CONST.STATUS_SEND_QUERY])
+  }
+
+  OTR.prototype.sendMsg = function (msg, meta) {
+    if ( this.REQUIRE_ENCRYPTION ||
+         this.msgstate !== CONST.MSGSTATE_PLAINTEXT
+    ) {
+      msg = CryptoJS.enc.Utf8.parse(msg)
+      msg = msg.toString(CryptoJS.enc.Latin1)
+    }
+
+    switch (this.msgstate) {
+      case CONST.MSGSTATE_PLAINTEXT:
+        if (this.REQUIRE_ENCRYPTION) {
+          this.storedMgs.push({msg: msg, meta: meta})
+          this.sendQueryMsg()
+          return
+        }
+        if (this.SEND_WHITESPACE_TAG && !this.receivedPlaintext) {
+          msg += CONST.WHITESPACE_TAG  // 16 byte tag
+          if (this.ALLOW_V3) msg += CONST.WHITESPACE_TAG_V3
+          if (this.ALLOW_V2) msg += CONST.WHITESPACE_TAG_V2
+        }
+        break
+      case CONST.MSGSTATE_FINISHED:
+        this.storedMgs.push({msg: msg, meta: meta})
+        this.notify('Message cannot be sent at this time.', 'warn')
+        return
+      case CONST.MSGSTATE_ENCRYPTED:
+        msg = this.prepareMsg(msg)
+        break
+      default:
+        throw new Error('Unknown message state.')
+    }
+
+    if (msg) this.io(msg, meta)
+  }
+
+  OTR.prototype.receiveMsg = function (msg) {
+
+    // parse type
+    msg = Parse.parseMsg(this, msg)
+
+    if (!msg) return
+
+    switch (msg.cls) {
+      case 'error':
+        this.notify(msg.msg)
+        return
+      case 'ake':
+        if ( msg.version === CONST.OTR_VERSION_3 &&
+          this.checkInstanceTags(msg.instance_tags)
+        ) {
+          this.notify(
+            'Received a message intended for a different session.', 'warn')
+          return  // ignore
+        }
+        this.ake.handleAKE(msg)
+        return
+      case 'data':
+        if ( msg.version === CONST.OTR_VERSION_3 &&
+          this.checkInstanceTags(msg.instance_tags)
+        ) {
+          this.notify(
+            'Received a message intended for a different session.', 'warn')
+          return  // ignore
+        }
+        msg.msg = this.handleDataMsg(msg)
+        msg.encrypted = true
+        break
+      case 'query':
+        if (this.msgstate === CONST.MSGSTATE_ENCRYPTED) this._akeInit()
+        this.doAKE(msg)
+        break
+      default:
+        // check for encrypted
+        if ( this.REQUIRE_ENCRYPTION ||
+             this.msgstate !== CONST.MSGSTATE_PLAINTEXT
+        ) this.notify('Received an unencrypted message.', 'warn')
+
+        // received a plaintext message
+        // stop sending the whitespace tag
+        this.receivedPlaintext = true
+
+        // received a whitespace tag
+        if (this.WHITESPACE_START_AKE && msg.ver.length > 0)
+          this.doAKE(msg)
+    }
+
+    if (msg.msg) this.trigger('ui', [msg.msg, !!msg.encrypted])
+  }
+
+  OTR.prototype.checkInstanceTags = function (it) {
+    var their_it = HLP.readLen(it.substr(0, 4))
+    var our_it = HLP.readLen(it.substr(4, 4))
+
+    if (our_it && our_it !== HLP.readLen(this.our_instance_tag))
+      return true
+
+    if (HLP.readLen(this.their_instance_tag)) {
+      if (HLP.readLen(this.their_instance_tag) !== their_it) return true
+    } else {
+      if (their_it < 100) return true
+      this.their_instance_tag = HLP.packINT(their_it)
+    }
+  }
+
+  OTR.prototype.doAKE = function (msg) {
+    if (this.ALLOW_V3 && ~msg.ver.indexOf(CONST.OTR_VERSION_3)) {
+      this.ake.initiateAKE(CONST.OTR_VERSION_3)
+    } else if (this.ALLOW_V2 && ~msg.ver.indexOf(CONST.OTR_VERSION_2)) {
+      this.ake.initiateAKE(CONST.OTR_VERSION_2)
+    } else {
+      this.notify('OTR conversation requested, ' +
+        'but no compatible protocol version found.', 'warn')
+    }
+  }
+
+  OTR.prototype.error = function (err) {
+    if (!this.debug) err = 'An OTR error has occurred.'
+    this.io('?OTR Error:' + err)
+    this.notify(err)
+  }
+
+  OTR.prototype.notify = function (err, severity) {
+    this.trigger('error', [err, severity || 'error'])
+  }
+
+  OTR.prototype.sendStored = function () {
+    var self = this
+    ;(this.storedMgs.splice(0)).forEach(function (elem) {
+      var msg = self.prepareMsg(elem.msg)
+      self.io(msg, elem.meta)
+    })
+  }
+
+  OTR.prototype.sendFile = function (filename) {
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED)
+      return this.notify('Not ready to encrypt.')
+
+    if (this.ake.otr_version !== CONST.OTR_VERSION_3)
+      return this.notify('Protocol v3 required.')
+
+    if (!filename) return this.notify('Please specify a filename.')
+
+    // utf8 filenames
+    var l1name = CryptoJS.enc.Utf8.parse(filename)
+    l1name = l1name.toString(CryptoJS.enc.Latin1)
+
+    if (l1name.length >= 65532) return this.notify('Filename is too long.')
+
+    var msg = '\x00'  // null byte
+    msg += '\x00\x08'  // type 8 tlv
+    msg += HLP.packSHORT(4 + l1name.length)  // length of value
+    msg += '\x00\x00\x00\x01'  // four bytes indicating file
+    msg += l1name
+
+    msg = this.prepareMsg(msg, filename)
+    this.io(msg)
+  }
+
+  OTR.prototype.endOtr = function (cb) {
+    if (this.msgstate === CONST.MSGSTATE_ENCRYPTED) {
+      if (typeof cb === 'function')
+        cb = new OTRCB(cb)
+      this.sendMsg('\x00\x00\x01\x00\x00', cb)
+      if (this.sm) {
+        if (this.smw) this.sm.worker.terminate()  // destroy webworker
+        this.sm = null
+      }
+    }
+    this.msgstate = CONST.MSGSTATE_PLAINTEXT
+    this.receivedPlaintext = false
+    this.trigger('status', [CONST.STATUS_END_OTR])
+  }
+
+  // attach methods
+
+  OTR.makeInstanceTag = function () {
+    var num = BigInt.randBigInt(32)
+    if (BigInt.greater(BigInt.str2bigInt('100', 16), num))
+      return OTR.makeInstanceTag()
+    return HLP.packINT(parseInt(BigInt.bigInt2str(num, 10), 10))
+  }
+
+}).call(this)
+
+
+  return {
+      OTR: this.OTR
+    , DSA: this.DSA
+  }
+
+}))
\ No newline at end of file
diff --git a/lib/jsxc.dep.min.js b/lib/jsxc.dep.min.js
new file mode 100644
index 0000000..fa59790
--- /dev/null
+++ b/lib/jsxc.dep.min.js
@@ -0,0 +1,67 @@
+/*!
+ * jsxc v1.0.0 - 2014-11-06
+ * 
+ * This file concatenates all dependencies of jsxc.
+ * 
+ */
+/*!
+ * Source: build/lib/strophe.js, license: multiple, url: http://strophe.im/strophejs/ */
+/*! This code was written by Tyler Akins and has been placed in the
+   public domain.  It would be nice if you left this header intact.
+   Base64 code from Tyler Akins -- http://rumkin.com
+*/
+/*!
+ * A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined
+ * in FIPS PUB 180-1
+ * Version 2.1a Copyright Paul Johnston 2000 - 2002.
+ * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+ * Distributed under the BSD License
+ * See http://pajhome.org.uk/crypt/md5 for details.
+ */
+function b64_sha1(s){return binb2b64(core_sha1(str2binb(s),8*s.length))}function str_sha1(s){return binb2str(core_sha1(str2binb(s),8*s.length))}function b64_hmac_sha1(key,data){return binb2b64(core_hmac_sha1(key,data))}function str_hmac_sha1(key,data){return binb2str(core_hmac_sha1(key,data))}function core_sha1(x,len){x[len>>5]|=128<<24-len%32,x[(len+64>>9<<4)+15]=len;var i,j,t,olda,oldb,oldc,oldd,olde,w=new Array(80),a=1732584193,b=-271733879,c=-1732584194,d=271733878,e=-1009589776;for( [...]
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+Function.prototype.bind||(Function.prototype.bind=function(obj){var func=this,_slice=Array.prototype.slice,_concat=Array.prototype.concat,_args=_slice.call(arguments,1);return function(){return func.apply(obj?obj:this,_concat.call(_args,_slice.call(arguments,0)))}}),Array.prototype.indexOf||(Array.prototype.indexOf=function(elt){var len=this.length,from=Number(arguments[1])||0;for(from=0>from?Math.ceil(from):Math.floor(from),0>from&&(from+=len);len>from;from++)if(from in this&&this[from] [...]
+}.bind(this),responseText},Strophe.Connection.prototype.mechanisms[Strophe.SASLMD5.prototype.name]=Strophe.SASLMD5}(function(){window.Strophe=arguments[0],window.$build=arguments[1],window.$msg=arguments[2],window.$iq=arguments[3],window.$pres=arguments[4]}),Strophe.Request=function(elem,func,rid,sends){this.id=++Strophe._requestId,this.xmlData=elem,this.data=Strophe.serialize(elem),this.origFunc=func,this.func=func,this.rid=rid,this.date=0/0,this.sends=sends||0,this.abort=!1,this.dead=n [...]
+ * Source: build/lib/strophe.muc.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+var Occupant,RoomConfig,XmppRoom,__bind=function(fn,me){return function(){return fn.apply(me,arguments)}};Strophe.addConnectionPlugin("muc",{_connection:null,rooms:{},roomNames:[],init:function(conn){return this._connection=conn,this._muc_handler=null,Strophe.addNamespace("MUC_OWNER",Strophe.NS.MUC+"#owner"),Strophe.addNamespace("MUC_ADMIN",Strophe.NS.MUC+"#admin"),Strophe.addNamespace("MUC_USER",Strophe.NS.MUC+"#user"),Strophe.addNamespace("MUC_ROOMCONF",Strophe.NS.MUC+"#roomconfig")},j [...]
+ * Source: build/lib/strophe.disco.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+Strophe.addConnectionPlugin("disco",{_connection:null,_identities:[],_features:[],_items:[],init:function(conn){this._connection=conn,this._identities=[],this._features=[],this._items=[],conn.addHandler(this._onDiscoInfo.bind(this),Strophe.NS.DISCO_INFO,"iq","get",null,null),conn.addHandler(this._onDiscoItems.bind(this),Strophe.NS.DISCO_ITEMS,"iq","get",null,null)},addIdentity:function(category,type,name,lang){for(var i=0;i<this._identities.length;i++)if(this._identities[i].category==cat [...]
+ * Source: build/lib/strophe.caps.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+/**
+ * Entity Capabilities (XEP-0115)
+ * 
+ * Depends on disco plugin.
+ * 
+ * See: http://xmpp.org/extensions/xep-0115.html
+ * 
+ * Authors: - Michael Weibel <michael.weibel at gmail.com> - Klaus Herberth <klaus at jsxc.org>
+ * Copyright: - Michael Weibel <michael.weibel at gmail.com>
+ * 
+ * @license MIT
+ */
+function($){Strophe.addConnectionPlugin("caps",{HASH:"sha-1",node:"http://strophe.im/strophejs/",_ver:"",_connection:null,_knownCapabilities:JSON.parse(localStorage.getItem("strophe.caps._knownCapabilities"))||{},_jidVerIndex:JSON.parse(localStorage.getItem("strophe.caps._jidVerIndex"))||{},init:function(conn){if(this._connection=conn,Strophe.addNamespace("CAPS","http://jabber.org/protocol/caps"),!this._connection.disco)throw"Caps plugin requires the disco plugin to be installed.";this._ [...]
+ * Source: build/lib/strophe.vcard.js, license: MIT, url: https://github.com/strophe/strophejs-plugins */
+var buildIq;buildIq=function(type,jid,vCardEl){var iq;return iq=$iq(jid?{type:type,to:jid}:{type:type}),iq.c("vCard",{xmlns:Strophe.NS.VCARD}),vCardEl&&iq.cnode(vCardEl),iq},Strophe.addConnectionPlugin("vcard",{_connection:null,init:function(conn){return this._connection=conn,Strophe.addNamespace("VCARD","vcard-temp")},get:function(handler_cb,jid,error_cb){var iq;return iq=buildIq("get",jid),this._connection.sendIQ(iq,handler_cb,error_cb)},set:function(handler_cb,vCardEl,jid){var iq;retu [...]
+ * Source: build/lib/strophe.jingle/strophe.jingle.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+function($){Strophe.addConnectionPlugin("jingle",{connection:null,sessions:{},jid2session:{},ice_config:{iceServers:[]},pc_constraints:{},media_constraints:{mandatory:{OfferToReceiveAudio:!0,OfferToReceiveVideo:!0}},localStream:null,init:function(conn){this.connection=conn,this.connection.disco&&(this.connection.disco.addFeature("urn:xmpp:jingle:1"),this.connection.disco.addFeature("urn:xmpp:jingle:apps:rtp:1"),this.connection.disco.addFeature("urn:xmpp:jingle:transports:ice-udp:1"),this [...]
+ * Source: build/lib/strophe.jingle/strophe.jingle.session.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+var JingleSession;!function($){JingleSession=function(me,sid,connection){this.me=me,this.sid=sid,this.connection=connection,this.initiator=null,this.responder=null,this.isInitiator=null,this.peerjid=null,this.state=null,this.peerconnection=null,this.remoteStream=null,this.localSDP=null,this.remoteSDP=null,this.localStreams=[],this.relayedStreams=[],this.remoteStreams=[],this.startTime=null,this.stopTime=null,this.media_constraints=null,this.pc_constraints=null,this.ice_config={},this.dri [...]
+ * Source: build/lib/strophe.jingle/strophe.jingle.sdp.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+var SDP;!function($){SDP=function(sdp){this.media=sdp.split("\r\nm=");for(var i=1;i<this.media.length;i++)this.media[i]="m="+this.media[i],i!=this.media.length-1&&(this.media[i]+="\r\n");this.session=this.media.shift()+"\r\n",this.raw=this.session+this.media.join("")},SDP.prototype.mangle=function(){var i,j,mline,lines,rtpmap,newdesc;for(i=0;i<this.media.length;i++)if(lines=this.media[i].split("\r\n"),lines.pop(),mline=SDPUtil.parse_mline(lines.shift()),"audio"==mline.media){for(newdesc= [...]
+ * Source: build/lib/strophe.jingle/strophe.jingle.adapter.js, license: MIT, url: https://github.com/ESTOS/strophe.jingle */
+var setupRTC,getUserMediaWithConstraints,TraceablePeerConnection;!function($){TraceablePeerConnection=function(ice_config,constraints){var self=this,RTCPeerconnection=navigator.mozGetUserMedia?mozRTCPeerConnection:webkitRTCPeerConnection;this.peerconnection=new RTCPeerconnection(ice_config,constraints),this.updateLog=[],this.stats={},this.statsinterval=null,this.maxstats=300,this.trace=function(what,info){self.updateLog.push({time:new Date,type:what,value:info||""})},this.onicecandidate= [...]
+return C}(Math);return function(){{var C=CryptoJS,C_lib=C.lib,WordArray=C_lib.WordArray,C_enc=C.enc;C_enc.Base64={stringify:function(wordArray){var words=wordArray.words,sigBytes=wordArray.sigBytes,map=this._map;wordArray.clamp();for(var base64Chars=[],i=0;sigBytes>i;i+=3)for(var byte1=words[i>>>2]>>>24-i%4*8&255,byte2=words[i+1>>>2]>>>24-(i+1)%4*8&255,byte3=words[i+2>>>2]>>>24-(i+2)%4*8&255,triplet=byte1<<16|byte2<<8|byte3,j=0;4>j&&sigBytes>i+.75*j;j++)base64Chars.push(map.charAt(triple [...]
+ * Source: build/lib/otr/build/dep/eventemitter.js, license: MIT, url: http://git.io/ee */
+/*!
+ * EventEmitter v4.2.3 - git.io/ee
+ * Oliver Caldwell
+ * MIT license
+ * @preserve
+ */
+function(){"use strict";function EventEmitter(){}function indexOfListener(listeners,listener){for(var i=listeners.length;i--;)if(listeners[i].listener===listener)return i;return-1}function alias(name){return function(){return this[name].apply(this,arguments)}}var proto=EventEmitter.prototype;proto.getListeners=function(evt){var response,key,events=this._getEvents();if("object"==typeof evt){response={};for(key in events)events.hasOwnProperty(key)&&evt.test(key)&&(response[key]=events[key] [...]
+case CONST.SMPSTATE_EXPECT2:if(HLP.debug.call(this,"smp tlv 3"),ms=HLP.readLen(msg.msg.substr(0,4)),11!==ms)return this.abort();if(msg=HLP.unpackMPIs(11,msg.msg.substring(4)),!(HLP.checkGroup(msg[0],N_MINUS_2)&&HLP.checkGroup(msg[3],N_MINUS_2)&&HLP.checkGroup(msg[6],N_MINUS_2)&&HLP.checkGroup(msg[7],N_MINUS_2)))return this.abort();if(!HLP.ZKP(3,msg[1],HLP.multPowMod(G,msg[2],msg[0],msg[1],N)))return this.abort();if(!HLP.ZKP(4,msg[4],HLP.multPowMod(G,msg[5],msg[3],msg[4],N)))return this.a [...]
+//# sourceMappingURL=jsxc.dep.min.js.map
\ No newline at end of file
diff --git a/lib/jsxc.dep.min.js.map b/lib/jsxc.dep.min.js.map
new file mode 100644
index 0000000..4bb0586
--- /dev/null
+++ b/lib/jsxc.dep.min.js.map
@@ -0,0 +1 @@
+{"version":3,"file":"jsxc.dep.min.js","sources":["jsxc.dep.js"],"names":["b64_sha1","s","binb2b64","core_sha1","str2binb","length","str_sha1","binb2str","b64_hmac_sha1","key","data","core_hmac_sha1","str_hmac_sha1","x","len","i","j","t","olda","oldb","oldc","oldd","olde","w","Array","a","b","c","d","e","rol","safe_add","sha1_ft","sha1_kt","bkey","ipad","opad","hash","concat","y","lsw","msw","num","cnt","str","bin","mask","charCodeAt","String","fromCharCode","binarray","triplet","tab","ch [...]
\ No newline at end of file
diff --git a/lib/otr/build/dep/bigint.js b/lib/otr/build/dep/bigint.js
new file mode 100644
index 0000000..b7271da
--- /dev/null
+++ b/lib/otr/build/dep/bigint.js
@@ -0,0 +1,1705 @@
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define(factory.bind(root, root.crypto || root.msCrypto))
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory(require('crypto'))
+  } else {
+    root.BigInt = factory(root.crypto || root.msCrypto)
+  }
+
+}(this, function (crypto) {
+
+  ////////////////////////////////////////////////////////////////////////////////////////
+  // Big Integer Library v. 5.5
+  // Created 2000, last modified 2013
+  // Leemon Baird
+  // www.leemon.com
+  //
+  // Version history:
+  // v 5.5  17 Mar 2013
+  //   - two lines of a form like "if (x<0) x+=n" had the "if" changed to "while" to
+  //     handle the case when x<-n. (Thanks to James Ansell for finding that bug)
+  // v 5.4  3 Oct 2009
+  //   - added "var i" to greaterShift() so i is not global. (Thanks to Péter Szabó for finding that bug)
+  //
+  // v 5.3  21 Sep 2009
+  //   - added randProbPrime(k) for probable primes
+  //   - unrolled loop in mont_ (slightly faster)
+  //   - millerRabin now takes a bigInt parameter rather than an int
+  //
+  // v 5.2  15 Sep 2009
+  //   - fixed capitalization in call to int2bigInt in randBigInt
+  //     (thanks to Emili Evripidou, Reinhold Behringer, and Samuel Macaleese for finding that bug)
+  //
+  // v 5.1  8 Oct 2007 
+  //   - renamed inverseModInt_ to inverseModInt since it doesn't change its parameters
+  //   - added functions GCD and randBigInt, which call GCD_ and randBigInt_
+  //   - fixed a bug found by Rob Visser (see comment with his name below)
+  //   - improved comments
+  //
+  // This file is public domain.   You can use it for any purpose without restriction.
+  // I do not guarantee that it is correct, so use it at your own risk.  If you use 
+  // it for something interesting, I'd appreciate hearing about it.  If you find 
+  // any bugs or make any improvements, I'd appreciate hearing about those too.
+  // It would also be nice if my name and URL were left in the comments.  But none 
+  // of that is required.
+  //
+  // This code defines a bigInt library for arbitrary-precision integers.
+  // A bigInt is an array of integers storing the value in chunks of bpe bits, 
+  // little endian (buff[0] is the least significant word).
+  // Negative bigInts are stored two's complement.  Almost all the functions treat
+  // bigInts as nonnegative.  The few that view them as two's complement say so
+  // in their comments.  Some functions assume their parameters have at least one 
+  // leading zero element. Functions with an underscore at the end of the name put
+  // their answer into one of the arrays passed in, and have unpredictable behavior 
+  // in case of overflow, so the caller must make sure the arrays are big enough to 
+  // hold the answer.  But the average user should never have to call any of the 
+  // underscored functions.  Each important underscored function has a wrapper function 
+  // of the same name without the underscore that takes care of the details for you.  
+  // For each underscored function where a parameter is modified, that same variable 
+  // must not be used as another argument too.  So, you cannot square x by doing 
+  // multMod_(x,x,n).  You must use squareMod_(x,n) instead, or do y=dup(x); multMod_(x,y,n).
+  // Or simply use the multMod(x,x,n) function without the underscore, where
+  // such issues never arise, because non-underscored functions never change
+  // their parameters; they always allocate new memory for the answer that is returned.
+  //
+  // These functions are designed to avoid frequent dynamic memory allocation in the inner loop.
+  // For most functions, if it needs a BigInt as a local variable it will actually use
+  // a global, and will only allocate to it only when it's not the right size.  This ensures
+  // that when a function is called repeatedly with same-sized parameters, it only allocates
+  // memory on the first call.
+  //
+  // Note that for cryptographic purposes, the calls to Math.random() must 
+  // be replaced with calls to a better pseudorandom number generator.
+  //
+  // In the following, "bigInt" means a bigInt with at least one leading zero element,
+  // and "integer" means a nonnegative integer less than radix.  In some cases, integer 
+  // can be negative.  Negative bigInts are 2s complement.
+  // 
+  // The following functions do not modify their inputs.
+  // Those returning a bigInt, string, or Array will dynamically allocate memory for that value.
+  // Those returning a boolean will return the integer 0 (false) or 1 (true).
+  // Those returning boolean or int will not allocate memory except possibly on the first 
+  // time they're called with a given parameter size.
+  // 
+  // bigInt  add(x,y)               //return (x+y) for bigInts x and y.  
+  // bigInt  addInt(x,n)            //return (x+n) where x is a bigInt and n is an integer.
+  // string  bigInt2str(x,base)     //return a string form of bigInt x in a given base, with 2 <= base <= 95
+  // int     bitSize(x)             //return how many bits long the bigInt x is, not counting leading zeros
+  // bigInt  dup(x)                 //return a copy of bigInt x
+  // boolean equals(x,y)            //is the bigInt x equal to the bigint y?
+  // boolean equalsInt(x,y)         //is bigint x equal to integer y?
+  // bigInt  expand(x,n)            //return a copy of x with at least n elements, adding leading zeros if needed
+  // Array   findPrimes(n)          //return array of all primes less than integer n
+  // bigInt  GCD(x,y)               //return greatest common divisor of bigInts x and y (each with same number of elements).
+  // boolean greater(x,y)           //is x>y?  (x and y are nonnegative bigInts)
+  // boolean greaterShift(x,y,shift)//is (x <<(shift*bpe)) > y?
+  // bigInt  int2bigInt(t,n,m)      //return a bigInt equal to integer t, with at least n bits and m array elements
+  // bigInt  inverseMod(x,n)        //return (x**(-1) mod n) for bigInts x and n.  If no inverse exists, it returns null
+  // int     inverseModInt(x,n)     //return x**(-1) mod n, for integers x and n.  Return 0 if there is no inverse
+  // boolean isZero(x)              //is the bigInt x equal to zero?
+  // boolean millerRabin(x,b)       //does one round of Miller-Rabin base integer b say that bigInt x is possibly prime? (b is bigInt, 1<b<x)
+  // boolean millerRabinInt(x,b)    //does one round of Miller-Rabin base integer b say that bigInt x is possibly prime? (b is int,    1<b<x)
+  // bigInt  mod(x,n)               //return a new bigInt equal to (x mod n) for bigInts x and n.
+  // int     modInt(x,n)            //return x mod n for bigInt x and integer n.
+  // bigInt  mult(x,y)              //return x*y for bigInts x and y. This is faster when y<x.
+  // bigInt  multMod(x,y,n)         //return (x*y mod n) for bigInts x,y,n.  For greater speed, let y<x.
+  // boolean negative(x)            //is bigInt x negative?
+  // bigInt  powMod(x,y,n)          //return (x**y mod n) where x,y,n are bigInts and ** is exponentiation.  0**0=1. Faster for odd n.
+  // bigInt  randBigInt(n,s)        //return an n-bit random BigInt (n>=1).  If s=1, then the most significant of those n bits is set to 1.
+  // bigInt  randTruePrime(k)       //return a new, random, k-bit, true prime bigInt using Maurer's algorithm.
+  // bigInt  randProbPrime(k)       //return a new, random, k-bit, probable prime bigInt (probability it's composite less than 2^-80).
+  // bigInt  str2bigInt(s,b,n,m)    //return a bigInt for number represented in string s in base b with at least n bits and m array elements
+  // bigInt  sub(x,y)               //return (x-y) for bigInts x and y.  Negative answers will be 2s complement
+  // bigInt  trim(x,k)              //return a copy of x with exactly k leading zero elements
+  //
+  //
+  // The following functions each have a non-underscored version, which most users should call instead.
+  // These functions each write to a single parameter, and the caller is responsible for ensuring the array 
+  // passed in is large enough to hold the result. 
+  //
+  // void    addInt_(x,n)          //do x=x+n where x is a bigInt and n is an integer
+  // void    add_(x,y)             //do x=x+y for bigInts x and y
+  // void    copy_(x,y)            //do x=y on bigInts x and y
+  // void    copyInt_(x,n)         //do x=n on bigInt x and integer n
+  // void    GCD_(x,y)             //set x to the greatest common divisor of bigInts x and y, (y is destroyed).  (This never overflows its array).
+  // boolean inverseMod_(x,n)      //do x=x**(-1) mod n, for bigInts x and n. Returns 1 (0) if inverse does (doesn't) exist
+  // void    mod_(x,n)             //do x=x mod n for bigInts x and n. (This never overflows its array).
+  // void    mult_(x,y)            //do x=x*y for bigInts x and y.
+  // void    multMod_(x,y,n)       //do x=x*y  mod n for bigInts x,y,n.
+  // void    powMod_(x,y,n)        //do x=x**y mod n, where x,y,n are bigInts (n is odd) and ** is exponentiation.  0**0=1.
+  // void    randBigInt_(b,n,s)    //do b = an n-bit random BigInt. if s=1, then nth bit (most significant bit) is set to 1. n>=1.
+  // void    randTruePrime_(ans,k) //do ans = a random k-bit true random prime (not just probable prime) with 1 in the msb.
+  // void    sub_(x,y)             //do x=x-y for bigInts x and y. Negative answers will be 2s complement.
+  //
+  // The following functions do NOT have a non-underscored version. 
+  // They each write a bigInt result to one or more parameters.  The caller is responsible for
+  // ensuring the arrays passed in are large enough to hold the results. 
+  //
+  // void addShift_(x,y,ys)       //do x=x+(y<<(ys*bpe))
+  // void carry_(x)               //do carries and borrows so each element of the bigInt x fits in bpe bits.
+  // void divide_(x,y,q,r)        //divide x by y giving quotient q and remainder r
+  // int  divInt_(x,n)            //do x=floor(x/n) for bigInt x and integer n, and return the remainder. (This never overflows its array).
+  // int  eGCD_(x,y,d,a,b)        //sets a,b,d to positive bigInts such that d = GCD_(x,y) = a*x-b*y
+  // void halve_(x)               //do x=floor(|x|/2)*sgn(x) for bigInt x in 2's complement.  (This never overflows its array).
+  // void leftShift_(x,n)         //left shift bigInt x by n bits.  n<bpe.
+  // void linComb_(x,y,a,b)       //do x=a*x+b*y for bigInts x and y and integers a and b
+  // void linCombShift_(x,y,b,ys) //do x=x+b*(y<<(ys*bpe)) for bigInts x and y, and integers b and ys
+  // void mont_(x,y,n,np)         //Montgomery multiplication (see comments where the function is defined)
+  // void multInt_(x,n)           //do x=x*n where x is a bigInt and n is an integer.
+  // void rightShift_(x,n)        //right shift bigInt x by n bits. (This never overflows its array).
+  // void squareMod_(x,n)         //do x=x*x  mod n for bigInts x,n
+  // void subShift_(x,y,ys)       //do x=x-(y<<(ys*bpe)). Negative answers will be 2s complement.
+  //
+  // The following functions are based on algorithms from the _Handbook of Applied Cryptography_
+  //    powMod_()           = algorithm 14.94, Montgomery exponentiation
+  //    eGCD_,inverseMod_() = algorithm 14.61, Binary extended GCD_
+  //    GCD_()              = algorothm 14.57, Lehmer's algorithm
+  //    mont_()             = algorithm 14.36, Montgomery multiplication
+  //    divide_()           = algorithm 14.20  Multiple-precision division
+  //    squareMod_()        = algorithm 14.16  Multiple-precision squaring
+  //    randTruePrime_()    = algorithm  4.62, Maurer's algorithm
+  //    millerRabin()       = algorithm  4.24, Miller-Rabin algorithm
+  //
+  // Profiling shows:
+  //     randTruePrime_() spends:
+  //         10% of its time in calls to powMod_()
+  //         85% of its time in calls to millerRabin()
+  //     millerRabin() spends:
+  //         99% of its time in calls to powMod_()   (always with a base of 2)
+  //     powMod_() spends:
+  //         94% of its time in calls to mont_()  (almost always with x==y)
+  //
+  // This suggests there are several ways to speed up this library slightly:
+  //     - convert powMod_ to use a Montgomery form of k-ary window (or maybe a Montgomery form of sliding window)
+  //         -- this should especially focus on being fast when raising 2 to a power mod n
+  //     - convert randTruePrime_() to use a minimum r of 1/3 instead of 1/2 with the appropriate change to the test
+  //     - tune the parameters in randTruePrime_(), including c, m, and recLimit
+  //     - speed up the single loop in mont_() that takes 95% of the runtime, perhaps by reducing checking
+  //       within the loop when all the parameters are the same length.
+  //
+  // There are several ideas that look like they wouldn't help much at all:
+  //     - replacing trial division in randTruePrime_() with a sieve (that speeds up something taking almost no time anyway)
+  //     - increase bpe from 15 to 30 (that would help if we had a 32*32->64 multiplier, but not with JavaScript's 32*32->32)
+  //     - speeding up mont_(x,y,n,np) when x==y by doing a non-modular, non-Montgomery square
+  //       followed by a Montgomery reduction.  The intermediate answer will be twice as long as x, so that
+  //       method would be slower.  This is unfortunate because the code currently spends almost all of its time
+  //       doing mont_(x,x,...), both for randTruePrime_() and powMod_().  A faster method for Montgomery squaring
+  //       would have a large impact on the speed of randTruePrime_() and powMod_().  HAC has a couple of poorly-worded
+  //       sentences that seem to imply it's faster to do a non-modular square followed by a single
+  //       Montgomery reduction, but that's obviously wrong.
+  ////////////////////////////////////////////////////////////////////////////////////////
+
+  //globals
+
+  // The number of significant bits in the fraction of a JavaScript
+  // floating-point number is 52, independent of platform.
+  // See: https://github.com/arlolra/otr/issues/41
+
+  var bpe = 26;          // bits stored per array element
+  var radix = 1 << bpe;  // equals 2^bpe
+  var mask = radix - 1;  // AND this with an array element to chop it down to bpe bits
+
+  //the digits for converting to different bases
+  var digitsStr='0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz_=!@#$%^&*()[]{}|;:,.<>/?`~ \\\'\"+-';
+
+  var one=int2bigInt(1,1,1);     //constant used in powMod_()
+
+  //the following global variables are scratchpad memory to 
+  //reduce dynamic memory allocation in the inner loop
+  var t=new Array(0);
+  var ss=t;       //used in mult_()
+  var s0=t;       //used in multMod_(), squareMod_()
+  var s1=t;       //used in powMod_(), multMod_(), squareMod_()
+  var s2=t;       //used in powMod_(), multMod_()
+  var s3=t;       //used in powMod_()
+  var s4=t, s5=t; //used in mod_()
+  var s6=t;       //used in bigInt2str()
+  var s7=t;       //used in powMod_()
+  var T=t;        //used in GCD_()
+  var sa=t;       //used in mont_()
+  var mr_x1=t, mr_r=t, mr_a=t;                                      //used in millerRabin()
+  var eg_v=t, eg_u=t, eg_A=t, eg_B=t, eg_C=t, eg_D=t;               //used in eGCD_(), inverseMod_()
+  var md_q1=t, md_q2=t, md_q3=t, md_r=t, md_r1=t, md_r2=t, md_tt=t; //used in mod_()
+
+  var primes=t, pows=t, s_i=t, s_i2=t, s_R=t, s_rm=t, s_q=t, s_n1=t;
+  var s_a=t, s_r2=t, s_n=t, s_b=t, s_d=t, s_x1=t, s_x2=t, s_aa=t; //used in randTruePrime_()
+    
+  var rpprb=t; //used in randProbPrimeRounds() (which also uses "primes")
+
+  ////////////////////////////////////////////////////////////////////////////////////////
+
+
+  //return array of all primes less than integer n
+  function findPrimes(n) {
+    var i,s,p,ans;
+    s=new Array(n);
+    for (i=0;i<n;i++)
+      s[i]=0;
+    s[0]=2;
+    p=0;    //first p elements of s are primes, the rest are a sieve
+    for(;s[p]<n;) {                  //s[p] is the pth prime
+      for(i=s[p]*s[p]; i<n; i+=s[p]) //mark multiples of s[p]
+        s[i]=1;
+      p++;
+      s[p]=s[p-1]+1;
+      for(; s[p]<n && s[s[p]]; s[p]++); //find next prime (where s[p]==0)
+    }
+    ans=new Array(p);
+    for(i=0;i<p;i++)
+      ans[i]=s[i];
+    return ans;
+  }
+
+
+  //does a single round of Miller-Rabin base b consider x to be a possible prime?
+  //x is a bigInt, and b is an integer, with b<x
+  function millerRabinInt(x,b) {
+    if (mr_x1.length!=x.length) {
+      mr_x1=dup(x);
+      mr_r=dup(x);
+      mr_a=dup(x);
+    }
+
+    copyInt_(mr_a,b);
+    return millerRabin(x,mr_a);
+  }
+
+  //does a single round of Miller-Rabin base b consider x to be a possible prime?
+  //x and b are bigInts with b<x
+  function millerRabin(x,b) {
+    var i,j,k,s;
+
+    if (mr_x1.length!=x.length) {
+      mr_x1=dup(x);
+      mr_r=dup(x);
+      mr_a=dup(x);
+    }
+
+    copy_(mr_a,b);
+    copy_(mr_r,x);
+    copy_(mr_x1,x);
+
+    addInt_(mr_r,-1);
+    addInt_(mr_x1,-1);
+
+    //s=the highest power of two that divides mr_r
+
+    /*
+    k=0;
+    for (i=0;i<mr_r.length;i++)
+      for (j=1;j<mask;j<<=1)
+        if (x[i] & j) {
+          s=(k<mr_r.length+bpe ? k : 0); 
+           i=mr_r.length;
+           j=mask;
+        } else
+          k++;
+    */
+
+    /* http://www.javascripter.net/math/primes/millerrabinbug-bigint54.htm */
+    if (isZero(mr_r)) return 0;
+    for (k=0; mr_r[k]==0; k++);
+    for (i=1,j=2; mr_r[k]%j==0; j*=2,i++ );
+    s = k*bpe + i - 1;
+    /* end */
+
+    if (s)                
+      rightShift_(mr_r,s);
+
+    powMod_(mr_a,mr_r,x);
+
+    if (!equalsInt(mr_a,1) && !equals(mr_a,mr_x1)) {
+      j=1;
+      while (j<=s-1 && !equals(mr_a,mr_x1)) {
+        squareMod_(mr_a,x);
+        if (equalsInt(mr_a,1)) {
+          return 0;
+        }
+        j++;
+      }
+      if (!equals(mr_a,mr_x1)) {
+        return 0;
+      }
+    }
+    return 1;  
+  }
+
+  //returns how many bits long the bigInt is, not counting leading zeros.
+  function bitSize(x) {
+    var j,z,w;
+    for (j=x.length-1; (x[j]==0) && (j>0); j--);
+    for (z=0,w=x[j]; w; (w>>=1),z++);
+    z+=bpe*j;
+    return z;
+  }
+
+  //return a copy of x with at least n elements, adding leading zeros if needed
+  function expand(x,n) {
+    var ans=int2bigInt(0,(x.length>n ? x.length : n)*bpe,0);
+    copy_(ans,x);
+    return ans;
+  }
+
+  //return a k-bit true random prime using Maurer's algorithm.
+  function randTruePrime(k) {
+    var ans=int2bigInt(0,k,0);
+    randTruePrime_(ans,k);
+    return trim(ans,1);
+  }
+
+  //return a k-bit random probable prime with probability of error < 2^-80
+  function randProbPrime(k) {
+    if (k>=600) return randProbPrimeRounds(k,2); //numbers from HAC table 4.3
+    if (k>=550) return randProbPrimeRounds(k,4);
+    if (k>=500) return randProbPrimeRounds(k,5);
+    if (k>=400) return randProbPrimeRounds(k,6);
+    if (k>=350) return randProbPrimeRounds(k,7);
+    if (k>=300) return randProbPrimeRounds(k,9);
+    if (k>=250) return randProbPrimeRounds(k,12); //numbers from HAC table 4.4
+    if (k>=200) return randProbPrimeRounds(k,15);
+    if (k>=150) return randProbPrimeRounds(k,18);
+    if (k>=100) return randProbPrimeRounds(k,27);
+                return randProbPrimeRounds(k,40); //number from HAC remark 4.26 (only an estimate)
+  }
+
+  //return a k-bit probable random prime using n rounds of Miller Rabin (after trial division with small primes)
+  function randProbPrimeRounds(k,n) {
+    var ans, i, divisible, B; 
+    B=30000;  //B is largest prime to use in trial division
+    ans=int2bigInt(0,k,0);
+    
+    //optimization: try larger and smaller B to find the best limit.
+    
+    if (primes.length==0)
+      primes=findPrimes(30000);  //check for divisibility by primes <=30000
+
+    if (rpprb.length!=ans.length)
+      rpprb=dup(ans);
+
+    for (;;) { //keep trying random values for ans until one appears to be prime
+      //optimization: pick a random number times L=2*3*5*...*p, plus a 
+      //   random element of the list of all numbers in [0,L) not divisible by any prime up to p.
+      //   This can reduce the amount of random number generation.
+      
+      randBigInt_(ans,k,0); //ans = a random odd number to check
+      ans[0] |= 1; 
+      divisible=0;
+    
+      //check ans for divisibility by small primes up to B
+      for (i=0; (i<primes.length) && (primes[i]<=B); i++)
+        if (modInt(ans,primes[i])==0 && !equalsInt(ans,primes[i])) {
+          divisible=1;
+          break;
+        }      
+      
+      //optimization: change millerRabin so the base can be bigger than the number being checked, then eliminate the while here.
+      
+      //do n rounds of Miller Rabin, with random bases less than ans
+      for (i=0; i<n && !divisible; i++) {
+        randBigInt_(rpprb,k,0);
+        while(!greater(ans,rpprb)) //pick a random rpprb that's < ans
+          randBigInt_(rpprb,k,0);
+        if (!millerRabin(ans,rpprb))
+          divisible=1;
+      }
+      
+      if(!divisible)
+        return ans;
+    }  
+  }
+
+  //return a new bigInt equal to (x mod n) for bigInts x and n.
+  function mod(x,n) {
+    var ans=dup(x);
+    mod_(ans,n);
+    return trim(ans,1);
+  }
+
+  //return (x+n) where x is a bigInt and n is an integer.
+  function addInt(x,n) {
+    var ans=expand(x,x.length+1);
+    addInt_(ans,n);
+    return trim(ans,1);
+  }
+
+  //return x*y for bigInts x and y. This is faster when y<x.
+  function mult(x,y) {
+    var ans=expand(x,x.length+y.length);
+    mult_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x**y mod n) where x,y,n are bigInts and ** is exponentiation.  0**0=1. Faster for odd n.
+  function powMod(x,y,n) {
+    var ans=expand(x,n.length);  
+    powMod_(ans,trim(y,2),trim(n,2),0);  //this should work without the trim, but doesn't
+    return trim(ans,1);
+  }
+
+  //return (x-y) for bigInts x and y.  Negative answers will be 2s complement
+  function sub(x,y) {
+    var ans=expand(x,(x.length>y.length ? x.length+1 : y.length+1)); 
+    sub_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x+y) for bigInts x and y.  
+  function add(x,y) {
+    var ans=expand(x,(x.length>y.length ? x.length+1 : y.length+1)); 
+    add_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x**(-1) mod n) for bigInts x and n.  If no inverse exists, it returns null
+  function inverseMod(x,n) {
+    var ans=expand(x,n.length); 
+    var s;
+    s=inverseMod_(ans,n);
+    return s ? trim(ans,1) : null;
+  }
+
+  //return (x*y mod n) for bigInts x,y,n.  For greater speed, let y<x.
+  function multMod(x,y,n) {
+    var ans=expand(x,n.length);
+    multMod_(ans,y,n);
+    return trim(ans,1);
+  }
+
+  //generate a k-bit true random prime using Maurer's algorithm,
+  //and put it into ans.  The bigInt ans must be large enough to hold it.
+  function randTruePrime_(ans,k) {
+    var c,w,m,pm,dd,j,r,B,divisible,z,zz,recSize,recLimit;
+
+    if (primes.length==0)
+      primes=findPrimes(30000);  //check for divisibility by primes <=30000
+
+    if (pows.length==0) {
+      pows=new Array(512);
+      for (j=0;j<512;j++) {
+        pows[j]=Math.pow(2,j/511.0-1.0);
+      }
+    }
+
+    //c and m should be tuned for a particular machine and value of k, to maximize speed
+    c=0.1;  //c=0.1 in HAC
+    m=20;   //generate this k-bit number by first recursively generating a number that has between k/2 and k-m bits
+    recLimit=20; //stop recursion when k <=recLimit.  Must have recLimit >= 2
+
+    if (s_i2.length!=ans.length) {
+      s_i2=dup(ans);
+      s_R =dup(ans);
+      s_n1=dup(ans);
+      s_r2=dup(ans);
+      s_d =dup(ans);
+      s_x1=dup(ans);
+      s_x2=dup(ans);
+      s_b =dup(ans);
+      s_n =dup(ans);
+      s_i =dup(ans);
+      s_rm=dup(ans);
+      s_q =dup(ans);
+      s_a =dup(ans);
+      s_aa=dup(ans);
+    }
+
+    if (k <= recLimit) {  //generate small random primes by trial division up to its square root
+      pm=(1<<((k+2)>>1))-1; //pm is binary number with all ones, just over sqrt(2^k)
+      copyInt_(ans,0);
+      for (dd=1;dd;) {
+        dd=0;
+        ans[0]= 1 | (1<<(k-1)) | randomBitInt(k);  //random, k-bit, odd integer, with msb 1
+        for (j=1;(j<primes.length) && ((primes[j]&pm)==primes[j]);j++) { //trial division by all primes 3...sqrt(2^k)
+          if (0==(ans[0]%primes[j])) {
+            dd=1;
+            break;
+          }
+        }
+      }
+      carry_(ans);
+      return;
+    }
+
+    B=c*k*k;    //try small primes up to B (or all the primes[] array if the largest is less than B).
+    if (k>2*m)  //generate this k-bit number by first recursively generating a number that has between k/2 and k-m bits
+      for (r=1; k-k*r<=m; )
+        r=pows[randomBitInt(9)];   //r=Math.pow(2,Math.random()-1);
+    else
+      r=0.5;
+
+    //simulation suggests the more complex algorithm using r=.333 is only slightly faster.
+
+    recSize=Math.floor(r*k)+1;
+
+    randTruePrime_(s_q,recSize);
+    copyInt_(s_i2,0);
+    s_i2[Math.floor((k-2)/bpe)] |= (1<<((k-2)%bpe));   //s_i2=2^(k-2)
+    divide_(s_i2,s_q,s_i,s_rm);                        //s_i=floor((2^(k-1))/(2q))
+
+    z=bitSize(s_i);
+
+    for (;;) {
+      for (;;) {  //generate z-bit numbers until one falls in the range [0,s_i-1]
+        randBigInt_(s_R,z,0);
+        if (greater(s_i,s_R))
+          break;
+      }                //now s_R is in the range [0,s_i-1]
+      addInt_(s_R,1);  //now s_R is in the range [1,s_i]
+      add_(s_R,s_i);   //now s_R is in the range [s_i+1,2*s_i]
+
+      copy_(s_n,s_q);
+      mult_(s_n,s_R); 
+      multInt_(s_n,2);
+      addInt_(s_n,1);    //s_n=2*s_R*s_q+1
+      
+      copy_(s_r2,s_R);
+      multInt_(s_r2,2);  //s_r2=2*s_R
+
+      //check s_n for divisibility by small primes up to B
+      for (divisible=0,j=0; (j<primes.length) && (primes[j]<B); j++)
+        if (modInt(s_n,primes[j])==0 && !equalsInt(s_n,primes[j])) {
+          divisible=1;
+          break;
+        }      
+
+      if (!divisible)    //if it passes small primes check, then try a single Miller-Rabin base 2
+        if (!millerRabinInt(s_n,2)) //this line represents 75% of the total runtime for randTruePrime_ 
+          divisible=1;
+
+      if (!divisible) {  //if it passes that test, continue checking s_n
+        addInt_(s_n,-3);
+        for (j=s_n.length-1;(s_n[j]==0) && (j>0); j--);  //strip leading zeros
+        for (zz=0,w=s_n[j]; w; (w>>=1),zz++);
+        zz+=bpe*j;                             //zz=number of bits in s_n, ignoring leading zeros
+        for (;;) {  //generate z-bit numbers until one falls in the range [0,s_n-1]
+          randBigInt_(s_a,zz,0);
+          if (greater(s_n,s_a))
+            break;
+        }                //now s_a is in the range [0,s_n-1]
+        addInt_(s_n,3);  //now s_a is in the range [0,s_n-4]
+        addInt_(s_a,2);  //now s_a is in the range [2,s_n-2]
+        copy_(s_b,s_a);
+        copy_(s_n1,s_n);
+        addInt_(s_n1,-1);
+        powMod_(s_b,s_n1,s_n);   //s_b=s_a^(s_n-1) modulo s_n
+        addInt_(s_b,-1);
+        if (isZero(s_b)) {
+          copy_(s_b,s_a);
+          powMod_(s_b,s_r2,s_n);
+          addInt_(s_b,-1);
+          copy_(s_aa,s_n);
+          copy_(s_d,s_b);
+          GCD_(s_d,s_n);  //if s_b and s_n are relatively prime, then s_n is a prime
+          if (equalsInt(s_d,1)) {
+            copy_(ans,s_aa);
+            return;     //if we've made it this far, then s_n is absolutely guaranteed to be prime
+          }
+        }
+      }
+    }
+  }
+
+  //Return an n-bit random BigInt (n>=1).  If s=1, then the most significant of those n bits is set to 1.
+  function randBigInt(n,s) {
+    var a,b;
+    a=Math.floor((n-1)/bpe)+2; //# array elements to hold the BigInt with a leading 0 element
+    b=int2bigInt(0,0,a);
+    randBigInt_(b,n,s);
+    return b;
+  }
+
+  //Set b to an n-bit random BigInt.  If s=1, then the most significant of those n bits is set to 1.
+  //Array b must be big enough to hold the result. Must have n>=1
+  function randBigInt_(b,n,s) {
+    var i,a;
+    for (i=0;i<b.length;i++)
+      b[i]=0;
+    a=Math.floor((n-1)/bpe)+1; //# array elements to hold the BigInt
+    for (i=0;i<a;i++) {
+      b[i]=randomBitInt(bpe);
+    }
+    b[a-1] &= (2<<((n-1)%bpe))-1;
+    if (s==1)
+      b[a-1] |= (1<<((n-1)%bpe));
+  }
+
+  //Return the greatest common divisor of bigInts x and y (each with same number of elements).
+  function GCD(x,y) {
+    var xc,yc;
+    xc=dup(x);
+    yc=dup(y);
+    GCD_(xc,yc);
+    return xc;
+  }
+
+  //set x to the greatest common divisor of bigInts x and y (each with same number of elements).
+  //y is destroyed.
+  function GCD_(x,y) {
+    var i,xp,yp,A,B,C,D,q,sing,qp;
+    if (T.length!=x.length)
+      T=dup(x);
+
+    sing=1;
+    while (sing) { //while y has nonzero elements other than y[0]
+      sing=0;
+      for (i=1;i<y.length;i++) //check if y has nonzero elements other than 0
+        if (y[i]) {
+          sing=1;
+          break;
+        }
+      if (!sing) break; //quit when y all zero elements except possibly y[0]
+
+      for (i=x.length;!x[i] && i>=0;i--);  //find most significant element of x
+      xp=x[i];
+      yp=y[i];
+      A=1; B=0; C=0; D=1;
+      while ((yp+C) && (yp+D)) {
+        q =Math.floor((xp+A)/(yp+C));
+        qp=Math.floor((xp+B)/(yp+D));
+        if (q!=qp)
+          break;
+        t= A-q*C;   A=C;   C=t;    //  do (A,B,xp, C,D,yp) = (C,D,yp, A,B,xp) - q*(0,0,0, C,D,yp)      
+        t= B-q*D;   B=D;   D=t;
+        t=xp-q*yp; xp=yp; yp=t;
+      }
+      if (B) {
+        copy_(T,x);
+        linComb_(x,y,A,B); //x=A*x+B*y
+        linComb_(y,T,D,C); //y=D*y+C*T
+      } else {
+        mod_(x,y);
+        copy_(T,x);
+        copy_(x,y);
+        copy_(y,T);
+      } 
+    }
+    if (y[0]==0)
+      return;
+    t=modInt(x,y[0]);
+    copyInt_(x,y[0]);
+    y[0]=t;
+    while (y[0]) {
+      x[0]%=y[0];
+      t=x[0]; x[0]=y[0]; y[0]=t;
+    }
+  }
+
+  //do x=x**(-1) mod n, for bigInts x and n.
+  //If no inverse exists, it sets x to zero and returns 0, else it returns 1.
+  //The x array must be at least as large as the n array.
+  function inverseMod_(x,n) {
+    var k=1+2*Math.max(x.length,n.length);
+
+    if(!(x[0]&1)  && !(n[0]&1)) {  //if both inputs are even, then inverse doesn't exist
+      copyInt_(x,0);
+      return 0;
+    }
+
+    if (eg_u.length!=k) {
+      eg_u=new Array(k);
+      eg_v=new Array(k);
+      eg_A=new Array(k);
+      eg_B=new Array(k);
+      eg_C=new Array(k);
+      eg_D=new Array(k);
+    }
+
+    copy_(eg_u,x);
+    copy_(eg_v,n);
+    copyInt_(eg_A,1);
+    copyInt_(eg_B,0);
+    copyInt_(eg_C,0);
+    copyInt_(eg_D,1);
+    for (;;) {
+      while(!(eg_u[0]&1)) {  //while eg_u is even
+        halve_(eg_u);
+        if (!(eg_A[0]&1) && !(eg_B[0]&1)) { //if eg_A==eg_B==0 mod 2
+          halve_(eg_A);
+          halve_(eg_B);      
+        } else {
+          add_(eg_A,n);  halve_(eg_A);
+          sub_(eg_B,x);  halve_(eg_B);
+        }
+      }
+
+      while (!(eg_v[0]&1)) {  //while eg_v is even
+        halve_(eg_v);
+        if (!(eg_C[0]&1) && !(eg_D[0]&1)) { //if eg_C==eg_D==0 mod 2
+          halve_(eg_C);
+          halve_(eg_D);      
+        } else {
+          add_(eg_C,n);  halve_(eg_C);
+          sub_(eg_D,x);  halve_(eg_D);
+        }
+      }
+
+      if (!greater(eg_v,eg_u)) { //eg_v <= eg_u
+        sub_(eg_u,eg_v);
+        sub_(eg_A,eg_C);
+        sub_(eg_B,eg_D);
+      } else {                   //eg_v > eg_u
+        sub_(eg_v,eg_u);
+        sub_(eg_C,eg_A);
+        sub_(eg_D,eg_B);
+      }
+
+      if (equalsInt(eg_u,0)) {
+        while (negative(eg_C)) //make sure answer is nonnegative
+          add_(eg_C,n);
+        copy_(x,eg_C);
+
+        if (!equalsInt(eg_v,1)) { //if GCD_(x,n)!=1, then there is no inverse
+          copyInt_(x,0);
+          return 0;
+        }
+        return 1;
+      }
+    }
+  }
+
+  //return x**(-1) mod n, for integers x and n.  Return 0 if there is no inverse
+  function inverseModInt(x,n) {
+    var a=1,b=0,t;
+    for (;;) {
+      if (x==1) return a;
+      if (x==0) return 0;
+      b-=a*Math.floor(n/x);
+      n%=x;
+
+      if (n==1) return b; //to avoid negatives, change this b to n-b, and each -= to +=
+      if (n==0) return 0;
+      a-=b*Math.floor(x/n);
+      x%=n;
+    }
+  }
+
+  //this deprecated function is for backward compatibility only. 
+  function inverseModInt_(x,n) {
+     return inverseModInt(x,n);
+  }
+
+
+  //Given positive bigInts x and y, change the bigints v, a, and b to positive bigInts such that:
+  //     v = GCD_(x,y) = a*x-b*y
+  //The bigInts v, a, b, must have exactly as many elements as the larger of x and y.
+  function eGCD_(x,y,v,a,b) {
+    var g=0;
+    var k=Math.max(x.length,y.length);
+    if (eg_u.length!=k) {
+      eg_u=new Array(k);
+      eg_A=new Array(k);
+      eg_B=new Array(k);
+      eg_C=new Array(k);
+      eg_D=new Array(k);
+    }
+    while(!(x[0]&1)  && !(y[0]&1)) {  //while x and y both even
+      halve_(x);
+      halve_(y);
+      g++;
+    }
+    copy_(eg_u,x);
+    copy_(v,y);
+    copyInt_(eg_A,1);
+    copyInt_(eg_B,0);
+    copyInt_(eg_C,0);
+    copyInt_(eg_D,1);
+    for (;;) {
+      while(!(eg_u[0]&1)) {  //while u is even
+        halve_(eg_u);
+        if (!(eg_A[0]&1) && !(eg_B[0]&1)) { //if A==B==0 mod 2
+          halve_(eg_A);
+          halve_(eg_B);      
+        } else {
+          add_(eg_A,y);  halve_(eg_A);
+          sub_(eg_B,x);  halve_(eg_B);
+        }
+      }
+
+      while (!(v[0]&1)) {  //while v is even
+        halve_(v);
+        if (!(eg_C[0]&1) && !(eg_D[0]&1)) { //if C==D==0 mod 2
+          halve_(eg_C);
+          halve_(eg_D);      
+        } else {
+          add_(eg_C,y);  halve_(eg_C);
+          sub_(eg_D,x);  halve_(eg_D);
+        }
+      }
+
+      if (!greater(v,eg_u)) { //v<=u
+        sub_(eg_u,v);
+        sub_(eg_A,eg_C);
+        sub_(eg_B,eg_D);
+      } else {                //v>u
+        sub_(v,eg_u);
+        sub_(eg_C,eg_A);
+        sub_(eg_D,eg_B);
+      }
+      if (equalsInt(eg_u,0)) {
+        while (negative(eg_C)) {   //make sure a (C) is nonnegative
+          add_(eg_C,y);
+          sub_(eg_D,x);
+        }
+        multInt_(eg_D,-1);  ///make sure b (D) is nonnegative
+        copy_(a,eg_C);
+        copy_(b,eg_D);
+        leftShift_(v,g);
+        return;
+      }
+    }
+  }
+
+
+  //is bigInt x negative?
+  function negative(x) {
+    return ((x[x.length-1]>>(bpe-1))&1);
+  }
+
+
+  //is (x << (shift*bpe)) > y?
+  //x and y are nonnegative bigInts
+  //shift is a nonnegative integer
+  function greaterShift(x,y,shift) {
+    var i, kx=x.length, ky=y.length;
+    var k=((kx+shift)<ky) ? (kx+shift) : ky;
+    for (i=ky-1-shift; i<kx && i>=0; i++) 
+      if (x[i]>0)
+        return 1; //if there are nonzeros in x to the left of the first column of y, then x is bigger
+    for (i=kx-1+shift; i<ky; i++)
+      if (y[i]>0)
+        return 0; //if there are nonzeros in y to the left of the first column of x, then x is not bigger
+    for (i=k-1; i>=shift; i--)
+      if      (x[i-shift]>y[i]) return 1;
+      else if (x[i-shift]<y[i]) return 0;
+    return 0;
+  }
+
+  //is x > y? (x and y both nonnegative)
+  function greater(x,y) {
+    var i;
+    var k=(x.length<y.length) ? x.length : y.length;
+
+    for (i=x.length;i<y.length;i++)
+      if (y[i])
+        return 0;  //y has more digits
+
+    for (i=y.length;i<x.length;i++)
+      if (x[i])
+        return 1;  //x has more digits
+
+    for (i=k-1;i>=0;i--)
+      if (x[i]>y[i])
+        return 1;
+      else if (x[i]<y[i])
+        return 0;
+    return 0;
+  }
+
+  //divide x by y giving quotient q and remainder r.  (q=floor(x/y),  r=x mod y).  All 4 are bigints.
+  //x must have at least one leading zero element.
+  //y must be nonzero.
+  //q and r must be arrays that are exactly the same length as x. (Or q can have more).
+  //Must have x.length >= y.length >= 2.
+  function divide_(x,y,q,r) {
+    var kx, ky;
+    var i,j,y1,y2,c,a,b;
+    copy_(r,x);
+    for (ky=y.length;y[ky-1]==0;ky--); //ky is number of elements in y, not including leading zeros
+
+    //normalize: ensure the most significant element of y has its highest bit set  
+    b=y[ky-1];
+    for (a=0; b; a++)
+      b>>=1;  
+    a=bpe-a;  //a is how many bits to shift so that the high order bit of y is leftmost in its array element
+    leftShift_(y,a);  //multiply both by 1<<a now, then divide both by that at the end
+    leftShift_(r,a);
+
+    //Rob Visser discovered a bug: the following line was originally just before the normalization.
+    for (kx=r.length;r[kx-1]==0 && kx>ky;kx--); //kx is number of elements in normalized x, not including leading zeros
+
+    copyInt_(q,0);                      // q=0
+    while (!greaterShift(y,r,kx-ky)) {  // while (leftShift_(y,kx-ky) <= r) {
+      subShift_(r,y,kx-ky);             //   r=r-leftShift_(y,kx-ky)
+      q[kx-ky]++;                       //   q[kx-ky]++;
+    }                                   // }
+
+    for (i=kx-1; i>=ky; i--) {
+      if (r[i]==y[ky-1])
+        q[i-ky]=mask;
+      else
+        q[i-ky]=Math.floor((r[i]*radix+r[i-1])/y[ky-1]);
+
+      //The following for(;;) loop is equivalent to the commented while loop, 
+      //except that the uncommented version avoids overflow.
+      //The commented loop comes from HAC, which assumes r[-1]==y[-1]==0
+      //  while (q[i-ky]*(y[ky-1]*radix+y[ky-2]) > r[i]*radix*radix+r[i-1]*radix+r[i-2])
+      //    q[i-ky]--;    
+      for (;;) {
+        y2=(ky>1 ? y[ky-2] : 0)*q[i-ky];
+        c=y2;
+        y2=y2 & mask;
+        c = (c - y2) / radix;
+        y1=c+q[i-ky]*y[ky-1];
+        c=y1;
+        y1=y1 & mask;
+        c = (c - y1) / radix;
+
+        if (c==r[i] ? y1==r[i-1] ? y2>(i>1 ? r[i-2] : 0) : y1>r[i-1] : c>r[i]) 
+          q[i-ky]--;
+        else
+          break;
+      }
+
+      linCombShift_(r,y,-q[i-ky],i-ky);    //r=r-q[i-ky]*leftShift_(y,i-ky)
+      if (negative(r)) {
+        addShift_(r,y,i-ky);         //r=r+leftShift_(y,i-ky)
+        q[i-ky]--;
+      }
+    }
+
+    rightShift_(y,a);  //undo the normalization step
+    rightShift_(r,a);  //undo the normalization step
+  }
+
+  //do carries and borrows so each element of the bigInt x fits in bpe bits.
+  function carry_(x) {
+    var i,k,c,b;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i];
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+    }
+  }
+
+  //return x mod n for bigInt x and integer n.
+  function modInt(x,n) {
+    var i,c=0;
+    for (i=x.length-1; i>=0; i--)
+      c=(c*radix+x[i])%n;
+    return c;
+  }
+
+  //convert the integer t into a bigInt with at least the given number of bits.
+  //the returned array stores the bigInt in bpe-bit chunks, little endian (buff[0] is least significant word)
+  //Pad the array with leading zeros so that it has at least minSize elements.
+  //There will always be at least one leading 0 element.
+  function int2bigInt(t,bits,minSize) {   
+    var i,k, buff;
+    k=Math.ceil(bits/bpe)+1;
+    k=minSize>k ? minSize : k;
+    buff=new Array(k);
+    copyInt_(buff,t);
+    return buff;
+  }
+
+  //return the bigInt given a string representation in a given base.  
+  //Pad the array with leading zeros so that it has at least minSize elements.
+  //If base=-1, then it reads in a space-separated list of array elements in decimal.
+  //The array will always have at least one leading zero, unless base=-1.
+  function str2bigInt(s,base,minSize) {
+    var d, i, j, x, y, kk;
+    var k=s.length;
+    if (base==-1) { //comma-separated list of array elements in decimal
+      x=new Array(0);
+      for (;;) {
+        y=new Array(x.length+1);
+        for (i=0;i<x.length;i++)
+          y[i+1]=x[i];
+        y[0]=parseInt(s,10);
+        x=y;
+        d=s.indexOf(',',0);
+        if (d<1) 
+          break;
+        s=s.substring(d+1);
+        if (s.length==0)
+          break;
+      }
+      if (x.length<minSize) {
+        y=new Array(minSize);
+        copy_(y,x);
+        return y;
+      }
+      return x;
+    }
+
+    // log2(base)*k
+    var bb = base, p = 0;
+    var b = base == 1 ? k : 0;
+    while (bb > 1) {
+      if (bb & 1) p = 1;
+      b += k;
+      bb >>= 1;
+    }
+    b += p*k;
+
+    x=int2bigInt(0,b,0);
+    for (i=0;i<k;i++) {
+      d=digitsStr.indexOf(s.substring(i,i+1),0);
+      if (base<=36 && d>=36)  //convert lowercase to uppercase if base<=36
+        d-=26;
+      if (d>=base || d<0) {   //stop at first illegal character
+        break;
+      }
+      multInt_(x,base);
+      addInt_(x,d);
+    }
+
+    for (k=x.length;k>0 && !x[k-1];k--); //strip off leading zeros
+    k=minSize>k+1 ? minSize : k+1;
+    y=new Array(k);
+    kk=k<x.length ? k : x.length;
+    for (i=0;i<kk;i++)
+      y[i]=x[i];
+    for (;i<k;i++)
+      y[i]=0;
+    return y;
+  }
+
+  //is bigint x equal to integer y?
+  //y must have less than bpe bits
+  function equalsInt(x,y) {
+    var i;
+    if (x[0]!=y)
+      return 0;
+    for (i=1;i<x.length;i++)
+      if (x[i])
+        return 0;
+    return 1;
+  }
+
+  //are bigints x and y equal?
+  //this works even if x and y are different lengths and have arbitrarily many leading zeros
+  function equals(x,y) {
+    var i;
+    var k=x.length<y.length ? x.length : y.length;
+    for (i=0;i<k;i++)
+      if (x[i]!=y[i])
+        return 0;
+    if (x.length>y.length) {
+      for (;i<x.length;i++)
+        if (x[i])
+          return 0;
+    } else {
+      for (;i<y.length;i++)
+        if (y[i])
+          return 0;
+    }
+    return 1;
+  }
+
+  //is the bigInt x equal to zero?
+  function isZero(x) {
+    var i;
+    for (i=0;i<x.length;i++)
+      if (x[i])
+        return 0;
+    return 1;
+  }
+
+  //convert a bigInt into a string in a given base, from base 2 up to base 95.
+  //Base -1 prints the contents of the array representing the number.
+  function bigInt2str(x,base) {
+    var i,t,s="";
+
+    if (s6.length!=x.length) 
+      s6=dup(x);
+    else
+      copy_(s6,x);
+
+    if (base==-1) { //return the list of array contents
+      for (i=x.length-1;i>0;i--)
+        s+=x[i]+',';
+      s+=x[0];
+    }
+    else { //return it in the given base
+      while (!isZero(s6)) {
+        t=divInt_(s6,base);  //t=s6 % base; s6=floor(s6/base);
+        s=digitsStr.substring(t,t+1)+s;
+      }
+    }
+    if (s.length==0)
+      s="0";
+    return s;
+  }
+
+  //returns a duplicate of bigInt x
+  function dup(x) {
+    var i, buff;
+    buff=new Array(x.length);
+    copy_(buff,x);
+    return buff;
+  }
+
+  //do x=y on bigInts x and y.  x must be an array at least as big as y (not counting the leading zeros in y).
+  function copy_(x,y) {
+    var i;
+    var k=x.length<y.length ? x.length : y.length;
+    for (i=0;i<k;i++)
+      x[i]=y[i];
+    for (i=k;i<x.length;i++)
+      x[i]=0;
+  }
+
+  //do x=y on bigInt x and integer y.  
+  function copyInt_(x,n) {
+    var i,c;
+    for (c=n,i=0;i<x.length;i++) {
+      x[i]=c & mask;
+      c>>=bpe;
+    }
+  }
+
+  //do x=x+n where x is a bigInt and n is an integer.
+  //x must be large enough to hold the result.
+  function addInt_(x,n) {
+    var i,k,c,b;
+    x[0]+=n;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i];
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+      if (!c) return; //stop carrying as soon as the carry is zero
+    }
+  }
+
+  //right shift bigInt x by n bits.
+  function rightShift_(x,n) {
+    var i;
+    var k=Math.floor(n/bpe);
+    if (k) {
+      for (i=0;i<x.length-k;i++) //right shift x by k elements
+        x[i]=x[i+k];
+      for (;i<x.length;i++)
+        x[i]=0;
+      n%=bpe;
+    }
+    for (i=0;i<x.length-1;i++) {
+      x[i]=mask & ((x[i+1]<<(bpe-n)) | (x[i]>>n));
+    }
+    x[i]>>=n;
+  }
+
+  //do x=floor(|x|/2)*sgn(x) for bigInt x in 2's complement
+  function halve_(x) {
+    var i;
+    for (i=0;i<x.length-1;i++) {
+      x[i]=mask & ((x[i+1]<<(bpe-1)) | (x[i]>>1));
+    }
+    x[i]=(x[i]>>1) | (x[i] & (radix>>1));  //most significant bit stays the same
+  }
+
+  //left shift bigInt x by n bits.
+  function leftShift_(x,n) {
+    var i;
+    var k=Math.floor(n/bpe);
+    if (k) {
+      for (i=x.length; i>=k; i--) //left shift x by k elements
+        x[i]=x[i-k];
+      for (;i>=0;i--)
+        x[i]=0;  
+      n%=bpe;
+    }
+    if (!n)
+      return;
+    for (i=x.length-1;i>0;i--) {
+      x[i]=mask & ((x[i]<<n) | (x[i-1]>>(bpe-n)));
+    }
+    x[i]=mask & (x[i]<<n);
+  }
+
+  //do x=x*n where x is a bigInt and n is an integer.
+  //x must be large enough to hold the result.
+  function multInt_(x,n) {
+    var i,k,c,b;
+    if (!n)
+      return;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i]*n;
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+    }
+  }
+
+  //do x=floor(x/n) for bigInt x and integer n, and return the remainder
+  function divInt_(x,n) {
+    var i,r=0,s;
+    for (i=x.length-1;i>=0;i--) {
+      s=r*radix+x[i];
+      x[i]=Math.floor(s/n);
+      r=s%n;
+    }
+    return r;
+  }
+
+  //do the linear combination x=a*x+b*y for bigInts x and y, and integers a and b.
+  //x must be large enough to hold the answer.
+  function linComb_(x,y,a,b) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    kk=x.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=a*x[i]+b*y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;i<kk;i++) {
+      c+=a*x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do the linear combination x=a*x+b*(y<<(ys*bpe)) for bigInts x and y, and integers a, b and ys.
+  //x must be large enough to hold the answer.
+  function linCombShift_(x,y,b,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]+b*y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x+(y<<(ys*bpe)) for bigInts x and y, and integers a,b and ys.
+  //x must be large enough to hold the answer.
+  function addShift_(x,y,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]+y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x-(y<<(ys*bpe)) for bigInts x and y, and integers a,b and ys.
+  //x must be large enough to hold the answer.
+  function subShift_(x,y,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]-y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x-y for bigInts x and y.
+  //x must be large enough to hold the answer.
+  //negative answers will be 2s complement
+  function sub_(x,y) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=x[i]-y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<x.length;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x+y for bigInts x and y.
+  //x must be large enough to hold the answer.
+  function add_(x,y) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=x[i]+y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<x.length;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x*y for bigInts x and y.  This is faster when y<x.
+  function mult_(x,y) {
+    var i;
+    if (ss.length!=2*x.length)
+      ss=new Array(2*x.length);
+    copyInt_(ss,0);
+    for (i=0;i<y.length;i++)
+      if (y[i])
+        linCombShift_(ss,x,y[i],i);   //ss=1*ss+y[i]*(x<<(i*bpe))
+    copy_(x,ss);
+  }
+
+  //do x=x mod n for bigInts x and n.
+  function mod_(x,n) {
+    if (s4.length!=x.length)
+      s4=dup(x);
+    else
+      copy_(s4,x);
+    if (s5.length!=x.length)
+      s5=dup(x);  
+    divide_(s4,n,s5,x);  //x = remainder of s4 / n
+  }
+
+  //do x=x*y mod n for bigInts x,y,n.
+  //for greater speed, let y<x.
+  function multMod_(x,y,n) {
+    var i;
+    if (s0.length!=2*x.length)
+      s0=new Array(2*x.length);
+    copyInt_(s0,0);
+    for (i=0;i<y.length;i++)
+      if (y[i])
+        linCombShift_(s0,x,y[i],i);   //s0=1*s0+y[i]*(x<<(i*bpe))
+    mod_(s0,n);
+    copy_(x,s0);
+  }
+
+  //do x=x*x mod n for bigInts x,n.
+  function squareMod_(x,n) {
+    var i,j,d,c,kx,kn,k;
+    for (kx=x.length; kx>0 && !x[kx-1]; kx--);  //ignore leading zeros in x
+    k=kx>n.length ? 2*kx : 2*n.length; //k=# elements in the product, which is twice the elements in the larger of x and n
+    if (s0.length!=k) 
+      s0=new Array(k);
+    copyInt_(s0,0);
+    for (i=0;i<kx;i++) {
+      c=s0[2*i]+x[i]*x[i];
+      s0[2*i]=c & mask;
+      c = (c - s0[2*i]) / radix;
+      for (j=i+1;j<kx;j++) {
+        c=s0[i+j]+2*x[i]*x[j]+c;
+        s0[i+j]=(c & mask);
+        c = (c - s0[i+j]) / radix;
+      }
+      s0[i+kx]=c;
+    }
+    mod_(s0,n);
+    copy_(x,s0);
+  }
+
+  //return x with exactly k leading zero elements
+  function trim(x,k) {
+    var i,y;
+    for (i=x.length; i>0 && !x[i-1]; i--);
+    y=new Array(i+k);
+    copy_(y,x);
+    return y;
+  }
+
+  //do x=x**y mod n, where x,y,n are bigInts and ** is exponentiation.  0**0=1.
+  //this is faster when n is odd.  x usually needs to have as many elements as n.
+  function powMod_(x,y,n) {
+    var k1,k2,kn,np;
+    if(s7.length!=n.length)
+      s7=dup(n);
+
+    //for even modulus, use a simple square-and-multiply algorithm,
+    //rather than using the more complex Montgomery algorithm.
+    if ((n[0]&1)==0) {
+      copy_(s7,x);
+      copyInt_(x,1);
+      while(!equalsInt(y,0)) {
+        if (y[0]&1)
+          multMod_(x,s7,n);
+        divInt_(y,2);
+        squareMod_(s7,n); 
+      }
+      return;
+    }
+
+    //calculate np from n for the Montgomery multiplications
+    copyInt_(s7,0);
+    for (kn=n.length;kn>0 && !n[kn-1];kn--);
+    np=radix-inverseModInt(modInt(n,radix),radix);
+    s7[kn]=1;
+    multMod_(x ,s7,n);   // x = x * 2**(kn*bp) mod n
+
+    if (s3.length!=x.length)
+      s3=dup(x);
+    else
+      copy_(s3,x);
+
+    for (k1=y.length-1;k1>0 & !y[k1]; k1--);  //k1=first nonzero element of y
+    if (y[k1]==0) {  //anything to the 0th power is 1
+      copyInt_(x,1);
+      return;
+    }
+    for (k2=1<<(bpe-1);k2 && !(y[k1] & k2); k2>>=1);  //k2=position of first 1 bit in y[k1]
+    for (;;) {
+      if (!(k2>>=1)) {  //look at next bit of y
+        k1--;
+        if (k1<0) {
+          mont_(x,one,n,np);
+          return;
+        }
+        k2=1<<(bpe-1);
+      }    
+      mont_(x,x,n,np);
+
+      if (k2 & y[k1]) //if next bit is a 1
+        mont_(x,s3,n,np);
+    }
+  }
+
+
+  //do x=x*y*Ri mod n for bigInts x,y,n, 
+  //  where Ri = 2**(-kn*bpe) mod n, and kn is the 
+  //  number of elements in the n array, not 
+  //  counting leading zeros.  
+  //x array must have at least as many elemnts as the n array
+  //It's OK if x and y are the same variable.
+  //must have:
+  //  x,y < n
+  //  n is odd
+  //  np = -(n^(-1)) mod radix
+  function mont_(x,y,n,np) {
+    var i,j,c,ui,t,t2,ks;
+    var kn=n.length;
+    var ky=y.length;
+
+    if (sa.length!=kn)
+      sa=new Array(kn);
+      
+    copyInt_(sa,0);
+
+    for (;kn>0 && n[kn-1]==0;kn--); //ignore leading zeros of n
+    for (;ky>0 && y[ky-1]==0;ky--); //ignore leading zeros of y
+    ks=sa.length-1; //sa will never have more than this many nonzero elements.  
+
+    //the following loop consumes 95% of the runtime for randTruePrime_() and powMod_() for large numbers
+    for (i=0; i<kn; i++) {
+      t=sa[0]+x[i]*y[0];
+      ui=((t & mask) * np) & mask;  //the inner "& mask" was needed on Safari (but not MSIE) at one time
+      c=(t+ui*n[0]);
+      c = (c - (c & mask)) / radix;
+      t=x[i];
+      
+      //do sa=(sa+x[i]*y+ui*n)/b   where b=2**bpe.  Loop is unrolled 5-fold for speed
+      j=1;
+      for (;j<ky-4;) {
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<ky;)   {
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<kn-4;) {
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<kn;)   {
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<ks;)   {
+        c+=sa[j];                t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      sa[j-1]=c & mask;
+    }
+
+    if (!greater(n,sa))
+      sub_(sa,n);
+    copy_(x,sa);
+  }
+
+
+  // otr.js additions
+
+
+  // computes num / den mod n
+  function divMod(num, den, n) {
+    return multMod(num, inverseMod(den, n), n)
+  }
+
+  // computes one - two mod n
+  function subMod(one, two, n) {
+    one = mod(one, n)
+    two = mod(two, n)
+    if (greater(two, one)) one = add(one, n)
+    return sub(one, two)
+  }
+
+  // computes 2^m as a bigInt
+  function twoToThe(m) {
+    var b = Math.floor(m / bpe) + 2
+    var t = new Array(b)
+    for (var i = 0; i < b; i++) t[i] = 0
+    t[b - 2] = 1 << (m % bpe)
+    return t
+  }
+
+  // cache these results for faster lookup
+  var _num2bin = (function () {
+    var i = 0, _num2bin= {}
+    for (; i < 0x100; ++i) {
+      _num2bin[i] = String.fromCharCode(i)  // 0 -> "\00"
+    }
+    return _num2bin
+  }())
+
+  // serialize a bigInt to an ascii string
+  // padded up to pad length
+  function bigInt2bits(bi, pad) {
+    pad || (pad = 0)
+    bi = dup(bi)
+    var ba = ''
+    while (!isZero(bi)) {
+      ba = _num2bin[bi[0] & 0xff] + ba
+      rightShift_(bi, 8)
+    }
+    while (ba.length < pad) {
+      ba = '\x00' + ba
+    }
+    return ba
+  }
+
+  // converts a byte array to a bigInt
+  function ba2bigInt(data) {
+    var mpi = str2bigInt('0', 10, data.length)
+    data.forEach(function (d, i) {
+      if (i) leftShift_(mpi, 8)
+      mpi[0] |= d
+    })
+    return mpi
+  }
+
+  // returns a function that returns an array of n bytes
+  var randomBytes = (function () {
+
+    // in node
+    if ( typeof crypto !== 'undefined' &&
+      typeof crypto.randomBytes === 'function' ) {
+      return function (n) {
+        try {
+          var buf = crypto.randomBytes(n)
+        } catch (e) { throw e }
+        return Array.prototype.slice.call(buf, 0)
+      }
+    }
+
+    // in browser
+    else if ( typeof crypto !== 'undefined' &&
+      typeof crypto.getRandomValues === 'function' ) {
+      return function (n) {
+        var buf = new Uint8Array(n)
+        crypto.getRandomValues(buf)
+        return Array.prototype.slice.call(buf, 0)
+      }
+    }
+
+    // err
+    else {
+      throw new Error('Keys should not be generated without CSPRNG.')
+    }
+
+  }())
+
+  // Salsa 20 in webworker needs a 40 byte seed
+  function getSeed() {
+    return randomBytes(40)
+  }
+
+  // returns a single random byte
+  function randomByte() {
+    return randomBytes(1)[0]
+  }
+
+  // returns a k-bit random integer
+  function randomBitInt(k) {
+    if (k > 31) throw new Error("Too many bits.")
+    var i = 0, r = 0
+    var b = Math.floor(k / 8)
+    var mask = (1 << (k % 8)) - 1
+    if (mask) r = randomByte() & mask
+    for (; i < b; i++)
+      r = (256 * r) + randomByte()
+    return r
+  }
+
+  return {
+      str2bigInt    : str2bigInt
+    , bigInt2str    : bigInt2str
+    , int2bigInt    : int2bigInt
+    , multMod       : multMod
+    , powMod        : powMod
+    , inverseMod    : inverseMod
+    , randBigInt    : randBigInt
+    , randBigInt_   : randBigInt_
+    , equals        : equals
+    , equalsInt     : equalsInt
+    , sub           : sub
+    , mod           : mod
+    , modInt        : modInt
+    , mult          : mult
+    , divInt_       : divInt_
+    , rightShift_   : rightShift_
+    , dup           : dup
+    , greater       : greater
+    , add           : add
+    , isZero        : isZero
+    , bitSize       : bitSize
+    , millerRabin   : millerRabin
+    , divide_       : divide_
+    , trim          : trim
+    , primes        : primes
+    , findPrimes    : findPrimes
+    , getSeed       : getSeed
+    , divMod        : divMod
+    , subMod        : subMod
+    , twoToThe      : twoToThe
+    , bigInt2bits   : bigInt2bits
+    , ba2bigInt     : ba2bigInt
+  }
+
+}))
\ No newline at end of file
diff --git a/lib/otr/build/dep/crypto.js b/lib/otr/build/dep/crypto.js
new file mode 100644
index 0000000..d34731b
--- /dev/null
+++ b/lib/otr/build/dep/crypto.js
@@ -0,0 +1,2434 @@
+;(function (root, factory) {
+
+  if (typeof define === "function" && define.amd) {
+    define(factory)
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory()
+  } else {
+    root.CryptoJS = factory()
+  }
+
+}(this, function () {
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * CryptoJS core components.
+ */
+var CryptoJS = CryptoJS || (function (Math, undefined) {
+    /**
+     * CryptoJS namespace.
+     */
+    var C = {};
+
+    /**
+     * Library namespace.
+     */
+    var C_lib = C.lib = {};
+
+    /**
+     * Base object for prototypal inheritance.
+     */
+    var Base = C_lib.Base = (function () {
+        function F() {}
+
+        return {
+            /**
+             * Creates a new object that inherits from this object.
+             *
+             * @param {Object} overrides Properties to copy into the new object.
+             *
+             * @return {Object} The new object.
+             *
+             * @static
+             *
+             * @example
+             *
+             *     var MyType = CryptoJS.lib.Base.extend({
+             *         field: 'value',
+             *
+             *         method: function () {
+             *         }
+             *     });
+             */
+            extend: function (overrides) {
+                // Spawn
+                F.prototype = this;
+                var subtype = new F();
+
+                // Augment
+                if (overrides) {
+                    subtype.mixIn(overrides);
+                }
+
+                // Create default initializer
+                if (!subtype.hasOwnProperty('init')) {
+                    subtype.init = function () {
+                        subtype.$super.init.apply(this, arguments);
+                    };
+                }
+
+                // Initializer's prototype is the subtype object
+                subtype.init.prototype = subtype;
+
+                // Reference supertype
+                subtype.$super = this;
+
+                return subtype;
+            },
+
+            /**
+             * Extends this object and runs the init method.
+             * Arguments to create() will be passed to init().
+             *
+             * @return {Object} The new object.
+             *
+             * @static
+             *
+             * @example
+             *
+             *     var instance = MyType.create();
+             */
+            create: function () {
+                var instance = this.extend();
+                instance.init.apply(instance, arguments);
+
+                return instance;
+            },
+
+            /**
+             * Initializes a newly created object.
+             * Override this method to add some logic when your objects are created.
+             *
+             * @example
+             *
+             *     var MyType = CryptoJS.lib.Base.extend({
+             *         init: function () {
+             *             // ...
+             *         }
+             *     });
+             */
+            init: function () {
+            },
+
+            /**
+             * Copies properties into this object.
+             *
+             * @param {Object} properties The properties to mix in.
+             *
+             * @example
+             *
+             *     MyType.mixIn({
+             *         field: 'value'
+             *     });
+             */
+            mixIn: function (properties) {
+                for (var propertyName in properties) {
+                    if (properties.hasOwnProperty(propertyName)) {
+                        this[propertyName] = properties[propertyName];
+                    }
+                }
+
+                // IE won't copy toString using the loop above
+                if (properties.hasOwnProperty('toString')) {
+                    this.toString = properties.toString;
+                }
+            },
+
+            /**
+             * Creates a copy of this object.
+             *
+             * @return {Object} The clone.
+             *
+             * @example
+             *
+             *     var clone = instance.clone();
+             */
+            clone: function () {
+                return this.init.prototype.extend(this);
+            }
+        };
+    }());
+
+    /**
+     * An array of 32-bit words.
+     *
+     * @property {Array} words The array of 32-bit words.
+     * @property {number} sigBytes The number of significant bytes in this word array.
+     */
+    var WordArray = C_lib.WordArray = Base.extend({
+        /**
+         * Initializes a newly created word array.
+         *
+         * @param {Array} words (Optional) An array of 32-bit words.
+         * @param {number} sigBytes (Optional) The number of significant bytes in the words.
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.lib.WordArray.create();
+         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);
+         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);
+         */
+        init: function (words, sigBytes) {
+            words = this.words = words || [];
+
+            if (sigBytes != undefined) {
+                this.sigBytes = sigBytes;
+            } else {
+                this.sigBytes = words.length * 4;
+            }
+        },
+
+        /**
+         * Converts this word array to a string.
+         *
+         * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex
+         *
+         * @return {string} The stringified word array.
+         *
+         * @example
+         *
+         *     var string = wordArray + '';
+         *     var string = wordArray.toString();
+         *     var string = wordArray.toString(CryptoJS.enc.Utf8);
+         */
+        toString: function (encoder) {
+            return (encoder || Hex).stringify(this);
+        },
+
+        /**
+         * Concatenates a word array to this word array.
+         *
+         * @param {WordArray} wordArray The word array to append.
+         *
+         * @return {WordArray} This word array.
+         *
+         * @example
+         *
+         *     wordArray1.concat(wordArray2);
+         */
+        concat: function (wordArray) {
+            // Shortcuts
+            var thisWords = this.words;
+            var thatWords = wordArray.words;
+            var thisSigBytes = this.sigBytes;
+            var thatSigBytes = wordArray.sigBytes;
+
+            // Clamp excess bits
+            this.clamp();
+
+            // Concat
+            if (thisSigBytes % 4) {
+                // Copy one byte at a time
+                for (var i = 0; i < thatSigBytes; i++) {
+                    var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                    thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);
+                }
+            } else if (thatWords.length > 0xffff) {
+                // Copy one word at a time
+                for (var i = 0; i < thatSigBytes; i += 4) {
+                    thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];
+                }
+            } else {
+                // Copy all words at once
+                thisWords.push.apply(thisWords, thatWords);
+            }
+            this.sigBytes += thatSigBytes;
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Removes insignificant bits.
+         *
+         * @example
+         *
+         *     wordArray.clamp();
+         */
+        clamp: function () {
+            // Shortcuts
+            var words = this.words;
+            var sigBytes = this.sigBytes;
+
+            // Clamp
+            words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);
+            words.length = Math.ceil(sigBytes / 4);
+        },
+
+        /**
+         * Creates a copy of this word array.
+         *
+         * @return {WordArray} The clone.
+         *
+         * @example
+         *
+         *     var clone = wordArray.clone();
+         */
+        clone: function () {
+            var clone = Base.clone.call(this);
+            clone.words = this.words.slice(0);
+
+            return clone;
+        },
+
+        /**
+         * Creates a word array filled with random bytes.
+         *
+         * @param {number} nBytes The number of random bytes to generate.
+         *
+         * @return {WordArray} The random word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.lib.WordArray.random(16);
+         */
+        random: function (nBytes) {
+            var words = [];
+            for (var i = 0; i < nBytes; i += 4) {
+                words.push((Math.random() * 0x100000000) | 0);
+            }
+
+            return new WordArray.init(words, nBytes);
+        }
+    });
+
+    /**
+     * Encoder namespace.
+     */
+    var C_enc = C.enc = {};
+
+    /**
+     * Hex encoding strategy.
+     */
+    var Hex = C_enc.Hex = {
+        /**
+         * Converts a word array to a hex string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The hex string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var hexString = CryptoJS.enc.Hex.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+
+            // Convert
+            var hexChars = [];
+            for (var i = 0; i < sigBytes; i++) {
+                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                hexChars.push((bite >>> 4).toString(16));
+                hexChars.push((bite & 0x0f).toString(16));
+            }
+
+            return hexChars.join('');
+        },
+
+        /**
+         * Converts a hex string to a word array.
+         *
+         * @param {string} hexStr The hex string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Hex.parse(hexString);
+         */
+        parse: function (hexStr) {
+            // Shortcut
+            var hexStrLength = hexStr.length;
+
+            // Convert
+            var words = [];
+            for (var i = 0; i < hexStrLength; i += 2) {
+                words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);
+            }
+
+            return new WordArray.init(words, hexStrLength / 2);
+        }
+    };
+
+    /**
+     * Latin1 encoding strategy.
+     */
+    var Latin1 = C_enc.Latin1 = {
+        /**
+         * Converts a word array to a Latin1 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The Latin1 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+
+            // Convert
+            var latin1Chars = [];
+            for (var i = 0; i < sigBytes; i++) {
+                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                latin1Chars.push(String.fromCharCode(bite));
+            }
+
+            return latin1Chars.join('');
+        },
+
+        /**
+         * Converts a Latin1 string to a word array.
+         *
+         * @param {string} latin1Str The Latin1 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Latin1.parse(latin1String);
+         */
+        parse: function (latin1Str) {
+            // Shortcut
+            var latin1StrLength = latin1Str.length;
+
+            // Convert
+            var words = [];
+            for (var i = 0; i < latin1StrLength; i++) {
+                words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);
+            }
+
+            return new WordArray.init(words, latin1StrLength);
+        }
+    };
+
+    /**
+     * UTF-8 encoding strategy.
+     */
+    var Utf8 = C_enc.Utf8 = {
+        /**
+         * Converts a word array to a UTF-8 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The UTF-8 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            try {
+                return decodeURIComponent(escape(Latin1.stringify(wordArray)));
+            } catch (e) {
+                throw new Error('Malformed UTF-8 data');
+            }
+        },
+
+        /**
+         * Converts a UTF-8 string to a word array.
+         *
+         * @param {string} utf8Str The UTF-8 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Utf8.parse(utf8String);
+         */
+        parse: function (utf8Str) {
+            return Latin1.parse(unescape(encodeURIComponent(utf8Str)));
+        }
+    };
+
+    /**
+     * Abstract buffered block algorithm template.
+     *
+     * The property blockSize must be implemented in a concrete subtype.
+     *
+     * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0
+     */
+    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({
+        /**
+         * Resets this block algorithm's data buffer to its initial state.
+         *
+         * @example
+         *
+         *     bufferedBlockAlgorithm.reset();
+         */
+        reset: function () {
+            // Initial values
+            this._data = new WordArray.init();
+            this._nDataBytes = 0;
+        },
+
+        /**
+         * Adds new data to this block algorithm's buffer.
+         *
+         * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.
+         *
+         * @example
+         *
+         *     bufferedBlockAlgorithm._append('data');
+         *     bufferedBlockAlgorithm._append(wordArray);
+         */
+        _append: function (data) {
+            // Convert string to WordArray, else assume WordArray already
+            if (typeof data == 'string') {
+                data = Utf8.parse(data);
+            }
+
+            // Append
+            this._data.concat(data);
+            this._nDataBytes += data.sigBytes;
+        },
+
+        /**
+         * Processes available data blocks.
+         *
+         * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.
+         *
+         * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.
+         *
+         * @return {WordArray} The processed data.
+         *
+         * @example
+         *
+         *     var processedData = bufferedBlockAlgorithm._process();
+         *     var processedData = bufferedBlockAlgorithm._process(!!'flush');
+         */
+        _process: function (doFlush) {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+            var dataSigBytes = data.sigBytes;
+            var blockSize = this.blockSize;
+            var blockSizeBytes = blockSize * 4;
+
+            // Count blocks ready
+            var nBlocksReady = dataSigBytes / blockSizeBytes;
+            if (doFlush) {
+                // Round up to include partial blocks
+                nBlocksReady = Math.ceil(nBlocksReady);
+            } else {
+                // Round down to include only full blocks,
+                // less the number of blocks that must remain in the buffer
+                nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);
+            }
+
+            // Count words ready
+            var nWordsReady = nBlocksReady * blockSize;
+
+            // Count bytes ready
+            var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);
+
+            // Process blocks
+            if (nWordsReady) {
+                for (var offset = 0; offset < nWordsReady; offset += blockSize) {
+                    // Perform concrete-algorithm logic
+                    this._doProcessBlock(dataWords, offset);
+                }
+
+                // Remove processed words
+                var processedWords = dataWords.splice(0, nWordsReady);
+                data.sigBytes -= nBytesReady;
+            }
+
+            // Return processed words
+            return new WordArray.init(processedWords, nBytesReady);
+        },
+
+        /**
+         * Creates a copy of this object.
+         *
+         * @return {Object} The clone.
+         *
+         * @example
+         *
+         *     var clone = bufferedBlockAlgorithm.clone();
+         */
+        clone: function () {
+            var clone = Base.clone.call(this);
+            clone._data = this._data.clone();
+
+            return clone;
+        },
+
+        _minBufferSize: 0
+    });
+
+    /**
+     * Abstract hasher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)
+     */
+    var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({
+        /**
+         * Configuration options.
+         */
+        cfg: Base.extend(),
+
+        /**
+         * Initializes a newly created hasher.
+         *
+         * @param {Object} cfg (Optional) The configuration options to use for this hash computation.
+         *
+         * @example
+         *
+         *     var hasher = CryptoJS.algo.SHA256.create();
+         */
+        init: function (cfg) {
+            // Apply config defaults
+            this.cfg = this.cfg.extend(cfg);
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this hasher to its initial state.
+         *
+         * @example
+         *
+         *     hasher.reset();
+         */
+        reset: function () {
+            // Reset data buffer
+            BufferedBlockAlgorithm.reset.call(this);
+
+            // Perform concrete-hasher logic
+            this._doReset();
+        },
+
+        /**
+         * Updates this hasher with a message.
+         *
+         * @param {WordArray|string} messageUpdate The message to append.
+         *
+         * @return {Hasher} This hasher.
+         *
+         * @example
+         *
+         *     hasher.update('message');
+         *     hasher.update(wordArray);
+         */
+        update: function (messageUpdate) {
+            // Append
+            this._append(messageUpdate);
+
+            // Update the hash
+            this._process();
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Finalizes the hash computation.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} messageUpdate (Optional) A final message update.
+         *
+         * @return {WordArray} The hash.
+         *
+         * @example
+         *
+         *     var hash = hasher.finalize();
+         *     var hash = hasher.finalize('message');
+         *     var hash = hasher.finalize(wordArray);
+         */
+        finalize: function (messageUpdate) {
+            // Final message update
+            if (messageUpdate) {
+                this._append(messageUpdate);
+            }
+
+            // Perform concrete-hasher logic
+            var hash = this._doFinalize();
+
+            return hash;
+        },
+
+        blockSize: 512/32,
+
+        /**
+         * Creates a shortcut function to a hasher's object interface.
+         *
+         * @param {Hasher} hasher The hasher to create a helper for.
+         *
+         * @return {Function} The shortcut function.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);
+         */
+        _createHelper: function (hasher) {
+            return function (message, cfg) {
+                return new hasher.init(cfg).finalize(message);
+            };
+        },
+
+        /**
+         * Creates a shortcut function to the HMAC's object interface.
+         *
+         * @param {Hasher} hasher The hasher to use in this HMAC helper.
+         *
+         * @return {Function} The shortcut function.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);
+         */
+        _createHmacHelper: function (hasher) {
+            return function (message, key) {
+                return new C_algo.HMAC.init(hasher, key).finalize(message);
+            };
+        }
+    });
+
+    /**
+     * Algorithm namespace.
+     */
+    var C_algo = C.algo = {};
+
+    return C;
+}(Math));
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var C_enc = C.enc;
+
+    /**
+     * Base64 encoding strategy.
+     */
+    var Base64 = C_enc.Base64 = {
+        /**
+         * Converts a word array to a Base64 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The Base64 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var base64String = CryptoJS.enc.Base64.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+            var map = this._map;
+
+            // Clamp excess bits
+            wordArray.clamp();
+
+            // Convert
+            var base64Chars = [];
+            for (var i = 0; i < sigBytes; i += 3) {
+                var byte1 = (words[i >>> 2]       >>> (24 - (i % 4) * 8))       & 0xff;
+                var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;
+                var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;
+
+                var triplet = (byte1 << 16) | (byte2 << 8) | byte3;
+
+                for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {
+                    base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));
+                }
+            }
+
+            // Add padding
+            var paddingChar = map.charAt(64);
+            if (paddingChar) {
+                while (base64Chars.length % 4) {
+                    base64Chars.push(paddingChar);
+                }
+            }
+
+            return base64Chars.join('');
+        },
+
+        /**
+         * Converts a Base64 string to a word array.
+         *
+         * @param {string} base64Str The Base64 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Base64.parse(base64String);
+         */
+        parse: function (base64Str) {
+            // Shortcuts
+            var base64StrLength = base64Str.length;
+            var map = this._map;
+
+            // Ignore padding
+            var paddingChar = map.charAt(64);
+            if (paddingChar) {
+                var paddingIndex = base64Str.indexOf(paddingChar);
+                if (paddingIndex != -1) {
+                    base64StrLength = paddingIndex;
+                }
+            }
+
+            // Convert
+            var words = [];
+            var nBytes = 0;
+            for (var i = 0; i < base64StrLength; i++) {
+                if (i % 4) {
+                    var bits1 = map.indexOf(base64Str.charAt(i - 1)) << ((i % 4) * 2);
+                    var bits2 = map.indexOf(base64Str.charAt(i)) >>> (6 - (i % 4) * 2);
+                    words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);
+                    nBytes++;
+                }
+            }
+
+            return WordArray.create(words, nBytes);
+        },
+
+        _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='
+    };
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * Cipher core components.
+ */
+CryptoJS.lib.Cipher || (function (undefined) {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var Base = C_lib.Base;
+    var WordArray = C_lib.WordArray;
+    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm;
+    var C_enc = C.enc;
+    var Utf8 = C_enc.Utf8;
+    var Base64 = C_enc.Base64;
+    var C_algo = C.algo;
+    var EvpKDF = C_algo.EvpKDF;
+
+    /**
+     * Abstract base cipher template.
+     *
+     * @property {number} keySize This cipher's key size. Default: 4 (128 bits)
+     * @property {number} ivSize This cipher's IV size. Default: 4 (128 bits)
+     * @property {number} _ENC_XFORM_MODE A constant representing encryption mode.
+     * @property {number} _DEC_XFORM_MODE A constant representing decryption mode.
+     */
+    var Cipher = C_lib.Cipher = BufferedBlockAlgorithm.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {WordArray} iv The IV to use for this operation.
+         */
+        cfg: Base.extend(),
+
+        /**
+         * Creates this cipher in encryption mode.
+         *
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {Cipher} A cipher instance.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.createEncryptor(keyWordArray, { iv: ivWordArray });
+         */
+        createEncryptor: function (key, cfg) {
+            return this.create(this._ENC_XFORM_MODE, key, cfg);
+        },
+
+        /**
+         * Creates this cipher in decryption mode.
+         *
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {Cipher} A cipher instance.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.createDecryptor(keyWordArray, { iv: ivWordArray });
+         */
+        createDecryptor: function (key, cfg) {
+            return this.create(this._DEC_XFORM_MODE, key, cfg);
+        },
+
+        /**
+         * Initializes a newly created cipher.
+         *
+         * @param {number} xformMode Either the encryption or decryption transormation mode constant.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.create(CryptoJS.algo.AES._ENC_XFORM_MODE, keyWordArray, { iv: ivWordArray });
+         */
+        init: function (xformMode, key, cfg) {
+            // Apply config defaults
+            this.cfg = this.cfg.extend(cfg);
+
+            // Store transform mode and key
+            this._xformMode = xformMode;
+            this._key = key;
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this cipher to its initial state.
+         *
+         * @example
+         *
+         *     cipher.reset();
+         */
+        reset: function () {
+            // Reset data buffer
+            BufferedBlockAlgorithm.reset.call(this);
+
+            // Perform concrete-cipher logic
+            this._doReset();
+        },
+
+        /**
+         * Adds data to be encrypted or decrypted.
+         *
+         * @param {WordArray|string} dataUpdate The data to encrypt or decrypt.
+         *
+         * @return {WordArray} The data after processing.
+         *
+         * @example
+         *
+         *     var encrypted = cipher.process('data');
+         *     var encrypted = cipher.process(wordArray);
+         */
+        process: function (dataUpdate) {
+            // Append
+            this._append(dataUpdate);
+
+            // Process available blocks
+            return this._process();
+        },
+
+        /**
+         * Finalizes the encryption or decryption process.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} dataUpdate The final data to encrypt or decrypt.
+         *
+         * @return {WordArray} The data after final processing.
+         *
+         * @example
+         *
+         *     var encrypted = cipher.finalize();
+         *     var encrypted = cipher.finalize('data');
+         *     var encrypted = cipher.finalize(wordArray);
+         */
+        finalize: function (dataUpdate) {
+            // Final data update
+            if (dataUpdate) {
+                this._append(dataUpdate);
+            }
+
+            // Perform concrete-cipher logic
+            var finalProcessedData = this._doFinalize();
+
+            return finalProcessedData;
+        },
+
+        keySize: 128/32,
+
+        ivSize: 128/32,
+
+        _ENC_XFORM_MODE: 1,
+
+        _DEC_XFORM_MODE: 2,
+
+        /**
+         * Creates shortcut functions to a cipher's object interface.
+         *
+         * @param {Cipher} cipher The cipher to create a helper for.
+         *
+         * @return {Object} An object with encrypt and decrypt shortcut functions.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var AES = CryptoJS.lib.Cipher._createHelper(CryptoJS.algo.AES);
+         */
+        _createHelper: (function () {
+            function selectCipherStrategy(key) {
+                if (typeof key == 'string') {
+                    return PasswordBasedCipher;
+                } else {
+                    return SerializableCipher;
+                }
+            }
+
+            return function (cipher) {
+                return {
+                    encrypt: function (message, key, cfg) {
+                        return selectCipherStrategy(key).encrypt(cipher, message, key, cfg);
+                    },
+
+                    decrypt: function (ciphertext, key, cfg) {
+                        return selectCipherStrategy(key).decrypt(cipher, ciphertext, key, cfg);
+                    }
+                };
+            };
+        }())
+    });
+
+    /**
+     * Abstract base stream cipher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 1 (32 bits)
+     */
+    var StreamCipher = C_lib.StreamCipher = Cipher.extend({
+        _doFinalize: function () {
+            // Process partial blocks
+            var finalProcessedBlocks = this._process(!!'flush');
+
+            return finalProcessedBlocks;
+        },
+
+        blockSize: 1
+    });
+
+    /**
+     * Mode namespace.
+     */
+    var C_mode = C.mode = {};
+
+    /**
+     * Abstract base block cipher mode template.
+     */
+    var BlockCipherMode = C_lib.BlockCipherMode = Base.extend({
+        /**
+         * Creates this mode for encryption.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.createEncryptor(cipher, iv.words);
+         */
+        createEncryptor: function (cipher, iv) {
+            return this.Encryptor.create(cipher, iv);
+        },
+
+        /**
+         * Creates this mode for decryption.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.createDecryptor(cipher, iv.words);
+         */
+        createDecryptor: function (cipher, iv) {
+            return this.Decryptor.create(cipher, iv);
+        },
+
+        /**
+         * Initializes a newly created mode.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.Encryptor.create(cipher, iv.words);
+         */
+        init: function (cipher, iv) {
+            this._cipher = cipher;
+            this._iv = iv;
+        }
+    });
+
+    /**
+     * Cipher Block Chaining mode.
+     */
+    var CBC = C_mode.CBC = (function () {
+        /**
+         * Abstract base CBC mode.
+         */
+        var CBC = BlockCipherMode.extend();
+
+        /**
+         * CBC encryptor.
+         */
+        CBC.Encryptor = CBC.extend({
+            /**
+             * Processes the data block at offset.
+             *
+             * @param {Array} words The data words to operate on.
+             * @param {number} offset The offset where the block starts.
+             *
+             * @example
+             *
+             *     mode.processBlock(data.words, offset);
+             */
+            processBlock: function (words, offset) {
+                // Shortcuts
+                var cipher = this._cipher;
+                var blockSize = cipher.blockSize;
+
+                // XOR and encrypt
+                xorBlock.call(this, words, offset, blockSize);
+                cipher.encryptBlock(words, offset);
+
+                // Remember this block to use with next block
+                this._prevBlock = words.slice(offset, offset + blockSize);
+            }
+        });
+
+        /**
+         * CBC decryptor.
+         */
+        CBC.Decryptor = CBC.extend({
+            /**
+             * Processes the data block at offset.
+             *
+             * @param {Array} words The data words to operate on.
+             * @param {number} offset The offset where the block starts.
+             *
+             * @example
+             *
+             *     mode.processBlock(data.words, offset);
+             */
+            processBlock: function (words, offset) {
+                // Shortcuts
+                var cipher = this._cipher;
+                var blockSize = cipher.blockSize;
+
+                // Remember this block to use with next block
+                var thisBlock = words.slice(offset, offset + blockSize);
+
+                // Decrypt and XOR
+                cipher.decryptBlock(words, offset);
+                xorBlock.call(this, words, offset, blockSize);
+
+                // This block becomes the previous block
+                this._prevBlock = thisBlock;
+            }
+        });
+
+        function xorBlock(words, offset, blockSize) {
+            // Shortcut
+            var iv = this._iv;
+
+            // Choose mixing block
+            if (iv) {
+                var block = iv;
+
+                // Remove IV for subsequent blocks
+                this._iv = undefined;
+            } else {
+                var block = this._prevBlock;
+            }
+
+            // XOR blocks
+            for (var i = 0; i < blockSize; i++) {
+                words[offset + i] ^= block[i];
+            }
+        }
+
+        return CBC;
+    }());
+
+    /**
+     * Padding namespace.
+     */
+    var C_pad = C.pad = {};
+
+    /**
+     * PKCS #5/7 padding strategy.
+     */
+    var Pkcs7 = C_pad.Pkcs7 = {
+        /**
+         * Pads data using the algorithm defined in PKCS #5/7.
+         *
+         * @param {WordArray} data The data to pad.
+         * @param {number} blockSize The multiple that the data should be padded to.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     CryptoJS.pad.Pkcs7.pad(wordArray, 4);
+         */
+        pad: function (data, blockSize) {
+            // Shortcut
+            var blockSizeBytes = blockSize * 4;
+
+            // Count padding bytes
+            var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;
+
+            // Create padding word
+            var paddingWord = (nPaddingBytes << 24) | (nPaddingBytes << 16) | (nPaddingBytes << 8) | nPaddingBytes;
+
+            // Create padding
+            var paddingWords = [];
+            for (var i = 0; i < nPaddingBytes; i += 4) {
+                paddingWords.push(paddingWord);
+            }
+            var padding = WordArray.create(paddingWords, nPaddingBytes);
+
+            // Add padding
+            data.concat(padding);
+        },
+
+        /**
+         * Unpads data that had been padded using the algorithm defined in PKCS #5/7.
+         *
+         * @param {WordArray} data The data to unpad.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     CryptoJS.pad.Pkcs7.unpad(wordArray);
+         */
+        unpad: function (data) {
+            // Get number of padding bytes from last byte
+            var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;
+
+            // Remove padding
+            data.sigBytes -= nPaddingBytes;
+        }
+    };
+
+    /**
+     * Abstract base block cipher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 4 (128 bits)
+     */
+    var BlockCipher = C_lib.BlockCipher = Cipher.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {Mode} mode The block mode to use. Default: CBC
+         * @property {Padding} padding The padding strategy to use. Default: Pkcs7
+         */
+        cfg: Cipher.cfg.extend({
+            mode: CBC,
+            padding: Pkcs7
+        }),
+
+        reset: function () {
+            // Reset cipher
+            Cipher.reset.call(this);
+
+            // Shortcuts
+            var cfg = this.cfg;
+            var iv = cfg.iv;
+            var mode = cfg.mode;
+
+            // Reset block mode
+            if (this._xformMode == this._ENC_XFORM_MODE) {
+                var modeCreator = mode.createEncryptor;
+            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
+                var modeCreator = mode.createDecryptor;
+
+                // Keep at least one block in the buffer for unpadding
+                this._minBufferSize = 1;
+            }
+            this._mode = modeCreator.call(mode, this, iv && iv.words);
+        },
+
+        _doProcessBlock: function (words, offset) {
+            this._mode.processBlock(words, offset);
+        },
+
+        _doFinalize: function () {
+            // Shortcut
+            var padding = this.cfg.padding;
+
+            // Finalize
+            if (this._xformMode == this._ENC_XFORM_MODE) {
+                // Pad data
+                padding.pad(this._data, this.blockSize);
+
+                // Process final blocks
+                var finalProcessedBlocks = this._process(!!'flush');
+            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
+                // Process final blocks
+                var finalProcessedBlocks = this._process(!!'flush');
+
+                // Unpad data
+                padding.unpad(finalProcessedBlocks);
+            }
+
+            return finalProcessedBlocks;
+        },
+
+        blockSize: 128/32
+    });
+
+    /**
+     * A collection of cipher parameters.
+     *
+     * @property {WordArray} ciphertext The raw ciphertext.
+     * @property {WordArray} key The key to this ciphertext.
+     * @property {WordArray} iv The IV used in the ciphering operation.
+     * @property {WordArray} salt The salt used with a key derivation function.
+     * @property {Cipher} algorithm The cipher algorithm.
+     * @property {Mode} mode The block mode used in the ciphering operation.
+     * @property {Padding} padding The padding scheme used in the ciphering operation.
+     * @property {number} blockSize The block size of the cipher.
+     * @property {Format} formatter The default formatting strategy to convert this cipher params object to a string.
+     */
+    var CipherParams = C_lib.CipherParams = Base.extend({
+        /**
+         * Initializes a newly created cipher params object.
+         *
+         * @param {Object} cipherParams An object with any of the possible cipher parameters.
+         *
+         * @example
+         *
+         *     var cipherParams = CryptoJS.lib.CipherParams.create({
+         *         ciphertext: ciphertextWordArray,
+         *         key: keyWordArray,
+         *         iv: ivWordArray,
+         *         salt: saltWordArray,
+         *         algorithm: CryptoJS.algo.AES,
+         *         mode: CryptoJS.mode.CBC,
+         *         padding: CryptoJS.pad.PKCS7,
+         *         blockSize: 4,
+         *         formatter: CryptoJS.format.OpenSSL
+         *     });
+         */
+        init: function (cipherParams) {
+            this.mixIn(cipherParams);
+        },
+
+        /**
+         * Converts this cipher params object to a string.
+         *
+         * @param {Format} formatter (Optional) The formatting strategy to use.
+         *
+         * @return {string} The stringified cipher params.
+         *
+         * @throws Error If neither the formatter nor the default formatter is set.
+         *
+         * @example
+         *
+         *     var string = cipherParams + '';
+         *     var string = cipherParams.toString();
+         *     var string = cipherParams.toString(CryptoJS.format.OpenSSL);
+         */
+        toString: function (formatter) {
+            return (formatter || this.formatter).stringify(this);
+        }
+    });
+
+    /**
+     * Format namespace.
+     */
+    var C_format = C.format = {};
+
+    /**
+     * OpenSSL formatting strategy.
+     */
+    var OpenSSLFormatter = C_format.OpenSSL = {
+        /**
+         * Converts a cipher params object to an OpenSSL-compatible string.
+         *
+         * @param {CipherParams} cipherParams The cipher params object.
+         *
+         * @return {string} The OpenSSL-compatible string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var openSSLString = CryptoJS.format.OpenSSL.stringify(cipherParams);
+         */
+        stringify: function (cipherParams) {
+            // Shortcuts
+            var ciphertext = cipherParams.ciphertext;
+            var salt = cipherParams.salt;
+
+            // Format
+            if (salt) {
+                var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);
+            } else {
+                var wordArray = ciphertext;
+            }
+
+            return wordArray.toString(Base64);
+        },
+
+        /**
+         * Converts an OpenSSL-compatible string to a cipher params object.
+         *
+         * @param {string} openSSLStr The OpenSSL-compatible string.
+         *
+         * @return {CipherParams} The cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipherParams = CryptoJS.format.OpenSSL.parse(openSSLString);
+         */
+        parse: function (openSSLStr) {
+            // Parse base64
+            var ciphertext = Base64.parse(openSSLStr);
+
+            // Shortcut
+            var ciphertextWords = ciphertext.words;
+
+            // Test for salt
+            if (ciphertextWords[0] == 0x53616c74 && ciphertextWords[1] == 0x65645f5f) {
+                // Extract salt
+                var salt = WordArray.create(ciphertextWords.slice(2, 4));
+
+                // Remove salt from ciphertext
+                ciphertextWords.splice(0, 4);
+                ciphertext.sigBytes -= 16;
+            }
+
+            return CipherParams.create({ ciphertext: ciphertext, salt: salt });
+        }
+    };
+
+    /**
+     * A cipher wrapper that returns ciphertext as a serializable cipher params object.
+     */
+    var SerializableCipher = C_lib.SerializableCipher = Base.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {Formatter} format The formatting strategy to convert cipher param objects to and from a string. Default: OpenSSL
+         */
+        cfg: Base.extend({
+            format: OpenSSLFormatter
+        }),
+
+        /**
+         * Encrypts a message.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {WordArray|string} message The message to encrypt.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {CipherParams} A cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key);
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv });
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         */
+        encrypt: function (cipher, message, key, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Encrypt
+            var encryptor = cipher.createEncryptor(key, cfg);
+            var ciphertext = encryptor.finalize(message);
+
+            // Shortcut
+            var cipherCfg = encryptor.cfg;
+
+            // Create and return serializable cipher params
+            return CipherParams.create({
+                ciphertext: ciphertext,
+                key: key,
+                iv: cipherCfg.iv,
+                algorithm: cipher,
+                mode: cipherCfg.mode,
+                padding: cipherCfg.padding,
+                blockSize: cipher.blockSize,
+                formatter: cfg.format
+            });
+        },
+
+        /**
+         * Decrypts serialized ciphertext.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {WordArray} The plaintext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         */
+        decrypt: function (cipher, ciphertext, key, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Convert string to CipherParams
+            ciphertext = this._parse(ciphertext, cfg.format);
+
+            // Decrypt
+            var plaintext = cipher.createDecryptor(key, cfg).finalize(ciphertext.ciphertext);
+
+            return plaintext;
+        },
+
+        /**
+         * Converts serialized ciphertext to CipherParams,
+         * else assumed CipherParams already and returns ciphertext unchanged.
+         *
+         * @param {CipherParams|string} ciphertext The ciphertext.
+         * @param {Formatter} format The formatting strategy to use to parse serialized ciphertext.
+         *
+         * @return {CipherParams} The unserialized ciphertext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher._parse(ciphertextStringOrParams, format);
+         */
+        _parse: function (ciphertext, format) {
+            if (typeof ciphertext == 'string') {
+                return format.parse(ciphertext, this);
+            } else {
+                return ciphertext;
+            }
+        }
+    });
+
+    /**
+     * Key derivation function namespace.
+     */
+    var C_kdf = C.kdf = {};
+
+    /**
+     * OpenSSL key derivation function.
+     */
+    var OpenSSLKdf = C_kdf.OpenSSL = {
+        /**
+         * Derives a key and IV from a password.
+         *
+         * @param {string} password The password to derive from.
+         * @param {number} keySize The size in words of the key to generate.
+         * @param {number} ivSize The size in words of the IV to generate.
+         * @param {WordArray|string} salt (Optional) A 64-bit salt to use. If omitted, a salt will be generated randomly.
+         *
+         * @return {CipherParams} A cipher params object with the key, IV, and salt.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32);
+         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32, 'saltsalt');
+         */
+        execute: function (password, keySize, ivSize, salt) {
+            // Generate random salt
+            if (!salt) {
+                salt = WordArray.random(64/8);
+            }
+
+            // Derive key and IV
+            var key = EvpKDF.create({ keySize: keySize + ivSize }).compute(password, salt);
+
+            // Separate key and IV
+            var iv = WordArray.create(key.words.slice(keySize), ivSize * 4);
+            key.sigBytes = keySize * 4;
+
+            // Return params
+            return CipherParams.create({ key: key, iv: iv, salt: salt });
+        }
+    };
+
+    /**
+     * A serializable cipher wrapper that derives the key from a password,
+     * and returns ciphertext as a serializable cipher params object.
+     */
+    var PasswordBasedCipher = C_lib.PasswordBasedCipher = SerializableCipher.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {KDF} kdf The key derivation function to use to generate a key and IV from a password. Default: OpenSSL
+         */
+        cfg: SerializableCipher.cfg.extend({
+            kdf: OpenSSLKdf
+        }),
+
+        /**
+         * Encrypts a message using a password.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {WordArray|string} message The message to encrypt.
+         * @param {string} password The password.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {CipherParams} A cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password');
+         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password', { format: CryptoJS.format.OpenSSL });
+         */
+        encrypt: function (cipher, message, password, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Derive key and other params
+            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize);
+
+            // Add IV to config
+            cfg.iv = derivedParams.iv;
+
+            // Encrypt
+            var ciphertext = SerializableCipher.encrypt.call(this, cipher, message, derivedParams.key, cfg);
+
+            // Mix in derived params
+            ciphertext.mixIn(derivedParams);
+
+            return ciphertext;
+        },
+
+        /**
+         * Decrypts serialized ciphertext using a password.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
+         * @param {string} password The password.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {WordArray} The plaintext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, 'password', { format: CryptoJS.format.OpenSSL });
+         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, 'password', { format: CryptoJS.format.OpenSSL });
+         */
+        decrypt: function (cipher, ciphertext, password, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Convert string to CipherParams
+            ciphertext = this._parse(ciphertext, cfg.format);
+
+            // Derive key and other params
+            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize, ciphertext.salt);
+
+            // Add IV to config
+            cfg.iv = derivedParams.iv;
+
+            // Decrypt
+            var plaintext = SerializableCipher.decrypt.call(this, cipher, ciphertext, derivedParams.key, cfg);
+
+            return plaintext;
+        }
+    });
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var BlockCipher = C_lib.BlockCipher;
+    var C_algo = C.algo;
+
+    // Lookup tables
+    var SBOX = [];
+    var INV_SBOX = [];
+    var SUB_MIX_0 = [];
+    var SUB_MIX_1 = [];
+    var SUB_MIX_2 = [];
+    var SUB_MIX_3 = [];
+    var INV_SUB_MIX_0 = [];
+    var INV_SUB_MIX_1 = [];
+    var INV_SUB_MIX_2 = [];
+    var INV_SUB_MIX_3 = [];
+
+    // Compute lookup tables
+    (function () {
+        // Compute double table
+        var d = [];
+        for (var i = 0; i < 256; i++) {
+            if (i < 128) {
+                d[i] = i << 1;
+            } else {
+                d[i] = (i << 1) ^ 0x11b;
+            }
+        }
+
+        // Walk GF(2^8)
+        var x = 0;
+        var xi = 0;
+        for (var i = 0; i < 256; i++) {
+            // Compute sbox
+            var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4);
+            sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63;
+            SBOX[x] = sx;
+            INV_SBOX[sx] = x;
+
+            // Compute multiplication
+            var x2 = d[x];
+            var x4 = d[x2];
+            var x8 = d[x4];
+
+            // Compute sub bytes, mix columns tables
+            var t = (d[sx] * 0x101) ^ (sx * 0x1010100);
+            SUB_MIX_0[x] = (t << 24) | (t >>> 8);
+            SUB_MIX_1[x] = (t << 16) | (t >>> 16);
+            SUB_MIX_2[x] = (t << 8)  | (t >>> 24);
+            SUB_MIX_3[x] = t;
+
+            // Compute inv sub bytes, inv mix columns tables
+            var t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100);
+            INV_SUB_MIX_0[sx] = (t << 24) | (t >>> 8);
+            INV_SUB_MIX_1[sx] = (t << 16) | (t >>> 16);
+            INV_SUB_MIX_2[sx] = (t << 8)  | (t >>> 24);
+            INV_SUB_MIX_3[sx] = t;
+
+            // Compute next counter
+            if (!x) {
+                x = xi = 1;
+            } else {
+                x = x2 ^ d[d[d[x8 ^ x2]]];
+                xi ^= d[d[xi]];
+            }
+        }
+    }());
+
+    // Precomputed Rcon lookup
+    var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36];
+
+    /**
+     * AES block cipher algorithm.
+     */
+    var AES = C_algo.AES = BlockCipher.extend({
+        _doReset: function () {
+            // Shortcuts
+            var key = this._key;
+            var keyWords = key.words;
+            var keySize = key.sigBytes / 4;
+
+            // Compute number of rounds
+            var nRounds = this._nRounds = keySize + 6
+
+            // Compute number of key schedule rows
+            var ksRows = (nRounds + 1) * 4;
+
+            // Compute key schedule
+            var keySchedule = this._keySchedule = [];
+            for (var ksRow = 0; ksRow < ksRows; ksRow++) {
+                if (ksRow < keySize) {
+                    keySchedule[ksRow] = keyWords[ksRow];
+                } else {
+                    var t = keySchedule[ksRow - 1];
+
+                    if (!(ksRow % keySize)) {
+                        // Rot word
+                        t = (t << 8) | (t >>> 24);
+
+                        // Sub word
+                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
+
+                        // Mix Rcon
+                        t ^= RCON[(ksRow / keySize) | 0] << 24;
+                    } else if (keySize > 6 && ksRow % keySize == 4) {
+                        // Sub word
+                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
+                    }
+
+                    keySchedule[ksRow] = keySchedule[ksRow - keySize] ^ t;
+                }
+            }
+
+            // Compute inv key schedule
+            var invKeySchedule = this._invKeySchedule = [];
+            for (var invKsRow = 0; invKsRow < ksRows; invKsRow++) {
+                var ksRow = ksRows - invKsRow;
+
+                if (invKsRow % 4) {
+                    var t = keySchedule[ksRow];
+                } else {
+                    var t = keySchedule[ksRow - 4];
+                }
+
+                if (invKsRow < 4 || ksRow <= 4) {
+                    invKeySchedule[invKsRow] = t;
+                } else {
+                    invKeySchedule[invKsRow] = INV_SUB_MIX_0[SBOX[t >>> 24]] ^ INV_SUB_MIX_1[SBOX[(t >>> 16) & 0xff]] ^
+                                               INV_SUB_MIX_2[SBOX[(t >>> 8) & 0xff]] ^ INV_SUB_MIX_3[SBOX[t & 0xff]];
+                }
+            }
+        },
+
+        encryptBlock: function (M, offset) {
+            this._doCryptBlock(M, offset, this._keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX);
+        },
+
+        decryptBlock: function (M, offset) {
+            // Swap 2nd and 4th rows
+            var t = M[offset + 1];
+            M[offset + 1] = M[offset + 3];
+            M[offset + 3] = t;
+
+            this._doCryptBlock(M, offset, this._invKeySchedule, INV_SUB_MIX_0, INV_SUB_MIX_1, INV_SUB_MIX_2, INV_SUB_MIX_3, INV_SBOX);
+
+            // Inv swap 2nd and 4th rows
+            var t = M[offset + 1];
+            M[offset + 1] = M[offset + 3];
+            M[offset + 3] = t;
+        },
+
+        _doCryptBlock: function (M, offset, keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX) {
+            // Shortcut
+            var nRounds = this._nRounds;
+
+            // Get input, add round key
+            var s0 = M[offset]     ^ keySchedule[0];
+            var s1 = M[offset + 1] ^ keySchedule[1];
+            var s2 = M[offset + 2] ^ keySchedule[2];
+            var s3 = M[offset + 3] ^ keySchedule[3];
+
+            // Key schedule row counter
+            var ksRow = 4;
+
+            // Rounds
+            for (var round = 1; round < nRounds; round++) {
+                // Shift rows, sub bytes, mix columns, add round key
+                var t0 = SUB_MIX_0[s0 >>> 24] ^ SUB_MIX_1[(s1 >>> 16) & 0xff] ^ SUB_MIX_2[(s2 >>> 8) & 0xff] ^ SUB_MIX_3[s3 & 0xff] ^ keySchedule[ksRow++];
+                var t1 = SUB_MIX_0[s1 >>> 24] ^ SUB_MIX_1[(s2 >>> 16) & 0xff] ^ SUB_MIX_2[(s3 >>> 8) & 0xff] ^ SUB_MIX_3[s0 & 0xff] ^ keySchedule[ksRow++];
+                var t2 = SUB_MIX_0[s2 >>> 24] ^ SUB_MIX_1[(s3 >>> 16) & 0xff] ^ SUB_MIX_2[(s0 >>> 8) & 0xff] ^ SUB_MIX_3[s1 & 0xff] ^ keySchedule[ksRow++];
+                var t3 = SUB_MIX_0[s3 >>> 24] ^ SUB_MIX_1[(s0 >>> 16) & 0xff] ^ SUB_MIX_2[(s1 >>> 8) & 0xff] ^ SUB_MIX_3[s2 & 0xff] ^ keySchedule[ksRow++];
+
+                // Update state
+                s0 = t0;
+                s1 = t1;
+                s2 = t2;
+                s3 = t3;
+            }
+
+            // Shift rows, sub bytes, add round key
+            var t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++];
+            var t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++];
+            var t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++];
+            var t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++];
+
+            // Set output
+            M[offset]     = t0;
+            M[offset + 1] = t1;
+            M[offset + 2] = t2;
+            M[offset + 3] = t3;
+        },
+
+        keySize: 256/32
+    });
+
+    /**
+     * Shortcut functions to the cipher's object interface.
+     *
+     * @example
+     *
+     *     var ciphertext = CryptoJS.AES.encrypt(message, key, cfg);
+     *     var plaintext  = CryptoJS.AES.decrypt(ciphertext, key, cfg);
+     */
+    C.AES = BlockCipher._createHelper(AES);
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var Hasher = C_lib.Hasher;
+    var C_algo = C.algo;
+
+    // Reusable object
+    var W = [];
+
+    /**
+     * SHA-1 hash algorithm.
+     */
+    var SHA1 = C_algo.SHA1 = Hasher.extend({
+        _doReset: function () {
+            this._hash = new WordArray.init([
+                0x67452301, 0xefcdab89,
+                0x98badcfe, 0x10325476,
+                0xc3d2e1f0
+            ]);
+        },
+
+        _doProcessBlock: function (M, offset) {
+            // Shortcut
+            var H = this._hash.words;
+
+            // Working variables
+            var a = H[0];
+            var b = H[1];
+            var c = H[2];
+            var d = H[3];
+            var e = H[4];
+
+            // Computation
+            for (var i = 0; i < 80; i++) {
+                if (i < 16) {
+                    W[i] = M[offset + i] | 0;
+                } else {
+                    var n = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];
+                    W[i] = (n << 1) | (n >>> 31);
+                }
+
+                var t = ((a << 5) | (a >>> 27)) + e + W[i];
+                if (i < 20) {
+                    t += ((b & c) | (~b & d)) + 0x5a827999;
+                } else if (i < 40) {
+                    t += (b ^ c ^ d) + 0x6ed9eba1;
+                } else if (i < 60) {
+                    t += ((b & c) | (b & d) | (c & d)) - 0x70e44324;
+                } else /* if (i < 80) */ {
+                    t += (b ^ c ^ d) - 0x359d3e2a;
+                }
+
+                e = d;
+                d = c;
+                c = (b << 30) | (b >>> 2);
+                b = a;
+                a = t;
+            }
+
+            // Intermediate hash value
+            H[0] = (H[0] + a) | 0;
+            H[1] = (H[1] + b) | 0;
+            H[2] = (H[2] + c) | 0;
+            H[3] = (H[3] + d) | 0;
+            H[4] = (H[4] + e) | 0;
+        },
+
+        _doFinalize: function () {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+
+            var nBitsTotal = this._nDataBytes * 8;
+            var nBitsLeft = data.sigBytes * 8;
+
+            // Add padding
+            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
+            data.sigBytes = dataWords.length * 4;
+
+            // Hash final blocks
+            this._process();
+
+            // Return final computed hash
+            return this._hash;
+        },
+
+        clone: function () {
+            var clone = Hasher.clone.call(this);
+            clone._hash = this._hash.clone();
+
+            return clone;
+        }
+    });
+
+    /**
+     * Shortcut function to the hasher's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     *
+     * @return {WordArray} The hash.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hash = CryptoJS.SHA1('message');
+     *     var hash = CryptoJS.SHA1(wordArray);
+     */
+    C.SHA1 = Hasher._createHelper(SHA1);
+
+    /**
+     * Shortcut function to the HMAC's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     * @param {WordArray|string} key The secret key.
+     *
+     * @return {WordArray} The HMAC.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hmac = CryptoJS.HmacSHA1(message, key);
+     */
+    C.HmacSHA1 = Hasher._createHmacHelper(SHA1);
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function (Math) {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var Hasher = C_lib.Hasher;
+    var C_algo = C.algo;
+
+    // Initialization and round constants tables
+    var H = [];
+    var K = [];
+
+    // Compute constants
+    (function () {
+        function isPrime(n) {
+            var sqrtN = Math.sqrt(n);
+            for (var factor = 2; factor <= sqrtN; factor++) {
+                if (!(n % factor)) {
+                    return false;
+                }
+            }
+
+            return true;
+        }
+
+        function getFractionalBits(n) {
+            return ((n - (n | 0)) * 0x100000000) | 0;
+        }
+
+        var n = 2;
+        var nPrime = 0;
+        while (nPrime < 64) {
+            if (isPrime(n)) {
+                if (nPrime < 8) {
+                    H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));
+                }
+                K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));
+
+                nPrime++;
+            }
+
+            n++;
+        }
+    }());
+
+    // Reusable object
+    var W = [];
+
+    /**
+     * SHA-256 hash algorithm.
+     */
+    var SHA256 = C_algo.SHA256 = Hasher.extend({
+        _doReset: function () {
+            this._hash = new WordArray.init(H.slice(0));
+        },
+
+        _doProcessBlock: function (M, offset) {
+            // Shortcut
+            var H = this._hash.words;
+
+            // Working variables
+            var a = H[0];
+            var b = H[1];
+            var c = H[2];
+            var d = H[3];
+            var e = H[4];
+            var f = H[5];
+            var g = H[6];
+            var h = H[7];
+
+            // Computation
+            for (var i = 0; i < 64; i++) {
+                if (i < 16) {
+                    W[i] = M[offset + i] | 0;
+                } else {
+                    var gamma0x = W[i - 15];
+                    var gamma0  = ((gamma0x << 25) | (gamma0x >>> 7))  ^
+                                  ((gamma0x << 14) | (gamma0x >>> 18)) ^
+                                   (gamma0x >>> 3);
+
+                    var gamma1x = W[i - 2];
+                    var gamma1  = ((gamma1x << 15) | (gamma1x >>> 17)) ^
+                                  ((gamma1x << 13) | (gamma1x >>> 19)) ^
+                                   (gamma1x >>> 10);
+
+                    W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];
+                }
+
+                var ch  = (e & f) ^ (~e & g);
+                var maj = (a & b) ^ (a & c) ^ (b & c);
+
+                var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));
+                var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7)  | (e >>> 25));
+
+                var t1 = h + sigma1 + ch + K[i] + W[i];
+                var t2 = sigma0 + maj;
+
+                h = g;
+                g = f;
+                f = e;
+                e = (d + t1) | 0;
+                d = c;
+                c = b;
+                b = a;
+                a = (t1 + t2) | 0;
+            }
+
+            // Intermediate hash value
+            H[0] = (H[0] + a) | 0;
+            H[1] = (H[1] + b) | 0;
+            H[2] = (H[2] + c) | 0;
+            H[3] = (H[3] + d) | 0;
+            H[4] = (H[4] + e) | 0;
+            H[5] = (H[5] + f) | 0;
+            H[6] = (H[6] + g) | 0;
+            H[7] = (H[7] + h) | 0;
+        },
+
+        _doFinalize: function () {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+
+            var nBitsTotal = this._nDataBytes * 8;
+            var nBitsLeft = data.sigBytes * 8;
+
+            // Add padding
+            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
+            data.sigBytes = dataWords.length * 4;
+
+            // Hash final blocks
+            this._process();
+
+            // Return final computed hash
+            return this._hash;
+        },
+
+        clone: function () {
+            var clone = Hasher.clone.call(this);
+            clone._hash = this._hash.clone();
+
+            return clone;
+        }
+    });
+
+    /**
+     * Shortcut function to the hasher's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     *
+     * @return {WordArray} The hash.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hash = CryptoJS.SHA256('message');
+     *     var hash = CryptoJS.SHA256(wordArray);
+     */
+    C.SHA256 = Hasher._createHelper(SHA256);
+
+    /**
+     * Shortcut function to the HMAC's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     * @param {WordArray|string} key The secret key.
+     *
+     * @return {WordArray} The HMAC.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hmac = CryptoJS.HmacSHA256(message, key);
+     */
+    C.HmacSHA256 = Hasher._createHmacHelper(SHA256);
+}(Math));
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var Base = C_lib.Base;
+    var C_enc = C.enc;
+    var Utf8 = C_enc.Utf8;
+    var C_algo = C.algo;
+
+    /**
+     * HMAC algorithm.
+     */
+    var HMAC = C_algo.HMAC = Base.extend({
+        /**
+         * Initializes a newly created HMAC.
+         *
+         * @param {Hasher} hasher The hash algorithm to use.
+         * @param {WordArray|string} key The secret key.
+         *
+         * @example
+         *
+         *     var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);
+         */
+        init: function (hasher, key) {
+            // Init hasher
+            hasher = this._hasher = new hasher.init();
+
+            // Convert string to WordArray, else assume WordArray already
+            if (typeof key == 'string') {
+                key = Utf8.parse(key);
+            }
+
+            // Shortcuts
+            var hasherBlockSize = hasher.blockSize;
+            var hasherBlockSizeBytes = hasherBlockSize * 4;
+
+            // Allow arbitrary length keys
+            if (key.sigBytes > hasherBlockSizeBytes) {
+                key = hasher.finalize(key);
+            }
+
+            // Clamp excess bits
+            key.clamp();
+
+            // Clone key for inner and outer pads
+            var oKey = this._oKey = key.clone();
+            var iKey = this._iKey = key.clone();
+
+            // Shortcuts
+            var oKeyWords = oKey.words;
+            var iKeyWords = iKey.words;
+
+            // XOR keys with pad constants
+            for (var i = 0; i < hasherBlockSize; i++) {
+                oKeyWords[i] ^= 0x5c5c5c5c;
+                iKeyWords[i] ^= 0x36363636;
+            }
+            oKey.sigBytes = iKey.sigBytes = hasherBlockSizeBytes;
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this HMAC to its initial state.
+         *
+         * @example
+         *
+         *     hmacHasher.reset();
+         */
+        reset: function () {
+            // Shortcut
+            var hasher = this._hasher;
+
+            // Reset
+            hasher.reset();
+            hasher.update(this._iKey);
+        },
+
+        /**
+         * Updates this HMAC with a message.
+         *
+         * @param {WordArray|string} messageUpdate The message to append.
+         *
+         * @return {HMAC} This HMAC instance.
+         *
+         * @example
+         *
+         *     hmacHasher.update('message');
+         *     hmacHasher.update(wordArray);
+         */
+        update: function (messageUpdate) {
+            this._hasher.update(messageUpdate);
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Finalizes the HMAC computation.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} messageUpdate (Optional) A final message update.
+         *
+         * @return {WordArray} The HMAC.
+         *
+         * @example
+         *
+         *     var hmac = hmacHasher.finalize();
+         *     var hmac = hmacHasher.finalize('message');
+         *     var hmac = hmacHasher.finalize(wordArray);
+         */
+        finalize: function (messageUpdate) {
+            // Shortcut
+            var hasher = this._hasher;
+
+            // Compute HMAC
+            var innerHash = hasher.finalize(messageUpdate);
+            hasher.reset();
+            var hmac = hasher.finalize(this._oKey.clone().concat(innerHash));
+
+            return hmac;
+        }
+    });
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * A noop padding strategy.
+ */
+CryptoJS.pad.NoPadding = {
+    pad: function () {
+    },
+
+    unpad: function () {
+    }
+};
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * Counter block mode.
+ */
+CryptoJS.mode.CTR = (function () {
+    var CTR = CryptoJS.lib.BlockCipherMode.extend();
+
+    var Encryptor = CTR.Encryptor = CTR.extend({
+        processBlock: function (words, offset) {
+            // Shortcuts
+            var cipher = this._cipher
+            var blockSize = cipher.blockSize;
+            var iv = this._iv;
+            var counter = this._counter;
+
+            // Generate keystream
+            if (iv) {
+                counter = this._counter = iv.slice(0);
+
+                // Remove IV for subsequent blocks
+                this._iv = undefined;
+            }
+            var keystream = counter.slice(0);
+            cipher.encryptBlock(keystream, 0);
+
+            // Increment counter
+            counter[blockSize - 1] = (counter[blockSize - 1] + 1) | 0
+
+            // Encrypt
+            for (var i = 0; i < blockSize; i++) {
+                words[offset + i] ^= keystream[i];
+            }
+        }
+    });
+
+    CTR.Decryptor = Encryptor;
+
+    return CTR;
+}());
+
+
+  return CryptoJS
+
+}))
\ No newline at end of file
diff --git a/lib/otr/build/dep/eventemitter.js b/lib/otr/build/dep/eventemitter.js
new file mode 100644
index 0000000..0443d1f
--- /dev/null
+++ b/lib/otr/build/dep/eventemitter.js
@@ -0,0 +1,455 @@
+/*!
+ * EventEmitter v4.2.3 - git.io/ee
+ * Oliver Caldwell
+ * MIT license
+ * @preserve
+ */
+
+(function () {
+	'use strict';
+
+	/**
+	 * Class for managing events.
+	 * Can be extended to provide event functionality in other classes.
+	 *
+	 * @class EventEmitter Manages event registering and emitting.
+	 */
+	function EventEmitter() {}
+
+	// Shortcuts to improve speed and size
+
+	// Easy access to the prototype
+	var proto = EventEmitter.prototype;
+
+	/**
+	 * Finds the index of the listener for the event in it's storage array.
+	 *
+	 * @param {Function[]} listeners Array of listeners to search through.
+	 * @param {Function} listener Method to look for.
+	 * @return {Number} Index of the specified listener, -1 if not found
+	 * @api private
+	 */
+	function indexOfListener(listeners, listener) {
+		var i = listeners.length;
+		while (i--) {
+			if (listeners[i].listener === listener) {
+				return i;
+			}
+		}
+
+		return -1;
+	}
+
+	/**
+	 * Alias a method while keeping the context correct, to allow for overwriting of target method.
+	 *
+	 * @param {String} name The name of the target method.
+	 * @return {Function} The aliased method
+	 * @api private
+	 */
+	function alias(name) {
+		return function aliasClosure() {
+			return this[name].apply(this, arguments);
+		};
+	}
+
+	/**
+	 * Returns the listener array for the specified event.
+	 * Will initialise the event object and listener arrays if required.
+	 * Will return an object if you use a regex search. The object contains keys for each matched event. So /ba[rz]/ might return an object containing bar and baz. But only if you have either defined them with defineEvent or added some listeners to them.
+	 * Each property in the object response is an array of listener functions.
+	 *
+	 * @param {String|RegExp} evt Name of the event to return the listeners from.
+	 * @return {Function[]|Object} All listener functions for the event.
+	 */
+	proto.getListeners = function getListeners(evt) {
+		var events = this._getEvents();
+		var response;
+		var key;
+
+		// Return a concatenated array of all matching events if
+		// the selector is a regular expression.
+		if (typeof evt === 'object') {
+			response = {};
+			for (key in events) {
+				if (events.hasOwnProperty(key) && evt.test(key)) {
+					response[key] = events[key];
+				}
+			}
+		}
+		else {
+			response = events[evt] || (events[evt] = []);
+		}
+
+		return response;
+	};
+
+	/**
+	 * Takes a list of listener objects and flattens it into a list of listener functions.
+	 *
+	 * @param {Object[]} listeners Raw listener objects.
+	 * @return {Function[]} Just the listener functions.
+	 */
+	proto.flattenListeners = function flattenListeners(listeners) {
+		var flatListeners = [];
+		var i;
+
+		for (i = 0; i < listeners.length; i += 1) {
+			flatListeners.push(listeners[i].listener);
+		}
+
+		return flatListeners;
+	};
+
+	/**
+	 * Fetches the requested listeners via getListeners but will always return the results inside an object. This is mainly for internal use but others may find it useful.
+	 *
+	 * @param {String|RegExp} evt Name of the event to return the listeners from.
+	 * @return {Object} All listener functions for an event in an object.
+	 */
+	proto.getListenersAsObject = function getListenersAsObject(evt) {
+		var listeners = this.getListeners(evt);
+		var response;
+
+		if (listeners instanceof Array) {
+			response = {};
+			response[evt] = listeners;
+		}
+
+		return response || listeners;
+	};
+
+	/**
+	 * Adds a listener function to the specified event.
+	 * The listener will not be added if it is a duplicate.
+	 * If the listener returns true then it will be removed after it is called.
+	 * If you pass a regular expression as the event name then the listener will be added to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to attach the listener to.
+	 * @param {Function} listener Method to be called when the event is emitted. If the function returns true then it will be removed after calling.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addListener = function addListener(evt, listener) {
+		var listeners = this.getListenersAsObject(evt);
+		var listenerIsWrapped = typeof listener === 'object';
+		var key;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key) && indexOfListener(listeners[key], listener) === -1) {
+				listeners[key].push(listenerIsWrapped ? listener : {
+					listener: listener,
+					once: false
+				});
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of addListener
+	 */
+	proto.on = alias('addListener');
+
+	/**
+	 * Semi-alias of addListener. It will add a listener that will be
+	 * automatically removed after it's first execution.
+	 *
+	 * @param {String|RegExp} evt Name of the event to attach the listener to.
+	 * @param {Function} listener Method to be called when the event is emitted. If the function returns true then it will be removed after calling.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addOnceListener = function addOnceListener(evt, listener) {
+		return this.addListener(evt, {
+			listener: listener,
+			once: true
+		});
+	};
+
+	/**
+	 * Alias of addOnceListener.
+	 */
+	proto.once = alias('addOnceListener');
+
+	/**
+	 * Defines an event name. This is required if you want to use a regex to add a listener to multiple events at once. If you don't do this then how do you expect it to know what event to add to? Should it just add to every possible match for a regex? No. That is scary and bad.
+	 * You need to tell it what event names should be matched by a regex.
+	 *
+	 * @param {String} evt Name of the event to create.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.defineEvent = function defineEvent(evt) {
+		this.getListeners(evt);
+		return this;
+	};
+
+	/**
+	 * Uses defineEvent to define multiple events.
+	 *
+	 * @param {String[]} evts An array of event names to define.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.defineEvents = function defineEvents(evts) {
+		for (var i = 0; i < evts.length; i += 1) {
+			this.defineEvent(evts[i]);
+		}
+		return this;
+	};
+
+	/**
+	 * Removes a listener function from the specified event.
+	 * When passed a regular expression as the event name, it will remove the listener from all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to remove the listener from.
+	 * @param {Function} listener Method to remove from the event.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeListener = function removeListener(evt, listener) {
+		var listeners = this.getListenersAsObject(evt);
+		var index;
+		var key;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key)) {
+				index = indexOfListener(listeners[key], listener);
+
+				if (index !== -1) {
+					listeners[key].splice(index, 1);
+				}
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of removeListener
+	 */
+	proto.off = alias('removeListener');
+
+	/**
+	 * Adds listeners in bulk using the manipulateListeners method.
+	 * If you pass an object as the second argument you can add to multiple events at once. The object should contain key value pairs of events and listeners or listener arrays. You can also pass it an event name and an array of listeners to be added.
+	 * You can also pass it a regular expression to add the array of listeners to all events that match it.
+	 * Yeah, this function does quite a bit. That's probably a bad thing.
+	 *
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to add to multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to add.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addListeners = function addListeners(evt, listeners) {
+		// Pass through to manipulateListeners
+		return this.manipulateListeners(false, evt, listeners);
+	};
+
+	/**
+	 * Removes listeners in bulk using the manipulateListeners method.
+	 * If you pass an object as the second argument you can remove from multiple events at once. The object should contain key value pairs of events and listeners or listener arrays.
+	 * You can also pass it an event name and an array of listeners to be removed.
+	 * You can also pass it a regular expression to remove the listeners from all events that match it.
+	 *
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to remove from multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to remove.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeListeners = function removeListeners(evt, listeners) {
+		// Pass through to manipulateListeners
+		return this.manipulateListeners(true, evt, listeners);
+	};
+
+	/**
+	 * Edits listeners in bulk. The addListeners and removeListeners methods both use this to do their job. You should really use those instead, this is a little lower level.
+	 * The first argument will determine if the listeners are removed (true) or added (false).
+	 * If you pass an object as the second argument you can add/remove from multiple events at once. The object should contain key value pairs of events and listeners or listener arrays.
+	 * You can also pass it an event name and an array of listeners to be added/removed.
+	 * You can also pass it a regular expression to manipulate the listeners of all events that match it.
+	 *
+	 * @param {Boolean} remove True if you want to remove listeners, false if you want to add.
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to add/remove from multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to add/remove.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.manipulateListeners = function manipulateListeners(remove, evt, listeners) {
+		var i;
+		var value;
+		var single = remove ? this.removeListener : this.addListener;
+		var multiple = remove ? this.removeListeners : this.addListeners;
+
+		// If evt is an object then pass each of it's properties to this method
+		if (typeof evt === 'object' && !(evt instanceof RegExp)) {
+			for (i in evt) {
+				if (evt.hasOwnProperty(i) && (value = evt[i])) {
+					// Pass the single listener straight through to the singular method
+					if (typeof value === 'function') {
+						single.call(this, i, value);
+					}
+					else {
+						// Otherwise pass back to the multiple function
+						multiple.call(this, i, value);
+					}
+				}
+			}
+		}
+		else {
+			// So evt must be a string
+			// And listeners must be an array of listeners
+			// Loop over it and pass each one to the multiple method
+			i = listeners.length;
+			while (i--) {
+				single.call(this, evt, listeners[i]);
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Removes all listeners from a specified event.
+	 * If you do not specify an event then all listeners will be removed.
+	 * That means every event will be emptied.
+	 * You can also pass a regex to remove all events that match it.
+	 *
+	 * @param {String|RegExp} [evt] Optional name of the event to remove all listeners for. Will remove from every event if not passed.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeEvent = function removeEvent(evt) {
+		var type = typeof evt;
+		var events = this._getEvents();
+		var key;
+
+		// Remove different things depending on the state of evt
+		if (type === 'string') {
+			// Remove all listeners for the specified event
+			delete events[evt];
+		}
+		else if (type === 'object') {
+			// Remove all events matching the regex.
+			for (key in events) {
+				if (events.hasOwnProperty(key) && evt.test(key)) {
+					delete events[key];
+				}
+			}
+		}
+		else {
+			// Remove all listeners in all events
+			delete this._events;
+		}
+
+		return this;
+	};
+
+	/**
+	 * Emits an event of your choice.
+	 * When emitted, every listener attached to that event will be executed.
+	 * If you pass the optional argument array then those arguments will be passed to every listener upon execution.
+	 * Because it uses `apply`, your array of arguments will be passed as if you wrote them out separately.
+	 * So they will not arrive within the array on the other side, they will be separate.
+	 * You can also pass a regular expression to emit to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to emit and execute listeners for.
+	 * @param {Array} [args] Optional array of arguments to be passed to each listener.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.emitEvent = function emitEvent(evt, args) {
+		var listeners = this.getListenersAsObject(evt);
+		var listener;
+		var i;
+		var key;
+		var response;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key)) {
+				i = listeners[key].length;
+
+				while (i--) {
+					// If the listener returns true then it shall be removed from the event
+					// The function is executed either with a basic call or an apply if there is an args array
+					listener = listeners[key][i];
+
+					if (listener.once === true) {
+						this.removeListener(evt, listener.listener);
+					}
+
+					response = listener.listener.apply(this, args || []);
+
+					if (response === this._getOnceReturnValue()) {
+						this.removeListener(evt, listener.listener);
+					}
+				}
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of emitEvent
+	 */
+	proto.trigger = alias('emitEvent');
+
+	/**
+	 * Subtly different from emitEvent in that it will pass its arguments on to the listeners, as opposed to taking a single array of arguments to pass on.
+	 * As with emitEvent, you can pass a regex in place of the event name to emit to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to emit and execute listeners for.
+	 * @param {...*} Optional additional arguments to be passed to each listener.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.emit = function emit(evt) {
+		var args = Array.prototype.slice.call(arguments, 1);
+		return this.emitEvent(evt, args);
+	};
+
+	/**
+	 * Sets the current value to check against when executing listeners. If a
+	 * listeners return value matches the one set here then it will be removed
+	 * after execution. This value defaults to true.
+	 *
+	 * @param {*} value The new value to check for when executing listeners.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.setOnceReturnValue = function setOnceReturnValue(value) {
+		this._onceReturnValue = value;
+		return this;
+	};
+
+	/**
+	 * Fetches the current value to check against when executing listeners. If
+	 * the listeners return value matches this one then it should be removed
+	 * automatically. It will return true by default.
+	 *
+	 * @return {*|Boolean} The current value to check for or the default, true.
+	 * @api private
+	 */
+	proto._getOnceReturnValue = function _getOnceReturnValue() {
+		if (this.hasOwnProperty('_onceReturnValue')) {
+			return this._onceReturnValue;
+		}
+		else {
+			return true;
+		}
+	};
+
+	/**
+	 * Fetches the events object and creates one if required.
+	 *
+	 * @return {Object} The events storage object.
+	 * @api private
+	 */
+	proto._getEvents = function _getEvents() {
+		return this._events || (this._events = {});
+	};
+
+	// Expose the class either via AMD, CommonJS or the global object
+	if (typeof define === 'function' && define.amd) {
+		define(function () {
+			return EventEmitter;
+		});
+	}
+	else if (typeof module === 'object' && module.exports){
+		module.exports = EventEmitter;
+	}
+	else {
+		this.EventEmitter = EventEmitter;
+	}
+}.call(this));
diff --git a/lib/otr/build/dep/salsa20.js b/lib/otr/build/dep/salsa20.js
new file mode 100644
index 0000000..485314b
--- /dev/null
+++ b/lib/otr/build/dep/salsa20.js
@@ -0,0 +1,254 @@
+// Salsa20 implementation
+// Contributed to Cryptocat by Dmitry Chestnykh
+// 21-01-2013
+
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define(factory)
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory()
+  } else {
+    root.Salsa20 = factory()
+  }
+
+}(this, function () {
+
+    function Salsa20(key, nonce) {
+        // Constants.
+        this.rounds = 20; // number of Salsa rounds
+        this.sigmaWords = [0x61707865, 0x3320646e, 0x79622d32, 0x6b206574];
+
+        // State.
+        this.keyWords = [];           // key words
+        this.nonceWords = [0, 0];     // nonce words
+        this.counterWords = [0, 0];   // block counter words
+
+        // Output buffer.
+        this.block = [];        // output block of 64 bytes
+        this.blockUsed = 64;     // number of block bytes used
+
+        this.setKey(key);
+        this.setNonce(nonce);
+    }
+
+    // setKey sets the key to the given 32-byte array.
+    Salsa20.prototype.setKey = function(key) {
+        for (var i = 0, j = 0; i < 8; i++, j += 4) {
+            this.keyWords[i] = (key[j] & 0xff)        |
+                              ((key[j+1] & 0xff)<<8)  |
+                              ((key[j+2] & 0xff)<<16) |
+                              ((key[j+3] & 0xff)<<24);
+        }
+        this._reset();
+    };
+
+    // setNonce sets the nonce to the given 8-byte array.
+    Salsa20.prototype.setNonce = function(nonce) {
+        this.nonceWords[0] = (nonce[0] & 0xff)      |
+                            ((nonce[1] & 0xff)<<8)  |
+                            ((nonce[2] & 0xff)<<16) |
+                            ((nonce[3] & 0xff)<<24);
+        this.nonceWords[1] = (nonce[4] & 0xff)      |
+                            ((nonce[5] & 0xff)<<8)  |
+                            ((nonce[6] & 0xff)<<16) |
+                            ((nonce[7] & 0xff)<<24);
+        this._reset();
+    };
+
+    // getBytes returns the next numberOfBytes bytes of stream.
+    Salsa20.prototype.getBytes = function(numberOfBytes) {
+        var out = new Array(numberOfBytes);
+        for (var i = 0; i < numberOfBytes; i++) {
+            if (this.blockUsed == 64) {
+                this._generateBlock();
+                this._incrementCounter();
+                this.blockUsed = 0;
+            }
+            out[i] = this.block[this.blockUsed];
+            this.blockUsed++;
+        }
+        return out;
+    };
+
+    Salsa20.prototype.getHexString = function(numberOfBytes) {
+        var hex=['0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f'];
+        var out = [];
+        var bytes = this.getBytes(numberOfBytes);
+        for(var i = 0; i < bytes.length; i++) {
+            out.push(hex[(bytes[i] >> 4) & 15]);
+            out.push(hex[bytes[i] & 15]);
+        }
+        return out.join('');
+    };
+
+    // Private methods.
+
+    Salsa20.prototype._reset = function() {
+        this.counterWords[0] = 0;
+        this.counterWords[1] = 0;
+        this.blockUsed = 64;
+    };
+
+    // _incrementCounter increments block counter.
+    Salsa20.prototype._incrementCounter = function() {
+        // Note: maximum 2^64 blocks.
+        this.counterWords[0] = (this.counterWords[0] + 1) & 0xffffffff;
+        if (this.counterWords[0] == 0) {
+            this.counterWords[1] = (this.counterWords[1] + 1) & 0xffffffff;
+        }
+    };
+
+    // _generateBlock generates 64 bytes from key, nonce, and counter,
+    // and puts the result into this.block.
+    Salsa20.prototype._generateBlock = function() {
+        var j0 = this.sigmaWords[0],
+            j1 = this.keyWords[0],
+            j2 = this.keyWords[1],
+            j3 = this.keyWords[2],
+            j4 = this.keyWords[3],
+            j5 = this.sigmaWords[1],
+            j6 = this.nonceWords[0],
+            j7 = this.nonceWords[1],
+            j8 = this.counterWords[0],
+            j9 = this.counterWords[1],
+            j10 = this.sigmaWords[2],
+            j11 = this.keyWords[4],
+            j12 = this.keyWords[5],
+            j13 = this.keyWords[6],
+            j14 = this.keyWords[7],
+            j15 = this.sigmaWords[3];
+
+            var x0 = j0, x1 = j1, x2 = j2, x3 = j3, x4 = j4, x5 = j5, x6 = j6, x7 = j7,
+                x8 = j8, x9 = j9, x10 = j10, x11 = j11, x12 = j12, x13 = j13, x14 = j14, x15 = j15;
+
+            var u;
+
+            for (var i = 0; i < this.rounds; i += 2) {
+                u = x0 + x12;
+                x4 ^= (u<<7) | (u>>>(32-7));
+                u = x4 + x0;
+                x8 ^= (u<<9) | (u>>>(32-9));
+                u = x8 + x4;
+                x12 ^= (u<<13) | (u>>>(32-13));
+                u = x12 + x8;
+                x0 ^= (u<<18) | (u>>>(32-18));
+
+                u = x5 + x1;
+                x9 ^= (u<<7) | (u>>>(32-7));
+                u = x9 + x5;
+                x13 ^= (u<<9) | (u>>>(32-9));
+                u = x13 + x9;
+                x1 ^= (u<<13) | (u>>>(32-13));
+                u = x1 + x13;
+                x5 ^= (u<<18) | (u>>>(32-18));
+
+                u = x10 + x6;
+                x14 ^= (u<<7) | (u>>>(32-7));
+                u = x14 + x10;
+                x2 ^= (u<<9) | (u>>>(32-9));
+                u = x2 + x14;
+                x6 ^= (u<<13) | (u>>>(32-13));
+                u = x6 + x2;
+                x10 ^= (u<<18) | (u>>>(32-18));
+
+                u = x15 + x11;
+                x3 ^= (u<<7) | (u>>>(32-7));
+                u = x3 + x15;
+                x7 ^= (u<<9) | (u>>>(32-9));
+                u = x7 + x3;
+                x11 ^= (u<<13) | (u>>>(32-13));
+                u = x11 + x7;
+                x15 ^= (u<<18) | (u>>>(32-18));
+
+                u = x0 + x3;
+                x1 ^= (u<<7) | (u>>>(32-7));
+                u = x1 + x0;
+                x2 ^= (u<<9) | (u>>>(32-9));
+                u = x2 + x1;
+                x3 ^= (u<<13) | (u>>>(32-13));
+                u = x3 + x2;
+                x0 ^= (u<<18) | (u>>>(32-18));
+
+                u = x5 + x4;
+                x6 ^= (u<<7) | (u>>>(32-7));
+                u = x6 + x5;
+                x7 ^= (u<<9) | (u>>>(32-9));
+                u = x7 + x6;
+                x4 ^= (u<<13) | (u>>>(32-13));
+                u = x4 + x7;
+                x5 ^= (u<<18) | (u>>>(32-18));
+
+                u = x10 + x9;
+                x11 ^= (u<<7) | (u>>>(32-7));
+                u = x11 + x10;
+                x8 ^= (u<<9) | (u>>>(32-9));
+                u = x8 + x11;
+                x9 ^= (u<<13) | (u>>>(32-13));
+                u = x9 + x8;
+                x10 ^= (u<<18) | (u>>>(32-18));
+
+                u = x15 + x14;
+                x12 ^= (u<<7) | (u>>>(32-7));
+                u = x12 + x15;
+                x13 ^= (u<<9) | (u>>>(32-9));
+                u = x13 + x12;
+                x14 ^= (u<<13) | (u>>>(32-13));
+                u = x14 + x13;
+                x15 ^= (u<<18) | (u>>>(32-18));
+            }
+
+            x0 += j0;
+            x1 += j1;
+            x2 += j2;
+            x3 += j3;
+            x4 += j4;
+            x5 += j5;
+            x6 += j6;
+            x7 += j7;
+            x8 += j8;
+            x9 += j9;
+            x10 += j10;
+            x11 += j11;
+            x12 += j12;
+            x13 += j13;
+            x14 += j14;
+            x15 += j15;
+
+            this.block[ 0] = ( x0 >>>  0) & 0xff; this.block[ 1] = ( x0 >>>  8) & 0xff;
+            this.block[ 2] = ( x0 >>> 16) & 0xff; this.block[ 3] = ( x0 >>> 24) & 0xff;
+            this.block[ 4] = ( x1 >>>  0) & 0xff; this.block[ 5] = ( x1 >>>  8) & 0xff;
+            this.block[ 6] = ( x1 >>> 16) & 0xff; this.block[ 7] = ( x1 >>> 24) & 0xff;
+            this.block[ 8] = ( x2 >>>  0) & 0xff; this.block[ 9] = ( x2 >>>  8) & 0xff;
+            this.block[10] = ( x2 >>> 16) & 0xff; this.block[11] = ( x2 >>> 24) & 0xff;
+            this.block[12] = ( x3 >>>  0) & 0xff; this.block[13] = ( x3 >>>  8) & 0xff;
+            this.block[14] = ( x3 >>> 16) & 0xff; this.block[15] = ( x3 >>> 24) & 0xff;
+            this.block[16] = ( x4 >>>  0) & 0xff; this.block[17] = ( x4 >>>  8) & 0xff;
+            this.block[18] = ( x4 >>> 16) & 0xff; this.block[19] = ( x4 >>> 24) & 0xff;
+            this.block[20] = ( x5 >>>  0) & 0xff; this.block[21] = ( x5 >>>  8) & 0xff;
+            this.block[22] = ( x5 >>> 16) & 0xff; this.block[23] = ( x5 >>> 24) & 0xff;
+            this.block[24] = ( x6 >>>  0) & 0xff; this.block[25] = ( x6 >>>  8) & 0xff;
+            this.block[26] = ( x6 >>> 16) & 0xff; this.block[27] = ( x6 >>> 24) & 0xff;
+            this.block[28] = ( x7 >>>  0) & 0xff; this.block[29] = ( x7 >>>  8) & 0xff;
+            this.block[30] = ( x7 >>> 16) & 0xff; this.block[31] = ( x7 >>> 24) & 0xff;
+            this.block[32] = ( x8 >>>  0) & 0xff; this.block[33] = ( x8 >>>  8) & 0xff;
+            this.block[34] = ( x8 >>> 16) & 0xff; this.block[35] = ( x8 >>> 24) & 0xff;
+            this.block[36] = ( x9 >>>  0) & 0xff; this.block[37] = ( x9 >>>  8) & 0xff;
+            this.block[38] = ( x9 >>> 16) & 0xff; this.block[39] = ( x9 >>> 24) & 0xff;
+            this.block[40] = (x10 >>>  0) & 0xff; this.block[41] = (x10 >>>  8) & 0xff;
+            this.block[42] = (x10 >>> 16) & 0xff; this.block[43] = (x10 >>> 24) & 0xff;
+            this.block[44] = (x11 >>>  0) & 0xff; this.block[45] = (x11 >>>  8) & 0xff;
+            this.block[46] = (x11 >>> 16) & 0xff; this.block[47] = (x11 >>> 24) & 0xff;
+            this.block[48] = (x12 >>>  0) & 0xff; this.block[49] = (x12 >>>  8) & 0xff;
+            this.block[50] = (x12 >>> 16) & 0xff; this.block[51] = (x12 >>> 24) & 0xff;
+            this.block[52] = (x13 >>>  0) & 0xff; this.block[53] = (x13 >>>  8) & 0xff;
+            this.block[54] = (x13 >>> 16) & 0xff; this.block[55] = (x13 >>> 24) & 0xff;
+            this.block[56] = (x14 >>>  0) & 0xff; this.block[57] = (x14 >>>  8) & 0xff;
+            this.block[58] = (x14 >>> 16) & 0xff; this.block[59] = (x14 >>> 24) & 0xff;
+            this.block[60] = (x15 >>>  0) & 0xff; this.block[61] = (x15 >>>  8) & 0xff;
+            this.block[62] = (x15 >>> 16) & 0xff; this.block[63] = (x15 >>> 24) & 0xff;
+    };
+
+  return Salsa20
+
+}))
\ No newline at end of file
diff --git a/lib/otr/build/dsa-webworker.js b/lib/otr/build/dsa-webworker.js
new file mode 100644
index 0000000..d84ad17
--- /dev/null
+++ b/lib/otr/build/dsa-webworker.js
@@ -0,0 +1,52 @@
+;(function (root) {
+  "use strict";
+
+  root.OTR = {}
+  root.DSA = {}
+  root.crypto = {
+    randomBytes: function () {
+      throw new Error("Haven't seeded yet.")
+    }
+  }
+
+  // default imports
+  var imports = [
+      'vendor/salsa20.js'
+    , 'vendor/bigint.js'
+    , 'vendor/crypto.js'
+    , 'vendor/eventemitter.js'
+    , 'lib/const.js'
+    , 'lib/helpers.js'
+    , 'lib/dsa.js'
+  ]
+
+  function sendMsg(type, val) {
+    postMessage({ type: type, val: val })
+  }
+
+  onmessage = function (e) {
+    var data = e.data;
+
+    if (data.imports) imports = data.imports
+    importScripts.apply(root, imports);
+
+    // use salsa20 since there's no prng in webworkers
+    var state = new root.Salsa20(data.seed.slice(0, 32), data.seed.slice(32))
+    root.crypto.randomBytes = function (n) {
+      return state.getBytes(n)
+    }
+
+    if (data.debug) sendMsg('debug', 'DSA key creation started')
+    var dsa
+    try {
+      dsa = new root.DSA()
+    } catch (e) {
+      if (data.debug) sendMsg('debug', e.toString())
+      return
+    }
+    if (data.debug) sendMsg('debug', 'DSA key creation finished')
+
+    sendMsg('data', dsa.packPrivate())
+  }
+
+}(this))
\ No newline at end of file
diff --git a/lib/otr/build/otr.js b/lib/otr/build/otr.js
new file mode 100644
index 0000000..418ac3d
--- /dev/null
+++ b/lib/otr/build/otr.js
@@ -0,0 +1,2631 @@
+/*!
+
+  otr.js v0.2.13 - 2014-09-07
+  (c) 2014 - Arlo Breault <arlolra at gmail.com>
+  Freely distributed under the MPL v2.0 license.
+
+  This file is concatenated for the browser.
+  Please see: https://github.com/arlolra/otr
+
+*/
+
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define([
+        "bigint"
+      , "crypto"
+      , "eventemitter"
+    ], function (BigInt, CryptoJS, EventEmitter) {
+      var root = {
+          BigInt: BigInt
+        , CryptoJS: CryptoJS
+        , EventEmitter: EventEmitter
+        , OTR: {}
+        , DSA: {}
+      }
+      return factory.call(root)
+    })
+  } else {
+    root.OTR = {}
+    root.DSA = {}
+    factory.call(root)
+  }
+
+}(this, function () {
+
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CONST = {
+
+    // diffie-heilman
+      N : 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7EDEE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB9ED529077096966D670C354E4ABC9804F1746C08CA237327FFFFFFFFFFFFFFFF'
+    , G : '2'
+
+    // otr message states
+    , MSGSTATE_PLAINTEXT : 0
+    , MSGSTATE_ENCRYPTED : 1
+    , MSGSTATE_FINISHED  : 2
+
+    // otr auth states
+    , AUTHSTATE_NONE               : 0
+    , AUTHSTATE_AWAITING_DHKEY     : 1
+    , AUTHSTATE_AWAITING_REVEALSIG : 2
+    , AUTHSTATE_AWAITING_SIG       : 3
+
+    // whitespace tags
+    , WHITESPACE_TAG    : '\x20\x09\x20\x20\x09\x09\x09\x09\x20\x09\x20\x09\x20\x09\x20\x20'
+    , WHITESPACE_TAG_V2 : '\x20\x20\x09\x09\x20\x20\x09\x20'
+    , WHITESPACE_TAG_V3 : '\x20\x20\x09\x09\x20\x20\x09\x09'
+
+    // otr tags
+    , OTR_TAG       : '?OTR'
+    , OTR_VERSION_1 : '\x00\x01'
+    , OTR_VERSION_2 : '\x00\x02'
+    , OTR_VERSION_3 : '\x00\x03'
+
+    // smp machine states
+    , SMPSTATE_EXPECT0 : 0
+    , SMPSTATE_EXPECT1 : 1
+    , SMPSTATE_EXPECT2 : 2
+    , SMPSTATE_EXPECT3 : 3
+    , SMPSTATE_EXPECT4 : 4
+
+    // unstandard status codes
+    , STATUS_SEND_QUERY  : 0
+    , STATUS_AKE_INIT    : 1
+    , STATUS_AKE_SUCCESS : 2
+    , STATUS_END_OTR     : 3
+
+  }
+
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = CONST
+  } else {
+    root.OTR.CONST = CONST
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var HLP = {}, CryptoJS, BigInt
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = HLP = {}
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+  } else {
+    if (root.OTR) root.OTR.HLP = HLP
+    if (root.DSA) root.DSA.HLP = HLP
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+  }
+
+  // data types (byte lengths)
+  var DTS = {
+      BYTE  : 1
+    , SHORT : 2
+    , INT   : 4
+    , CTR   : 8
+    , MAC   : 20
+    , SIG   : 40
+  }
+
+  // otr message wrapper begin and end
+  var WRAPPER_BEGIN = "?OTR"
+    , WRAPPER_END   = "."
+
+  var TWO = BigInt.str2bigInt('2', 10)
+
+  HLP.debug = function (msg) {
+    // used as HLP.debug.call(ctx, msg)
+    if ( this.debug &&
+         typeof this.debug !== 'function' &&
+         typeof console !== 'undefined'
+    ) console.log(msg)
+  }
+
+  HLP.extend = function (child, parent) {
+    for (var key in parent) {
+      if (Object.hasOwnProperty.call(parent, key))
+        child[key] = parent[key]
+    }
+    function Ctor() { this.constructor = child }
+    Ctor.prototype = parent.prototype
+    child.prototype = new Ctor()
+    child.__super__ = parent.prototype
+  }
+
+  // assumes 32-bit
+  function intCompare(x, y) {
+    var z = ~(x ^ y)
+    z &= z >> 16
+    z &= z >> 8
+    z &= z >> 4
+    z &= z >> 2
+    z &= z >> 1
+    return z & 1
+  }
+
+  // constant-time string comparison
+  HLP.compare = function (str1, str2) {
+    if (str1.length !== str2.length)
+      return false
+    var i = 0, result = 0
+    for (; i < str1.length; i++)
+      result |= str1[i].charCodeAt(0) ^ str2[i].charCodeAt(0)
+    return intCompare(result, 0)
+  }
+
+  HLP.randomExponent = function () {
+    return BigInt.randBigInt(1536)
+  }
+
+  HLP.smpHash = function (version, fmpi, smpi) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packBytes(version, DTS.BYTE)))
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(fmpi)))
+    if (smpi) sha256.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(smpi)))
+    var hash = sha256.finalize()
+    return HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+  }
+
+  HLP.makeMac = function (aesctr, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var mac = CryptoJS.HmacSHA256(CryptoJS.enc.Latin1.parse(aesctr), pass)
+    return HLP.mask(mac.toString(CryptoJS.enc.Latin1), 0, 160)
+  }
+
+  HLP.make1Mac = function (aesctr, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var mac = CryptoJS.HmacSHA1(CryptoJS.enc.Latin1.parse(aesctr), pass)
+    return mac.toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.encryptAes = function (msg, c, iv) {
+    var opts = {
+        mode: CryptoJS.mode.CTR
+      , iv: CryptoJS.enc.Latin1.parse(iv)
+      , padding: CryptoJS.pad.NoPadding
+    }
+    var aesctr = CryptoJS.AES.encrypt(
+        msg
+      , CryptoJS.enc.Latin1.parse(c)
+      , opts
+    )
+    var aesctr_decoded = CryptoJS.enc.Base64.parse(aesctr.toString())
+    return CryptoJS.enc.Latin1.stringify(aesctr_decoded)
+  }
+
+  HLP.decryptAes = function (msg, c, iv) {
+    msg = CryptoJS.enc.Latin1.parse(msg)
+    var opts = {
+        mode: CryptoJS.mode.CTR
+      , iv: CryptoJS.enc.Latin1.parse(iv)
+      , padding: CryptoJS.pad.NoPadding
+    }
+    return CryptoJS.AES.decrypt(
+        CryptoJS.enc.Base64.stringify(msg)
+      , CryptoJS.enc.Latin1.parse(c)
+      , opts
+    )
+  }
+
+  HLP.multPowMod = function (a, b, c, d, e) {
+    return BigInt.multMod(BigInt.powMod(a, b, e), BigInt.powMod(c, d, e), e)
+  }
+
+  HLP.ZKP = function (v, c, d, e) {
+    return BigInt.equals(c, HLP.smpHash(v, d, e))
+  }
+
+  // greater than, or equal
+  HLP.GTOE = function (a, b) {
+    return (BigInt.equals(a, b) || BigInt.greater(a, b))
+  }
+
+  HLP.between = function (x, a, b) {
+    return (BigInt.greater(x, a) && BigInt.greater(b, x))
+  }
+
+  HLP.checkGroup = function (g, N_MINUS_2) {
+    return HLP.GTOE(g, TWO) && HLP.GTOE(N_MINUS_2, g)
+  }
+
+  HLP.h1 = function (b, secbytes) {
+    var sha1 = CryptoJS.algo.SHA1.create()
+    sha1.update(CryptoJS.enc.Latin1.parse(b))
+    sha1.update(CryptoJS.enc.Latin1.parse(secbytes))
+    return (sha1.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.h2 = function (b, secbytes) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(b))
+    sha256.update(CryptoJS.enc.Latin1.parse(secbytes))
+    return (sha256.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.mask = function (bytes, start, n) {
+    return bytes.substr(start / 8, n / 8)
+  }
+
+  var _toString = String.fromCharCode;
+  HLP.packBytes = function (val, bytes) {
+    val = val.toString(16)
+    var nex, res = ''  // big-endian, unsigned long
+    for (; bytes > 0; bytes--) {
+      nex = val.length ? val.substr(-2, 2) : '0'
+      val = val.substr(0, val.length - 2)
+      res = _toString(parseInt(nex, 16)) + res
+    }
+    return res
+  }
+
+  HLP.packINT = function (d) {
+    return HLP.packBytes(d, DTS.INT)
+  }
+
+  HLP.packCtr = function (d) {
+    return HLP.padCtr(HLP.packBytes(d, DTS.CTR))
+  }
+
+  HLP.padCtr = function (ctr) {
+    return ctr + '\x00\x00\x00\x00\x00\x00\x00\x00'
+  }
+
+  HLP.unpackCtr = function (d) {
+    d = HLP.toByteArray(d.substring(0, 8))
+    return HLP.unpack(d)
+  }
+
+  HLP.unpack = function (arr) {
+    var val = 0, i = 0, len = arr.length
+    for (; i < len; i++) {
+      val = (val * 256) + arr[i]
+    }
+    return val
+  }
+
+  HLP.packData = function (d) {
+    return HLP.packINT(d.length) + d
+  }
+
+  HLP.bits2bigInt = function (bits) {
+    bits = HLP.toByteArray(bits)
+    return BigInt.ba2bigInt(bits)
+  }
+
+  HLP.packMPI = function (mpi) {
+    return HLP.packData(BigInt.bigInt2bits(BigInt.trim(mpi, 0)))
+  }
+
+  HLP.packSHORT = function (short) {
+    return HLP.packBytes(short, DTS.SHORT)
+  }
+
+  HLP.unpackSHORT = function (short) {
+    short = HLP.toByteArray(short)
+    return HLP.unpack(short)
+  }
+
+  HLP.packTLV = function (type, value) {
+    return HLP.packSHORT(type) + HLP.packSHORT(value.length) + value
+  }
+
+  HLP.readLen = function (msg) {
+    msg = HLP.toByteArray(msg.substring(0, 4))
+    return HLP.unpack(msg)
+  }
+
+  HLP.readData = function (data) {
+    var n = HLP.unpack(data.splice(0, 4))
+    return [n, data]
+  }
+
+  HLP.readMPI = function (data) {
+    data = HLP.toByteArray(data)
+    data = HLP.readData(data)
+    return BigInt.ba2bigInt(data[1])
+  }
+
+  HLP.packMPIs = function (arr) {
+    return arr.reduce(function (prv, cur) {
+      return prv + HLP.packMPI(cur)
+    }, '')
+  }
+
+  HLP.unpackMPIs = function (num, mpis) {
+    var i = 0, arr = []
+    for (; i < num; i++) arr.push('MPI')
+    return (HLP.splitype(arr, mpis)).map(function (m) {
+      return HLP.readMPI(m)
+    })
+  }
+
+  HLP.wrapMsg = function (msg, fs, v3, our_it, their_it) {
+    msg = CryptoJS.enc.Base64.stringify(CryptoJS.enc.Latin1.parse(msg))
+    msg = WRAPPER_BEGIN + ":" + msg + WRAPPER_END
+
+    var its
+    if (v3) {
+      its = '|'
+      its += (HLP.readLen(our_it)).toString(16)
+      its += '|'
+      its += (HLP.readLen(their_it)).toString(16)
+    }
+
+    if (!fs) return [null, msg]
+
+    var n = Math.ceil(msg.length / fs)
+    if (n > 65535) return ['Too many fragments']
+    if (n == 1) return [null, msg]
+
+    var k, bi, ei, frag, mf, mfs = []
+    for (k = 1; k <= n; k++) {
+      bi = (k - 1) * fs
+      ei = k * fs
+      frag = msg.slice(bi, ei)
+      mf = WRAPPER_BEGIN
+      if (v3) mf += its
+      mf += ',' + k + ','
+      mf += n + ','
+      mf += frag + ','
+      mfs.push(mf)
+    }
+
+    return [null, mfs]
+  }
+
+  HLP.splitype = function splitype(arr, msg) {
+    var data = []
+    arr.forEach(function (a) {
+      var str
+      switch (a) {
+        case 'PUBKEY':
+          str = splitype(['SHORT', 'MPI', 'MPI', 'MPI', 'MPI'], msg).join('')
+          break
+        case 'DATA':  // falls through
+        case 'MPI':
+          str = msg.substring(0, HLP.readLen(msg) + 4)
+          break
+        default:
+          str = msg.substring(0, DTS[a])
+      }
+      data.push(str)
+      msg = msg.substring(str.length)
+    })
+    return data
+  }
+
+  // https://github.com/msgpack/msgpack-javascript/blob/master/msgpack.js
+
+  var _bin2num = (function () {
+    var i = 0, _bin2num = {}
+    for (; i < 0x100; ++i) {
+      _bin2num[String.fromCharCode(i)] = i  // "\00" -> 0x00
+    }
+    for (i = 0x80; i < 0x100; ++i) {  // [Webkit][Gecko]
+      _bin2num[String.fromCharCode(0xf700 + i)] = i  // "\f780" -> 0x80
+    }
+    return _bin2num
+  }())
+
+  HLP.toByteArray = function (data) {
+    var rv = []
+      , ary = data.split("")
+      , i = -1
+      , iz = ary.length
+      , remain = iz % 8
+
+    while (remain--) {
+      ++i
+      rv[i] = _bin2num[ary[i]]
+    }
+    remain = iz >> 3
+    while (remain--) {
+      rv.push(_bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]])
+    }
+    return rv
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, Worker, WWPath, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = DSA
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    WWPath = require('path').join(__dirname, '/dsa-webworker.js')
+    HLP = require('./helpers.js')
+  } else {
+    // copy over and expose internals
+    Object.keys(root.DSA).forEach(function (k) {
+      DSA[k] = root.DSA[k]
+    })
+    root.DSA = DSA
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    Worker = root.Worker
+    WWPath = 'dsa-webworker.js'
+    HLP = DSA.HLP
+  }
+
+  var ZERO = BigInt.str2bigInt('0', 10)
+    , ONE = BigInt.str2bigInt('1', 10)
+    , TWO = BigInt.str2bigInt('2', 10)
+    , KEY_TYPE = '\x00\x00'
+
+  var DEBUG = false
+  function timer() {
+    var start = (new Date()).getTime()
+    return function (s) {
+      if (!DEBUG || typeof console === 'undefined') return
+      var t = (new Date()).getTime()
+      console.log(s + ': ' + (t - start))
+      start = t
+    }
+  }
+
+  function makeRandom(min, max) {
+    var c = BigInt.randBigInt(BigInt.bitSize(max))
+    if (!HLP.between(c, min, max)) return makeRandom(min, max)
+    return c
+  }
+
+  // altered BigInt.randProbPrime()
+  // n rounds of Miller Rabin (after trial division with small primes)
+  var rpprb = []
+  function isProbPrime(k, n) {
+    var i, B = 30000, l = BigInt.bitSize(k)
+    var primes = BigInt.primes
+
+    if (primes.length === 0)
+      primes = BigInt.findPrimes(B)
+
+    if (rpprb.length != k.length)
+      rpprb = BigInt.dup(k)
+
+    // check ans for divisibility by small primes up to B
+    for (i = 0; (i < primes.length) && (primes[i] <= B); i++)
+      if (BigInt.modInt(k, primes[i]) === 0 && !BigInt.equalsInt(k, primes[i]))
+        return 0
+
+    // do n rounds of Miller Rabin, with random bases less than k
+    for (i = 0; i < n; i++) {
+      BigInt.randBigInt_(rpprb, l, 0)
+      while(!BigInt.greater(k, rpprb))  // pick a random rpprb that's < k
+        BigInt.randBigInt_(rpprb, l, 0)
+      if (!BigInt.millerRabin(k, rpprb))
+        return 0
+    }
+
+    return 1
+  }
+
+  var bit_lengths = {
+      '1024': { N: 160, repeat: 40 }  // 40x should give 2^-80 confidence
+    , '2048': { N: 224, repeat: 56 }
+  }
+
+  var primes = {}
+
+  // follows go lang http://golang.org/src/pkg/crypto/dsa/dsa.go
+  // fips version was removed in 0c99af0df3e7
+  function generatePrimes(bit_length) {
+
+    var t = timer()  // for debugging
+
+    // number of MR tests to perform
+    var repeat = bit_lengths[bit_length].repeat
+
+    var N = bit_lengths[bit_length].N
+
+    var LM1 = BigInt.twoToThe(bit_length - 1)
+    var bl4 = 4 * bit_length
+    var brk = false
+
+    var q, p, rem, counter
+    for (;;) {
+
+      q = BigInt.randBigInt(N, 1)
+      q[0] |= 1
+
+      if (!isProbPrime(q, repeat)) continue
+      t('q')
+
+      for (counter = 0; counter < bl4; counter++) {
+        p = BigInt.randBigInt(bit_length, 1)
+        p[0] |= 1
+
+        rem = BigInt.mod(p, q)
+        rem = BigInt.sub(rem, ONE)
+        p = BigInt.sub(p, rem)
+
+        if (BigInt.greater(LM1, p)) continue
+        if (!isProbPrime(p, repeat)) continue
+
+        t('p')
+        primes[bit_length] = { p: p, q: q }
+        brk = true
+        break
+      }
+
+      if (brk) break
+    }
+
+    var h = BigInt.dup(TWO)
+    var pm1 = BigInt.sub(p, ONE)
+    var e = BigInt.multMod(pm1, BigInt.inverseMod(q, p), p)
+
+    var g
+    for (;;) {
+      g = BigInt.powMod(h, e, p)
+      if (BigInt.equals(g, ONE)) {
+        h = BigInt.add(h, ONE)
+        continue
+      }
+      primes[bit_length].g = g
+      t('g')
+      return
+    }
+
+    throw new Error('Unreachable!')
+  }
+
+  function DSA(obj, opts) {
+    if (!(this instanceof DSA)) return new DSA(obj, opts)
+
+    // options
+    opts = opts || {}
+
+    // inherit
+    if (obj) {
+      var self = this
+      ;['p', 'q', 'g', 'y', 'x'].forEach(function (prop) {
+        self[prop] = obj[prop]
+      })
+      this.type = obj.type || KEY_TYPE
+      return
+    }
+
+    // default to 1024
+    var bit_length = parseInt(opts.bit_length ? opts.bit_length : 1024, 10)
+
+    if (!bit_lengths[bit_length])
+      throw new Error('Unsupported bit length.')
+
+    // set primes
+    if (!primes[bit_length])
+      generatePrimes(bit_length)
+
+    this.p = primes[bit_length].p
+    this.q = primes[bit_length].q
+    this.g = primes[bit_length].g
+
+    // key type
+    this.type = KEY_TYPE
+
+    // private key
+    this.x = makeRandom(ZERO, this.q)
+
+    // public keys (p, q, g, y)
+    this.y = BigInt.powMod(this.g, this.x, this.p)
+
+    // nocache?
+    if (opts.nocache) primes[bit_length] = null
+  }
+
+  DSA.prototype = {
+
+    constructor: DSA,
+
+    packPublic: function () {
+      var str = this.type
+      str += HLP.packMPI(this.p)
+      str += HLP.packMPI(this.q)
+      str += HLP.packMPI(this.g)
+      str += HLP.packMPI(this.y)
+      return str
+    },
+
+    packPrivate: function () {
+      var str = this.packPublic() + HLP.packMPI(this.x)
+      str = CryptoJS.enc.Latin1.parse(str)
+      return str.toString(CryptoJS.enc.Base64)
+    },
+
+    // http://www.imperialviolet.org/2013/06/15/suddendeathentropy.html
+    generateNonce: function (m) {
+      var priv = BigInt.bigInt2bits(BigInt.trim(this.x, 0))
+      var rand = BigInt.bigInt2bits(BigInt.randBigInt(256))
+
+      var sha256 = CryptoJS.algo.SHA256.create()
+      sha256.update(CryptoJS.enc.Latin1.parse(priv))
+      sha256.update(m)
+      sha256.update(CryptoJS.enc.Latin1.parse(rand))
+
+      var hash = sha256.finalize()
+      hash = HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+      BigInt.rightShift_(hash, 256 - BigInt.bitSize(this.q))
+
+      return HLP.between(hash, ZERO, this.q) ? hash : this.generateNonce(m)
+    },
+
+    sign: function (m) {
+      m = CryptoJS.enc.Latin1.parse(m)
+      var b = BigInt.str2bigInt(m.toString(CryptoJS.enc.Hex), 16)
+      var k, r = ZERO, s = ZERO
+      while (BigInt.isZero(s) || BigInt.isZero(r)) {
+        k = this.generateNonce(m)
+        r = BigInt.mod(BigInt.powMod(this.g, k, this.p), this.q)
+        if (BigInt.isZero(r)) continue
+        s = BigInt.inverseMod(k, this.q)
+        s = BigInt.mult(s, BigInt.add(b, BigInt.mult(this.x, r)))
+        s = BigInt.mod(s, this.q)
+      }
+      return [r, s]
+    },
+
+    fingerprint: function () {
+      var pk = this.packPublic()
+      if (this.type === KEY_TYPE) pk = pk.substring(2)
+      pk = CryptoJS.enc.Latin1.parse(pk)
+      return CryptoJS.SHA1(pk).toString(CryptoJS.enc.Hex)
+    }
+
+  }
+
+  DSA.parsePublic = function (str, priv) {
+    var fields = ['SHORT', 'MPI', 'MPI', 'MPI', 'MPI']
+    if (priv) fields.push('MPI')
+    str = HLP.splitype(fields, str)
+    var obj = {
+        type: str[0]
+      , p: HLP.readMPI(str[1])
+      , q: HLP.readMPI(str[2])
+      , g: HLP.readMPI(str[3])
+      , y: HLP.readMPI(str[4])
+    }
+    if (priv) obj.x = HLP.readMPI(str[5])
+    return new DSA(obj)
+  }
+
+  function tokenizeStr(str) {
+    var start, end
+
+    start = str.indexOf("(")
+    end = str.lastIndexOf(")")
+
+    if (start < 0 || end < 0)
+      throw new Error("Malformed S-Expression")
+
+    str = str.substring(start + 1, end)
+
+    var splt = str.search(/\s/)
+    var obj = {
+        type: str.substring(0, splt)
+      , val: []
+    }
+
+    str = str.substring(splt + 1, end)
+    start = str.indexOf("(")
+
+    if (start < 0) obj.val.push(str)
+    else {
+
+      var i, len, ss, es
+      while (start > -1) {
+        i = start + 1
+        len = str.length
+        for (ss = 1, es = 0; i < len && es < ss; i++) {
+          if (str[i] === "(") ss++
+          if (str[i] === ")") es++
+        }
+        obj.val.push(tokenizeStr(str.substring(start, ++i)))
+        str = str.substring(++i)
+        start = str.indexOf("(")
+      }
+
+    }
+    return obj
+  }
+
+  function parseLibotr(obj) {
+    if (!obj.type) throw new Error("Parse error.")
+
+    var o, val
+    if (obj.type === "privkeys") {
+      o = []
+      obj.val.forEach(function (i) {
+        o.push(parseLibotr(i))
+      })
+      return o
+    }
+
+    o = {}
+    obj.val.forEach(function (i) {
+
+      val = i.val[0]
+      if (typeof val === "string") {
+
+        if (val.indexOf("#") === 0) {
+          val = val.substring(1, val.lastIndexOf("#"))
+          val = BigInt.str2bigInt(val, 16)
+        }
+
+      } else {
+        val = parseLibotr(i)
+      }
+
+      o[i.type] = val
+    })
+
+    return o
+  }
+
+  DSA.parsePrivate = function (str, libotr) {
+    if (!libotr) {
+      str = CryptoJS.enc.Base64.parse(str)
+      str = str.toString(CryptoJS.enc.Latin1)
+      return DSA.parsePublic(str, true)
+    }
+    // only returning the first key found
+    return parseLibotr(tokenizeStr(str))[0]["private-key"].dsa
+  }
+
+  DSA.verify = function (key, m, r, s) {
+    if (!HLP.between(r, ZERO, key.q) || !HLP.between(s, ZERO, key.q))
+      return false
+
+    var hm = CryptoJS.enc.Latin1.parse(m)  // CryptoJS.SHA1(m)
+    hm = BigInt.str2bigInt(hm.toString(CryptoJS.enc.Hex), 16)
+
+    var w = BigInt.inverseMod(s, key.q)
+    var u1 = BigInt.multMod(hm, w, key.q)
+    var u2 = BigInt.multMod(r, w, key.q)
+
+    u1 = BigInt.powMod(key.g, u1, key.p)
+    u2 = BigInt.powMod(key.y, u2, key.p)
+
+    var v = BigInt.mod(BigInt.multMod(u1, u2, key.p), key.q)
+
+    return BigInt.equals(v, r)
+  }
+
+  DSA.createInWebWorker = function (options, cb) {
+    var opts = {
+        path: WWPath
+      , seed: BigInt.getSeed
+    }
+    if (options && typeof options === 'object')
+      Object.keys(options).forEach(function (k) {
+        opts[k] = options[k]
+      })
+
+    // load optional dep. in node
+    if (typeof module !== 'undefined' && module.exports)
+      Worker = require('webworker-threads').Worker
+
+    var worker = new Worker(opts.path)
+    worker.onmessage = function (e) {
+      var data = e.data
+      switch (data.type) {
+        case "debug":
+          if (!DEBUG || typeof console === 'undefined') return
+          console.log(data.val)
+          break;
+        case "data":
+          worker.terminate()
+          cb(DSA.parsePrivate(data.val))
+          break;
+        default:
+          throw new Error("Unrecognized type.")
+      }
+    }
+    worker.postMessage({
+        seed: opts.seed()
+      , imports: opts.imports
+      , debug: DEBUG
+    })
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var Parse = {}, CryptoJS, CONST, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = Parse
+    CryptoJS = require('../vendor/crypto.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+  } else {
+    root.OTR.Parse = Parse
+    CryptoJS = root.CryptoJS
+    CONST = root.OTR.CONST
+    HLP = root.OTR.HLP
+  }
+
+  // whitespace tags
+  var tags = {}
+  tags[CONST.WHITESPACE_TAG_V2] = CONST.OTR_VERSION_2
+  tags[CONST.WHITESPACE_TAG_V3] = CONST.OTR_VERSION_3
+
+  Parse.parseMsg = function (otr, msg) {
+
+    var ver = []
+
+    // is this otr?
+    var start = msg.indexOf(CONST.OTR_TAG)
+    if (!~start) {
+
+      // restart fragments
+      this.initFragment(otr)
+
+      // whitespace tags
+      ind = msg.indexOf(CONST.WHITESPACE_TAG)
+
+      if (~ind) {
+
+        msg = msg.split('')
+        msg.splice(ind, 16)
+
+        var tag, len = msg.length
+        for (; ind < len;) {
+          tag = msg.slice(ind, ind + 8).join('')
+          if (Object.hasOwnProperty.call(tags, tag)) {
+            msg.splice(ind, 8)
+            ver.push(tags[tag])
+            continue
+          }
+          ind += 8
+        }
+
+        msg = msg.join('')
+
+      }
+
+      return { msg: msg, ver: ver }
+    }
+
+    var ind = start + CONST.OTR_TAG.length
+    var com = msg[ind]
+
+    // message fragment
+    if (com === ',' || com === '|') {
+      return this.msgFragment(otr, msg.substring(ind + 1), (com === '|'))
+    }
+
+    this.initFragment(otr)
+
+    // query message
+    if (~['?', 'v'].indexOf(com)) {
+
+      // version 1
+      if (msg[ind] === '?') {
+        ver.push(CONST.OTR_VERSION_1)
+        ind += 1
+      }
+
+      // other versions
+      var vers = {
+          '2': CONST.OTR_VERSION_2
+        , '3': CONST.OTR_VERSION_3
+      }
+      var qs = msg.substring(ind + 1)
+      var qi = qs.indexOf('?')
+
+      if (qi >= 1) {
+        qs = qs.substring(0, qi).split('')
+        if (msg[ind] === 'v') {
+          qs.forEach(function (q) {
+            if (Object.hasOwnProperty.call(vers, q)) ver.push(vers[q])
+          })
+        }
+      }
+
+      return { cls: 'query', ver: ver }
+    }
+
+    // otr message
+    if (com === ':') {
+
+      ind += 1
+
+      var info = msg.substring(ind, ind + 4)
+      if (info.length < 4) return { msg: msg }
+      info = CryptoJS.enc.Base64.parse(info).toString(CryptoJS.enc.Latin1)
+
+      var version = info.substring(0, 2)
+      var type = info.substring(2)
+
+      // supporting otr versions 2 and 3
+      if (!otr['ALLOW_V' + HLP.unpackSHORT(version)]) return { msg: msg }
+
+      ind += 4
+
+      var end = msg.substring(ind).indexOf('.')
+      if (!~end) return { msg: msg }
+
+      msg = CryptoJS.enc.Base64.parse(msg.substring(ind, ind + end))
+      msg = CryptoJS.enc.Latin1.stringify(msg)
+
+      // instance tags
+      var instance_tags
+      if (version === CONST.OTR_VERSION_3) {
+        instance_tags = msg.substring(0, 8)
+        msg = msg.substring(8)
+      }
+
+      var cls
+      if (~['\x02', '\x0a', '\x11', '\x12'].indexOf(type)) {
+        cls = 'ake'
+      } else if (type === '\x03') {
+        cls = 'data'
+      }
+
+      return {
+          version: version
+        , type: type
+        , msg: msg
+        , cls: cls
+        , instance_tags: instance_tags
+      }
+    }
+
+    // error message
+    if (msg.substring(ind, ind + 7) === ' Error:') {
+      if (otr.ERROR_START_AKE) {
+        otr.sendQueryMsg()
+      }
+      return { msg: msg.substring(ind + 7), cls: 'error' }
+    }
+
+    return { msg: msg }
+  }
+
+  Parse.initFragment = function (otr) {
+    otr.fragment = { s: '', j: 0, k: 0 }
+  }
+
+  Parse.msgFragment = function (otr, msg, v3) {
+
+    msg = msg.split(',')
+
+    // instance tags
+    if (v3) {
+      var its = msg.shift().split('|')
+      var their_it = HLP.packINT(parseInt(its[0], 16))
+      var our_it = HLP.packINT(parseInt(its[1], 16))
+      if (otr.checkInstanceTags(their_it + our_it)) return  // ignore
+    }
+
+    if (msg.length < 4 ||
+      isNaN(parseInt(msg[0], 10)) ||
+      isNaN(parseInt(msg[1], 10))
+    ) return
+
+    var k = parseInt(msg[0], 10)
+    var n = parseInt(msg[1], 10)
+    msg = msg[2]
+
+    if (n < k || n === 0 || k === 0) {
+      this.initFragment(otr)
+      return
+    }
+
+    if (k === 1) {
+      this.initFragment(otr)
+      otr.fragment = { k: 1, n: n, s: msg }
+    } else if (n === otr.fragment.n && k === (otr.fragment.k + 1)) {
+      otr.fragment.s += msg
+      otr.fragment.k += 1
+    } else {
+      this.initFragment(otr)
+    }
+
+    if (n === k) {
+      msg = otr.fragment.s
+      this.initFragment(otr)
+      return this.parseMsg(otr, msg)
+    }
+
+    return
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, CONST, HLP, DSA
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = AKE
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+    DSA = require('./dsa.js')
+  } else {
+    root.OTR.AKE = AKE
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    CONST = root.OTR.CONST
+    HLP = root.OTR.HLP
+    DSA = root.DSA
+  }
+
+  // diffie-hellman modulus
+  // see group 5, RFC 3526
+  var N = BigInt.str2bigInt(CONST.N, 16)
+  var N_MINUS_2 = BigInt.sub(N, BigInt.str2bigInt('2', 10))
+
+  function hMac(gx, gy, pk, kid, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var hmac = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, pass)
+    hmac.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(gx)))
+    hmac.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(gy)))
+    hmac.update(CryptoJS.enc.Latin1.parse(pk))
+    hmac.update(CryptoJS.enc.Latin1.parse(kid))
+    return (hmac.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  // AKE constructor
+  function AKE(otr) {
+    if (!(this instanceof AKE)) return new AKE(otr)
+
+    // otr instance
+    this.otr = otr
+
+    // our keys
+    this.our_dh = otr.our_old_dh
+    this.our_keyid = otr.our_keyid - 1
+
+    // their keys
+    this.their_y = null
+    this.their_keyid = null
+    this.their_priv_pk = null
+
+    // state
+    this.ssid = null
+    this.transmittedRS = false
+    this.r = null
+
+    // bind methods
+    var self = this
+    ;['sendMsg'].forEach(function (meth) {
+      self[meth] = self[meth].bind(self)
+    })
+  }
+
+  AKE.prototype = {
+
+    constructor: AKE,
+
+    createKeys: function(g) {
+      var s = BigInt.powMod(g, this.our_dh.privateKey, N)
+      var secbytes = HLP.packMPI(s)
+      this.ssid = HLP.mask(HLP.h2('\x00', secbytes), 0, 64)  // first 64-bits
+      var tmp = HLP.h2('\x01', secbytes)
+      this.c = HLP.mask(tmp, 0, 128)  // first 128-bits
+      this.c_prime = HLP.mask(tmp, 128, 128)  // second 128-bits
+      this.m1 = HLP.h2('\x02', secbytes)
+      this.m2 = HLP.h2('\x03', secbytes)
+      this.m1_prime = HLP.h2('\x04', secbytes)
+      this.m2_prime = HLP.h2('\x05', secbytes)
+    },
+
+    verifySignMac: function (mac, aesctr, m2, c, their_y, our_dh_pk, m1, ctr) {
+      // verify mac
+      var vmac = HLP.makeMac(aesctr, m2)
+      if (!HLP.compare(mac, vmac))
+        return ['MACs do not match.']
+
+      // decrypt x
+      var x = HLP.decryptAes(aesctr.substring(4), c, ctr)
+      x = HLP.splitype(['PUBKEY', 'INT', 'SIG'], x.toString(CryptoJS.enc.Latin1))
+
+      var m = hMac(their_y, our_dh_pk, x[0], x[1], m1)
+      var pub = DSA.parsePublic(x[0])
+
+      var r = HLP.bits2bigInt(x[2].substring(0, 20))
+      var s = HLP.bits2bigInt(x[2].substring(20))
+
+      // verify sign m
+      if (!DSA.verify(pub, m, r, s)) return ['Cannot verify signature of m.']
+
+      return [null, HLP.readLen(x[1]), pub]
+    },
+
+    makeM: function (their_y, m1, c, m2) {
+      var pk = this.otr.priv.packPublic()
+      var kid = HLP.packINT(this.our_keyid)
+      var m = hMac(this.our_dh.publicKey, their_y, pk, kid, m1)
+      m = this.otr.priv.sign(m)
+      var msg = pk + kid
+      msg += BigInt.bigInt2bits(m[0], 20)  // pad to 20 bytes
+      msg += BigInt.bigInt2bits(m[1], 20)
+      msg = CryptoJS.enc.Latin1.parse(msg)
+      var aesctr = HLP.packData(HLP.encryptAes(msg, c, HLP.packCtr(0)))
+      var mac = HLP.makeMac(aesctr, m2)
+      return aesctr + mac
+    },
+
+    akeSuccess: function (version) {
+      HLP.debug.call(this.otr, 'success')
+
+      if (BigInt.equals(this.their_y, this.our_dh.publicKey))
+        return this.otr.error('equal keys - we have a problem.')
+
+      this.otr.our_old_dh = this.our_dh
+      this.otr.their_priv_pk = this.their_priv_pk
+
+      if (!(
+        (this.their_keyid === this.otr.their_keyid &&
+         BigInt.equals(this.their_y, this.otr.their_y)) ||
+        (this.their_keyid === (this.otr.their_keyid - 1) &&
+         BigInt.equals(this.their_y, this.otr.their_old_y))
+      )) {
+
+        this.otr.their_y = this.their_y
+        this.otr.their_old_y = null
+        this.otr.their_keyid = this.their_keyid
+
+        // rotate keys
+        this.otr.sessKeys[0] = [ new this.otr.DHSession(
+            this.otr.our_dh
+          , this.otr.their_y
+        ), null ]
+        this.otr.sessKeys[1] = [ new this.otr.DHSession(
+            this.otr.our_old_dh
+          , this.otr.their_y
+        ), null ]
+
+      }
+
+      // ake info
+      this.otr.ssid = this.ssid
+      this.otr.transmittedRS = this.transmittedRS
+      this.otr_version = version
+
+      // go encrypted
+      this.otr.authstate = CONST.AUTHSTATE_NONE
+      this.otr.msgstate = CONST.MSGSTATE_ENCRYPTED
+
+      // null out values
+      this.r = null
+      this.myhashed = null
+      this.dhcommit = null
+      this.encrypted = null
+      this.hashed = null
+
+      this.otr.trigger('status', [CONST.STATUS_AKE_SUCCESS])
+
+      // send stored msgs
+      this.otr.sendStored()
+    },
+
+    handleAKE: function (msg) {
+      var send, vsm, type
+      var version = msg.version
+
+      switch (msg.type) {
+
+        case '\x02':
+          HLP.debug.call(this.otr, 'd-h key message')
+
+          msg = HLP.splitype(['DATA', 'DATA'], msg.msg)
+
+          if (this.otr.authstate === CONST.AUTHSTATE_AWAITING_DHKEY) {
+            var ourHash = HLP.readMPI(this.myhashed)
+            var theirHash = HLP.readMPI(msg[1])
+            if (BigInt.greater(ourHash, theirHash)) {
+              type = '\x02'
+              send = this.dhcommit
+              break  // ignore
+            } else {
+              // forget
+              this.our_dh = this.otr.dh()
+              this.otr.authstate = CONST.AUTHSTATE_NONE
+              this.r = null
+              this.myhashed = null
+            }
+          } else if (
+            this.otr.authstate === CONST.AUTHSTATE_AWAITING_SIG
+          ) this.our_dh = this.otr.dh()
+
+          this.otr.authstate = CONST.AUTHSTATE_AWAITING_REVEALSIG
+
+          this.encrypted = msg[0].substring(4)
+          this.hashed = msg[1].substring(4)
+
+          type = '\x0a'
+          send = HLP.packMPI(this.our_dh.publicKey)
+          break
+
+        case '\x0a':
+          HLP.debug.call(this.otr, 'reveal signature message')
+
+          msg = HLP.splitype(['MPI'], msg.msg)
+
+          if (this.otr.authstate !== CONST.AUTHSTATE_AWAITING_DHKEY) {
+            if (this.otr.authstate === CONST.AUTHSTATE_AWAITING_SIG) {
+              if (!BigInt.equals(this.their_y, HLP.readMPI(msg[0]))) return
+            } else {
+              return  // ignore
+            }
+          }
+
+          this.otr.authstate = CONST.AUTHSTATE_AWAITING_SIG
+
+          this.their_y = HLP.readMPI(msg[0])
+
+          // verify gy is legal 2 <= gy <= N-2
+          if (!HLP.checkGroup(this.their_y, N_MINUS_2))
+            return this.otr.error('Illegal g^y.')
+
+          this.createKeys(this.their_y)
+
+          type = '\x11'
+          send = HLP.packMPI(this.r)
+          send += this.makeM(this.their_y, this.m1, this.c, this.m2)
+
+          this.m1 = null
+          this.m2 = null
+          this.c = null
+          break
+
+        case '\x11':
+          HLP.debug.call(this.otr, 'signature message')
+
+          if (this.otr.authstate !== CONST.AUTHSTATE_AWAITING_REVEALSIG)
+            return  // ignore
+
+          msg = HLP.splitype(['DATA', 'DATA', 'MAC'], msg.msg)
+
+          this.r = HLP.readMPI(msg[0])
+
+          // decrypt their_y
+          var key = CryptoJS.enc.Hex.parse(BigInt.bigInt2str(this.r, 16))
+          key = CryptoJS.enc.Latin1.stringify(key)
+
+          var gxmpi = HLP.decryptAes(this.encrypted, key, HLP.packCtr(0))
+          gxmpi = gxmpi.toString(CryptoJS.enc.Latin1)
+
+          this.their_y = HLP.readMPI(gxmpi)
+
+          // verify hash
+          var hash = CryptoJS.SHA256(CryptoJS.enc.Latin1.parse(gxmpi))
+
+          if (!HLP.compare(this.hashed, hash.toString(CryptoJS.enc.Latin1)))
+            return this.otr.error('Hashed g^x does not match.')
+
+          // verify gx is legal 2 <= g^x <= N-2
+          if (!HLP.checkGroup(this.their_y, N_MINUS_2))
+            return this.otr.error('Illegal g^x.')
+
+          this.createKeys(this.their_y)
+
+          vsm = this.verifySignMac(
+              msg[2]
+            , msg[1]
+            , this.m2
+            , this.c
+            , this.their_y
+            , this.our_dh.publicKey
+            , this.m1
+            , HLP.packCtr(0)
+          )
+          if (vsm[0]) return this.otr.error(vsm[0])
+
+          // store their key
+          this.their_keyid = vsm[1]
+          this.their_priv_pk = vsm[2]
+
+          send = this.makeM(
+              this.their_y
+            , this.m1_prime
+            , this.c_prime
+            , this.m2_prime
+          )
+
+          this.m1 = null
+          this.m2 = null
+          this.m1_prime = null
+          this.m2_prime = null
+          this.c = null
+          this.c_prime = null
+
+          this.sendMsg(version, '\x12', send)
+          this.akeSuccess(version)
+          return
+
+        case '\x12':
+          HLP.debug.call(this.otr, 'data message')
+
+          if (this.otr.authstate !== CONST.AUTHSTATE_AWAITING_SIG)
+            return  // ignore
+
+          msg = HLP.splitype(['DATA', 'MAC'], msg.msg)
+
+          vsm = this.verifySignMac(
+              msg[1]
+            , msg[0]
+            , this.m2_prime
+            , this.c_prime
+            , this.their_y
+            , this.our_dh.publicKey
+            , this.m1_prime
+            , HLP.packCtr(0)
+          )
+          if (vsm[0]) return this.otr.error(vsm[0])
+
+          // store their key
+          this.their_keyid = vsm[1]
+          this.their_priv_pk = vsm[2]
+
+          this.m1_prime = null
+          this.m2_prime = null
+          this.c_prime = null
+
+          this.transmittedRS = true
+          this.akeSuccess(version)
+          return
+
+        default:
+          return  // ignore
+
+      }
+
+      this.sendMsg(version, type, send)
+    },
+
+    sendMsg: function (version, type, msg) {
+      var send = version + type
+      var v3 = (version === CONST.OTR_VERSION_3)
+
+      // instance tags for v3
+      if (v3) {
+        HLP.debug.call(this.otr, 'instance tags')
+        send += this.otr.our_instance_tag
+        send += this.otr.their_instance_tag
+      }
+
+      send += msg
+
+      // fragment message if necessary
+      send = HLP.wrapMsg(
+          send
+        , this.otr.fragment_size
+        , v3
+        , this.otr.our_instance_tag
+        , this.otr.their_instance_tag
+      )
+      if (send[0]) return this.otr.error(send[0])
+
+      this.otr.io(send[1])
+    },
+
+    initiateAKE: function (version) {
+      HLP.debug.call(this.otr, 'd-h commit message')
+
+      this.otr.trigger('status', [CONST.STATUS_AKE_INIT])
+
+      this.otr.authstate = CONST.AUTHSTATE_AWAITING_DHKEY
+
+      var gxmpi = HLP.packMPI(this.our_dh.publicKey)
+      gxmpi = CryptoJS.enc.Latin1.parse(gxmpi)
+
+      this.r = BigInt.randBigInt(128)
+      var key = CryptoJS.enc.Hex.parse(BigInt.bigInt2str(this.r, 16))
+      key = CryptoJS.enc.Latin1.stringify(key)
+
+      this.myhashed = CryptoJS.SHA256(gxmpi)
+      this.myhashed = HLP.packData(this.myhashed.toString(CryptoJS.enc.Latin1))
+
+      this.dhcommit = HLP.packData(HLP.encryptAes(gxmpi, key, HLP.packCtr(0)))
+      this.dhcommit += this.myhashed
+
+      this.sendMsg(version, '\x02', this.dhcommit)
+    }
+
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt,  EventEmitter, CONST, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = SM
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    EventEmitter = require('../vendor/eventemitter.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+  } else {
+    root.OTR.SM = SM
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    EventEmitter = root.EventEmitter
+    CONST = root.OTR.CONST
+    HLP = root.OTR.HLP
+  }
+
+  // diffie-hellman modulus and generator
+  // see group 5, RFC 3526
+  var G = BigInt.str2bigInt(CONST.G, 10)
+  var N = BigInt.str2bigInt(CONST.N, 16)
+  var N_MINUS_2 = BigInt.sub(N, BigInt.str2bigInt('2', 10))
+
+  // to calculate D's for zero-knowledge proofs
+  var Q = BigInt.sub(N, BigInt.str2bigInt('1', 10))
+  BigInt.divInt_(Q, 2)  // meh
+
+  function SM(reqs) {
+    if (!(this instanceof SM)) return new SM(reqs)
+
+    this.version = 1
+
+    this.our_fp = reqs.our_fp
+    this.their_fp = reqs.their_fp
+    this.ssid = reqs.ssid
+
+    this.debug = !!reqs.debug
+
+    // initial state
+    this.init()
+  }
+
+  // inherit from EE
+  HLP.extend(SM, EventEmitter)
+
+  // set the initial values
+  // also used when aborting
+  SM.prototype.init = function () {
+    this.smpstate = CONST.SMPSTATE_EXPECT1
+    this.secret = null
+  }
+
+  SM.prototype.makeSecret = function (our, secret) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packBytes(this.version, 1)))
+    sha256.update(CryptoJS.enc.Hex.parse(our ? this.our_fp : this.their_fp))
+    sha256.update(CryptoJS.enc.Hex.parse(our ? this.their_fp : this.our_fp))
+    sha256.update(CryptoJS.enc.Latin1.parse(this.ssid))
+    sha256.update(CryptoJS.enc.Latin1.parse(secret))
+    var hash = sha256.finalize()
+    this.secret = HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+  }
+
+  SM.prototype.makeG2s = function () {
+    this.a2 = HLP.randomExponent()
+    this.a3 = HLP.randomExponent()
+    this.g2a = BigInt.powMod(G, this.a2, N)
+    this.g3a = BigInt.powMod(G, this.a3, N)
+    if ( !HLP.checkGroup(this.g2a, N_MINUS_2) ||
+         !HLP.checkGroup(this.g3a, N_MINUS_2)
+    ) this.makeG2s()
+  }
+
+  SM.prototype.computeGs = function (g2a, g3a) {
+    this.g2 = BigInt.powMod(g2a, this.a2, N)
+    this.g3 = BigInt.powMod(g3a, this.a3, N)
+  }
+
+  SM.prototype.computePQ = function (r) {
+    this.p = BigInt.powMod(this.g3, r, N)
+    this.q = HLP.multPowMod(G, r, this.g2, this.secret, N)
+  }
+
+  SM.prototype.computeR = function () {
+    this.r = BigInt.powMod(this.QoQ, this.a3, N)
+  }
+
+  SM.prototype.computeRab = function (r) {
+    return BigInt.powMod(r, this.a3, N)
+  }
+
+  SM.prototype.computeC = function (v, r) {
+    return HLP.smpHash(v, BigInt.powMod(G, r, N))
+  }
+
+  SM.prototype.computeD = function (r, a, c) {
+    return BigInt.subMod(r, BigInt.multMod(a, c, Q), Q)
+  }
+
+  // the bulk of the work
+  SM.prototype.handleSM = function (msg) {
+    var send, r2, r3, r7, t1, t2, t3, t4, rab, tmp2, cR, d7, ms, trust
+
+    var expectStates = {
+        2: CONST.SMPSTATE_EXPECT1
+      , 3: CONST.SMPSTATE_EXPECT2
+      , 4: CONST.SMPSTATE_EXPECT3
+      , 5: CONST.SMPSTATE_EXPECT4
+      , 7: CONST.SMPSTATE_EXPECT1
+    }
+
+    if (msg.type === 6) {
+      this.init()
+      this.trigger('abort')
+      return
+    }
+
+    // abort! there was an error
+    if (this.smpstate !== expectStates[msg.type])
+      return this.abort()
+
+    switch (this.smpstate) {
+
+      case CONST.SMPSTATE_EXPECT1:
+        HLP.debug.call(this, 'smp tlv 2')
+
+        // user specified question
+        var ind, question
+        if (msg.type === 7) {
+          ind = msg.msg.indexOf('\x00')
+          question = msg.msg.substring(0, ind)
+          msg.msg = msg.msg.substring(ind + 1)
+        }
+
+        // 0:g2a, 1:c2, 2:d2, 3:g3a, 4:c3, 5:d3
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 6) return this.abort()
+        msg = HLP.unpackMPIs(6, msg.msg.substring(4))
+
+        if ( !HLP.checkGroup(msg[0], N_MINUS_2) ||
+             !HLP.checkGroup(msg[3], N_MINUS_2)
+        ) return this.abort()
+
+        // verify znp's
+        if (!HLP.ZKP(1, msg[1], HLP.multPowMod(G, msg[2], msg[0], msg[1], N)))
+          return this.abort()
+
+        if (!HLP.ZKP(2, msg[4], HLP.multPowMod(G, msg[5], msg[3], msg[4], N)))
+          return this.abort()
+
+        this.g3ao = msg[3]  // save for later
+
+        this.makeG2s()
+
+        // zero-knowledge proof that the exponents
+        // associated with g2a & g3a are known
+        r2 = HLP.randomExponent()
+        r3 = HLP.randomExponent()
+        this.c2 = this.computeC(3, r2)
+        this.c3 = this.computeC(4, r3)
+        this.d2 = this.computeD(r2, this.a2, this.c2)
+        this.d3 = this.computeD(r3, this.a3, this.c3)
+
+        this.computeGs(msg[0], msg[3])
+
+        this.smpstate = CONST.SMPSTATE_EXPECT0
+
+        // assume utf8 question
+        question = CryptoJS.enc.Latin1
+          .parse(question)
+          .toString(CryptoJS.enc.Utf8)
+
+        // invoke question
+        this.trigger('question', [question])
+        return
+
+      case CONST.SMPSTATE_EXPECT2:
+        HLP.debug.call(this, 'smp tlv 3')
+
+        // 0:g2a, 1:c2, 2:d2, 3:g3a, 4:c3, 5:d3, 6:p, 7:q, 8:cP, 9:d5, 10:d6
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 11) return this.abort()
+        msg = HLP.unpackMPIs(11, msg.msg.substring(4))
+
+        if ( !HLP.checkGroup(msg[0], N_MINUS_2) ||
+             !HLP.checkGroup(msg[3], N_MINUS_2) ||
+             !HLP.checkGroup(msg[6], N_MINUS_2) ||
+             !HLP.checkGroup(msg[7], N_MINUS_2)
+        ) return this.abort()
+
+        // verify znp of c3 / c3
+        if (!HLP.ZKP(3, msg[1], HLP.multPowMod(G, msg[2], msg[0], msg[1], N)))
+          return this.abort()
+
+        if (!HLP.ZKP(4, msg[4], HLP.multPowMod(G, msg[5], msg[3], msg[4], N)))
+          return this.abort()
+
+        this.g3ao = msg[3]  // save for later
+
+        this.computeGs(msg[0], msg[3])
+
+        // verify znp of cP
+        t1 = HLP.multPowMod(this.g3, msg[9], msg[6], msg[8], N)
+        t2 = HLP.multPowMod(G, msg[9], this.g2, msg[10], N)
+        t2 = BigInt.multMod(t2, BigInt.powMod(msg[7], msg[8], N), N)
+
+        if (!HLP.ZKP(5, msg[8], t1, t2))
+          return this.abort()
+
+        var r4 = HLP.randomExponent()
+        this.computePQ(r4)
+
+        // zero-knowledge proof that P & Q
+        // were generated according to the protocol
+        var r5 = HLP.randomExponent()
+        var r6 = HLP.randomExponent()
+        var tmp = HLP.multPowMod(G, r5, this.g2, r6, N)
+        var cP = HLP.smpHash(6, BigInt.powMod(this.g3, r5, N), tmp)
+        var d5 = this.computeD(r5, r4, cP)
+        var d6 = this.computeD(r6, this.secret, cP)
+
+        // store these
+        this.QoQ = BigInt.divMod(this.q, msg[7], N)
+        this.PoP = BigInt.divMod(this.p, msg[6], N)
+
+        this.computeR()
+
+        // zero-knowledge proof that R
+        // was generated according to the protocol
+        r7 = HLP.randomExponent()
+        tmp2 = BigInt.powMod(this.QoQ, r7, N)
+        cR = HLP.smpHash(7, BigInt.powMod(G, r7, N), tmp2)
+        d7 = this.computeD(r7, this.a3, cR)
+
+        this.smpstate = CONST.SMPSTATE_EXPECT4
+
+        send = HLP.packINT(8) + HLP.packMPIs([
+            this.p
+          , this.q
+          , cP
+          , d5
+          , d6
+          , this.r
+          , cR
+          , d7
+        ])
+
+        // TLV
+        send = HLP.packTLV(4, send)
+        break
+
+      case CONST.SMPSTATE_EXPECT3:
+        HLP.debug.call(this, 'smp tlv 4')
+
+        // 0:p, 1:q, 2:cP, 3:d5, 4:d6, 5:r, 6:cR, 7:d7
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 8) return this.abort()
+        msg = HLP.unpackMPIs(8, msg.msg.substring(4))
+
+        if ( !HLP.checkGroup(msg[0], N_MINUS_2) ||
+             !HLP.checkGroup(msg[1], N_MINUS_2) ||
+             !HLP.checkGroup(msg[5], N_MINUS_2)
+        ) return this.abort()
+
+        // verify znp of cP
+        t1 = HLP.multPowMod(this.g3, msg[3], msg[0], msg[2], N)
+        t2 = HLP.multPowMod(G, msg[3], this.g2, msg[4], N)
+        t2 = BigInt.multMod(t2, BigInt.powMod(msg[1], msg[2], N), N)
+
+        if (!HLP.ZKP(6, msg[2], t1, t2))
+          return this.abort()
+
+        // verify znp of cR
+        t3 = HLP.multPowMod(G, msg[7], this.g3ao, msg[6], N)
+        this.QoQ = BigInt.divMod(msg[1], this.q, N)  // save Q over Q
+        t4 = HLP.multPowMod(this.QoQ, msg[7], msg[5], msg[6], N)
+
+        if (!HLP.ZKP(7, msg[6], t3, t4))
+          return this.abort()
+
+        this.computeR()
+
+        // zero-knowledge proof that R
+        // was generated according to the protocol
+        r7 = HLP.randomExponent()
+        tmp2 = BigInt.powMod(this.QoQ, r7, N)
+        cR = HLP.smpHash(8, BigInt.powMod(G, r7, N), tmp2)
+        d7 = this.computeD(r7, this.a3, cR)
+
+        send = HLP.packINT(3) + HLP.packMPIs([ this.r, cR, d7 ])
+        send = HLP.packTLV(5, send)
+
+        rab = this.computeRab(msg[5])
+        trust = !!BigInt.equals(rab, BigInt.divMod(msg[0], this.p, N))
+
+        this.trigger('trust', [trust, 'answered'])
+        this.init()
+        break
+
+      case CONST.SMPSTATE_EXPECT4:
+        HLP.debug.call(this, 'smp tlv 5')
+
+        // 0:r, 1:cR, 2:d7
+        ms = HLP.readLen(msg.msg.substr(0, 4))
+        if (ms !== 3) return this.abort()
+        msg = HLP.unpackMPIs(3, msg.msg.substring(4))
+
+        if (!HLP.checkGroup(msg[0], N_MINUS_2)) return this.abort()
+
+        // verify znp of cR
+        t3 = HLP.multPowMod(G, msg[2], this.g3ao, msg[1], N)
+        t4 = HLP.multPowMod(this.QoQ, msg[2], msg[0], msg[1], N)
+        if (!HLP.ZKP(8, msg[1], t3, t4))
+          return this.abort()
+
+        rab = this.computeRab(msg[0])
+        trust = !!BigInt.equals(rab, this.PoP)
+
+        this.trigger('trust', [trust, 'asked'])
+        this.init()
+        return
+
+    }
+
+    this.sendMsg(send)
+  }
+
+  // send a message
+  SM.prototype.sendMsg = function (send) {
+    this.trigger('send', [this.ssid, '\x00' + send])
+  }
+
+  SM.prototype.rcvSecret = function (secret, question) {
+    HLP.debug.call(this, 'receive secret')
+
+    var fn, our = false
+    if (this.smpstate === CONST.SMPSTATE_EXPECT0) {
+      fn = this.answer
+    } else {
+      fn = this.initiate
+      our = true
+    }
+
+    this.makeSecret(our, secret)
+    fn.call(this, question)
+  }
+
+  SM.prototype.answer = function () {
+    HLP.debug.call(this, 'smp answer')
+
+    var r4 = HLP.randomExponent()
+    this.computePQ(r4)
+
+    // zero-knowledge proof that P & Q
+    // were generated according to the protocol
+    var r5 = HLP.randomExponent()
+    var r6 = HLP.randomExponent()
+    var tmp = HLP.multPowMod(G, r5, this.g2, r6, N)
+    var cP = HLP.smpHash(5, BigInt.powMod(this.g3, r5, N), tmp)
+    var d5 = this.computeD(r5, r4, cP)
+    var d6 = this.computeD(r6, this.secret, cP)
+
+    this.smpstate = CONST.SMPSTATE_EXPECT3
+
+    var send = HLP.packINT(11) + HLP.packMPIs([
+        this.g2a
+      , this.c2
+      , this.d2
+      , this.g3a
+      , this.c3
+      , this.d3
+      , this.p
+      , this.q
+      , cP
+      , d5
+      , d6
+    ])
+
+    this.sendMsg(HLP.packTLV(3, send))
+  }
+
+  SM.prototype.initiate = function (question) {
+    HLP.debug.call(this, 'smp initiate')
+
+    if (this.smpstate !== CONST.SMPSTATE_EXPECT1)
+      this.abort()  // abort + restart
+
+    this.makeG2s()
+
+    // zero-knowledge proof that the exponents
+    // associated with g2a & g3a are known
+    var r2 = HLP.randomExponent()
+    var r3 = HLP.randomExponent()
+    this.c2 = this.computeC(1, r2)
+    this.c3 = this.computeC(2, r3)
+    this.d2 = this.computeD(r2, this.a2, this.c2)
+    this.d3 = this.computeD(r3, this.a3, this.c3)
+
+    // set the next expected state
+    this.smpstate = CONST.SMPSTATE_EXPECT2
+
+    var send = ''
+    var type = 2
+
+    if (question) {
+      send += question
+      send += '\x00'
+      type = 7
+    }
+
+    send += HLP.packINT(6) + HLP.packMPIs([
+        this.g2a
+      , this.c2
+      , this.d2
+      , this.g3a
+      , this.c3
+      , this.d3
+    ])
+
+    this.sendMsg(HLP.packTLV(type, send))
+  }
+
+  SM.prototype.abort = function () {
+    this.init()
+    this.sendMsg(HLP.packTLV(6, ''))
+    this.trigger('abort')
+  }
+
+}).call(this)
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, EventEmitter, Worker, SMWPath
+    , CONST, HLP, Parse, AKE, SM, DSA
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = OTR
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    EventEmitter = require('../vendor/eventemitter.js')
+    SMWPath = require('path').join(__dirname, '/sm-webworker.js')
+    CONST = require('./const.js')
+    HLP = require('./helpers.js')
+    Parse = require('./parse.js')
+    AKE = require('./ake.js')
+    SM = require('./sm.js')
+    DSA = require('./dsa.js')
+    // expose CONST for consistency with docs
+    OTR.CONST = CONST
+  } else {
+    // copy over and expose internals
+    Object.keys(root.OTR).forEach(function (k) {
+      OTR[k] = root.OTR[k]
+    })
+    root.OTR = OTR
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    EventEmitter = root.EventEmitter
+    Worker = root.Worker
+    SMWPath = 'sm-webworker.js'
+    CONST = OTR.CONST
+    HLP = OTR.HLP
+    Parse = OTR.Parse
+    AKE = OTR.AKE
+    SM = OTR.SM
+    DSA = root.DSA
+  }
+
+  // diffie-hellman modulus and generator
+  // see group 5, RFC 3526
+  var G = BigInt.str2bigInt(CONST.G, 10)
+  var N = BigInt.str2bigInt(CONST.N, 16)
+
+  // JavaScript integers
+  var MAX_INT = Math.pow(2, 53) - 1  // doubles
+  var MAX_UINT = Math.pow(2, 31) - 1  // bitwise operators
+
+  // an internal callback
+  function OTRCB(cb) {
+    this.cb = cb
+  }
+
+  // OTR contructor
+  function OTR(options) {
+    if (!(this instanceof OTR)) return new OTR(options)
+
+    // options
+    options = options || {}
+
+    // private keys
+    if (options.priv && !(options.priv instanceof DSA))
+      throw new Error('Requires long-lived DSA key.')
+
+    this.priv = options.priv ? options.priv : new DSA()
+
+    this.fragment_size = options.fragment_size || 0
+    if (this.fragment_size < 0)
+      throw new Error('Fragment size must be a positive integer.')
+
+    this.send_interval = options.send_interval || 0
+    if (this.send_interval < 0)
+      throw new Error('Send interval must be a positive integer.')
+
+    this.outgoing = []
+
+    // instance tag
+    this.our_instance_tag = options.instance_tag || OTR.makeInstanceTag()
+
+    // debug
+    this.debug = !!options.debug
+
+    // smp in webworker options
+    // this is still experimental and undocumented
+    this.smw = options.smw
+
+    // init vals
+    this.init()
+
+    // bind methods
+    var self = this
+    ;['sendMsg', 'receiveMsg'].forEach(function (meth) {
+      self[meth] = self[meth].bind(self)
+    })
+
+    EventEmitter.call(this)
+  }
+
+  // inherit from EE
+  HLP.extend(OTR, EventEmitter)
+
+  // add to prototype
+  OTR.prototype.init = function () {
+
+    this.msgstate = CONST.MSGSTATE_PLAINTEXT
+    this.authstate = CONST.AUTHSTATE_NONE
+
+    this.ALLOW_V2 = true
+    this.ALLOW_V3 = true
+
+    this.REQUIRE_ENCRYPTION = false
+    this.SEND_WHITESPACE_TAG = false
+    this.WHITESPACE_START_AKE = false
+    this.ERROR_START_AKE = false
+
+    Parse.initFragment(this)
+
+    // their keys
+    this.their_y = null
+    this.their_old_y = null
+    this.their_keyid = 0
+    this.their_priv_pk = null
+    this.their_instance_tag = '\x00\x00\x00\x00'
+
+    // our keys
+    this.our_dh = this.dh()
+    this.our_old_dh = this.dh()
+    this.our_keyid = 2
+
+    // session keys
+    this.sessKeys = [ new Array(2), new Array(2) ]
+
+    // saved
+    this.storedMgs = []
+    this.oldMacKeys = []
+
+    // smp
+    this.sm = null  // initialized after AKE
+
+    // when ake is complete
+    // save their keys and the session
+    this._akeInit()
+
+    // receive plaintext message since switching to plaintext
+    // used to decide when to stop sending pt tags when SEND_WHITESPACE_TAG
+    this.receivedPlaintext = false
+
+  }
+
+  OTR.prototype._akeInit = function () {
+    this.ake = new AKE(this)
+    this.transmittedRS = false
+    this.ssid = null
+  }
+
+  // smp over webworker
+  OTR.prototype._SMW = function (otr, reqs) {
+    this.otr = otr
+    var opts = {
+        path: SMWPath
+      , seed: BigInt.getSeed
+    }
+    if (typeof otr.smw === 'object')
+      Object.keys(otr.smw).forEach(function (k) {
+        opts[k] = otr.smw[k]
+      })
+
+    // load optional dep. in node
+    if (typeof module !== 'undefined' && module.exports)
+      Worker = require('webworker-threads').Worker
+
+    this.worker = new Worker(opts.path)
+    var self = this
+    this.worker.onmessage = function (e) {
+      var d = e.data
+      if (!d) return
+      self.trigger(d.method, d.args)
+    }
+    this.worker.postMessage({
+        type: 'seed'
+      , seed: opts.seed()
+      , imports: opts.imports
+    })
+    this.worker.postMessage({
+        type: 'init'
+      , reqs: reqs
+    })
+  }
+
+  // inherit from EE
+  HLP.extend(OTR.prototype._SMW, EventEmitter)
+
+  // shim sm methods
+  ;['handleSM', 'rcvSecret', 'abort'].forEach(function (m) {
+    OTR.prototype._SMW.prototype[m] = function () {
+      this.worker.postMessage({
+          type: 'method'
+        , method: m
+        , args: Array.prototype.slice.call(arguments, 0)
+      })
+    }
+  })
+
+  OTR.prototype._smInit = function () {
+    var reqs = {
+        ssid: this.ssid
+      , our_fp: this.priv.fingerprint()
+      , their_fp: this.their_priv_pk.fingerprint()
+      , debug: this.debug
+    }
+    if (this.smw) {
+      if (this.sm) this.sm.worker.terminate()  // destroy prev webworker
+      this.sm = new this._SMW(this, reqs)
+    } else {
+      this.sm = new SM(reqs)
+    }
+    var self = this
+    ;['trust', 'abort', 'question'].forEach(function (e) {
+      self.sm.on(e, function () {
+        self.trigger('smp', [e].concat(Array.prototype.slice.call(arguments)))
+      })
+    })
+    this.sm.on('send', function (ssid, send) {
+      if (self.ssid === ssid) {
+        send = self.prepareMsg(send)
+        self.io(send)
+      }
+    })
+  }
+
+  OTR.prototype.io = function (msg, meta) {
+
+    // buffer
+    msg = ([].concat(msg)).map(function(m){
+       return { msg: m, meta: meta }
+    })
+    this.outgoing = this.outgoing.concat(msg)
+
+    var self = this
+    ;(function send(first) {
+      if (!first) {
+        if (!self.outgoing.length) return
+        var elem = self.outgoing.shift(), cb = null
+        if (elem.meta instanceof OTRCB) {
+          cb = elem.meta.cb
+          elem.meta = null
+        }
+        self.trigger('io', [elem.msg, elem.meta])
+        if (cb) cb()
+      }
+      setTimeout(send, first ? 0 : self.send_interval)
+    }(true))
+
+  }
+
+  OTR.prototype.dh = function dh() {
+    var keys = { privateKey: BigInt.randBigInt(320) }
+    keys.publicKey = BigInt.powMod(G, keys.privateKey, N)
+    return keys
+  }
+
+  // session constructor
+  OTR.prototype.DHSession = function DHSession(our_dh, their_y) {
+    if (!(this instanceof DHSession)) return new DHSession(our_dh, their_y)
+
+    // shared secret
+    var s = BigInt.powMod(their_y, our_dh.privateKey, N)
+    var secbytes = HLP.packMPI(s)
+
+    // session id
+    this.id = HLP.mask(HLP.h2('\x00', secbytes), 0, 64)  // first 64-bits
+
+    // are we the high or low end of the connection?
+    var sq = BigInt.greater(our_dh.publicKey, their_y)
+    var sendbyte = sq ? '\x01' : '\x02'
+    var rcvbyte  = sq ? '\x02' : '\x01'
+
+    // sending and receiving keys
+    this.sendenc = HLP.mask(HLP.h1(sendbyte, secbytes), 0, 128)  // f16 bytes
+    this.sendmac = CryptoJS.SHA1(CryptoJS.enc.Latin1.parse(this.sendenc))
+    this.sendmac = this.sendmac.toString(CryptoJS.enc.Latin1)
+
+    this.rcvenc = HLP.mask(HLP.h1(rcvbyte, secbytes), 0, 128)
+    this.rcvmac = CryptoJS.SHA1(CryptoJS.enc.Latin1.parse(this.rcvenc))
+    this.rcvmac = this.rcvmac.toString(CryptoJS.enc.Latin1)
+    this.rcvmacused = false
+
+    // extra symmetric key
+    this.extra_symkey = HLP.h2('\xff', secbytes)
+
+    // counters
+    this.send_counter = 0
+    this.rcv_counter = 0
+  }
+
+  OTR.prototype.rotateOurKeys = function () {
+
+    // reveal old mac keys
+    var self = this
+    this.sessKeys[1].forEach(function (sk) {
+      if (sk && sk.rcvmacused) self.oldMacKeys.push(sk.rcvmac)
+    })
+
+    // rotate our keys
+    this.our_old_dh = this.our_dh
+    this.our_dh = this.dh()
+    this.our_keyid += 1
+
+    this.sessKeys[1][0] = this.sessKeys[0][0]
+    this.sessKeys[1][1] = this.sessKeys[0][1]
+    this.sessKeys[0] = [
+        this.their_y ?
+            new this.DHSession(this.our_dh, this.their_y) : null
+      , this.their_old_y ?
+            new this.DHSession(this.our_dh, this.their_old_y) : null
+    ]
+
+  }
+
+  OTR.prototype.rotateTheirKeys = function (their_y) {
+
+    // increment their keyid
+    this.their_keyid += 1
+
+    // reveal old mac keys
+    var self = this
+    this.sessKeys.forEach(function (sk) {
+      if (sk[1] && sk[1].rcvmacused) self.oldMacKeys.push(sk[1].rcvmac)
+    })
+
+    // rotate their keys / session
+    this.their_old_y = this.their_y
+    this.sessKeys[0][1] = this.sessKeys[0][0]
+    this.sessKeys[1][1] = this.sessKeys[1][0]
+
+    // new keys / sessions
+    this.their_y = their_y
+    this.sessKeys[0][0] = new this.DHSession(this.our_dh, this.their_y)
+    this.sessKeys[1][0] = new this.DHSession(this.our_old_dh, this.their_y)
+
+  }
+
+  OTR.prototype.prepareMsg = function (msg, esk) {
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED || this.their_keyid === 0)
+      return this.notify('Not ready to encrypt.')
+
+    var sessKeys = this.sessKeys[1][0]
+
+    if (sessKeys.send_counter >= MAX_INT)
+      return this.notify('Should have rekeyed by now.')
+
+    sessKeys.send_counter += 1
+
+    var ctr = HLP.packCtr(sessKeys.send_counter)
+
+    var send = this.ake.otr_version + '\x03'  // version and type
+    var v3 = (this.ake.otr_version === CONST.OTR_VERSION_3)
+
+    if (v3) {
+      send += this.our_instance_tag
+      send += this.their_instance_tag
+    }
+
+    send += '\x00'  // flag
+    send += HLP.packINT(this.our_keyid - 1)
+    send += HLP.packINT(this.their_keyid)
+    send += HLP.packMPI(this.our_dh.publicKey)
+    send += ctr.substring(0, 8)
+
+    if (Math.ceil(msg.length / 8) >= MAX_UINT)  // * 16 / 128
+      return this.notify('Message is too long.')
+
+    var aes = HLP.encryptAes(
+        CryptoJS.enc.Latin1.parse(msg)
+      , sessKeys.sendenc
+      , ctr
+    )
+
+    send += HLP.packData(aes)
+    send += HLP.make1Mac(send, sessKeys.sendmac)
+    send += HLP.packData(this.oldMacKeys.splice(0).join(''))
+
+    send = HLP.wrapMsg(
+        send
+      , this.fragment_size
+      , v3
+      , this.our_instance_tag
+      , this.their_instance_tag
+    )
+    if (send[0]) return this.notify(send[0])
+
+    // emit extra symmetric key
+    if (esk) this.trigger('file', ['send', sessKeys.extra_symkey, esk])
+
+    return send[1]
+  }
+
+  OTR.prototype.handleDataMsg = function (msg) {
+    var vt = msg.version + msg.type
+
+    if (this.ake.otr_version === CONST.OTR_VERSION_3)
+      vt += msg.instance_tags
+
+    var types = ['BYTE', 'INT', 'INT', 'MPI', 'CTR', 'DATA', 'MAC', 'DATA']
+    msg = HLP.splitype(types, msg.msg)
+
+    // ignore flag
+    var ign = (msg[0] === '\x01')
+
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED || msg.length !== 8) {
+      if (!ign) this.error('Received an unreadable encrypted message.')
+      return
+    }
+
+    var our_keyid = this.our_keyid - HLP.readLen(msg[2])
+    var their_keyid = this.their_keyid - HLP.readLen(msg[1])
+
+    if (our_keyid < 0 || our_keyid > 1) {
+      if (!ign) this.error('Not of our latest keys.')
+      return
+    }
+
+    if (their_keyid < 0 || their_keyid > 1) {
+      if (!ign) this.error('Not of your latest keys.')
+      return
+    }
+
+    var their_y = their_keyid ? this.their_old_y : this.their_y
+
+    if (their_keyid === 1 && !their_y) {
+      if (!ign) this.error('Do not have that key.')
+      return
+    }
+
+    var sessKeys = this.sessKeys[our_keyid][their_keyid]
+
+    var ctr = HLP.unpackCtr(msg[4])
+    if (ctr <= sessKeys.rcv_counter) {
+      if (!ign) this.error('Counter in message is not larger.')
+      return
+    }
+    sessKeys.rcv_counter = ctr
+
+    // verify mac
+    vt += msg.slice(0, 6).join('')
+    var vmac = HLP.make1Mac(vt, sessKeys.rcvmac)
+
+    if (!HLP.compare(msg[6], vmac)) {
+      if (!ign) this.error('MACs do not match.')
+      return
+    }
+    sessKeys.rcvmacused = true
+
+    var out = HLP.decryptAes(
+        msg[5].substring(4)
+      , sessKeys.rcvenc
+      , HLP.padCtr(msg[4])
+    )
+    out = out.toString(CryptoJS.enc.Latin1)
+
+    if (!our_keyid) this.rotateOurKeys()
+    if (!their_keyid) this.rotateTheirKeys(HLP.readMPI(msg[3]))
+
+    // parse TLVs
+    var ind = out.indexOf('\x00')
+    if (~ind) {
+      this.handleTLVs(out.substring(ind + 1), sessKeys)
+      out = out.substring(0, ind)
+    }
+
+    out = CryptoJS.enc.Latin1.parse(out)
+    return out.toString(CryptoJS.enc.Utf8)
+  }
+
+  OTR.prototype.handleTLVs = function (tlvs, sessKeys) {
+    var type, len, msg
+    for (; tlvs.length; ) {
+      type = HLP.unpackSHORT(tlvs.substr(0, 2))
+      len = HLP.unpackSHORT(tlvs.substr(2, 2))
+
+      msg = tlvs.substr(4, len)
+
+      // TODO: handle pathological cases better
+      if (msg.length < len) break
+
+      switch (type) {
+        case 1:
+          // Disconnected
+          this.msgstate = CONST.MSGSTATE_FINISHED
+          this.trigger('status', [CONST.STATUS_END_OTR])
+          break
+        case 2: case 3: case 4:
+        case 5: case 6: case 7:
+          // SMP
+          if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED) {
+            if (this.sm) this.sm.abort()
+            return
+          }
+          if (!this.sm) this._smInit()
+          this.sm.handleSM({ msg: msg, type: type })
+          break
+        case 8:
+          // utf8 filenames
+          msg = msg.substring(4) // remove 4-byte indication
+          msg = CryptoJS.enc.Latin1.parse(msg)
+          msg = msg.toString(CryptoJS.enc.Utf8)
+
+          // Extra Symkey
+          this.trigger('file', ['receive', sessKeys.extra_symkey, msg])
+          break
+      }
+
+      tlvs = tlvs.substring(4 + len)
+    }
+  }
+
+  OTR.prototype.smpSecret = function (secret, question) {
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED)
+      return this.notify('Must be encrypted for SMP.')
+
+    if (typeof secret !== 'string' || secret.length < 1)
+      return this.notify('Secret is required.')
+
+    if (!this.sm) this._smInit()
+
+    // utf8 inputs
+    secret = CryptoJS.enc.Utf8.parse(secret).toString(CryptoJS.enc.Latin1)
+    question = CryptoJS.enc.Utf8.parse(question).toString(CryptoJS.enc.Latin1)
+
+    this.sm.rcvSecret(secret, question)
+  }
+
+  OTR.prototype.sendQueryMsg = function () {
+    var versions = {}
+      , msg = CONST.OTR_TAG
+
+    if (this.ALLOW_V2) versions['2'] = true
+    if (this.ALLOW_V3) versions['3'] = true
+
+    // but we don't allow v1
+    // if (versions['1']) msg += '?'
+
+    var vs = Object.keys(versions)
+    if (vs.length) {
+      msg += 'v'
+      vs.forEach(function (v) {
+        if (v !== '1') msg += v
+      })
+      msg += '?'
+    }
+
+    this.io(msg)
+    this.trigger('status', [CONST.STATUS_SEND_QUERY])
+  }
+
+  OTR.prototype.sendMsg = function (msg, meta) {
+    if ( this.REQUIRE_ENCRYPTION ||
+         this.msgstate !== CONST.MSGSTATE_PLAINTEXT
+    ) {
+      msg = CryptoJS.enc.Utf8.parse(msg)
+      msg = msg.toString(CryptoJS.enc.Latin1)
+    }
+
+    switch (this.msgstate) {
+      case CONST.MSGSTATE_PLAINTEXT:
+        if (this.REQUIRE_ENCRYPTION) {
+          this.storedMgs.push({msg: msg, meta: meta})
+          this.sendQueryMsg()
+          return
+        }
+        if (this.SEND_WHITESPACE_TAG && !this.receivedPlaintext) {
+          msg += CONST.WHITESPACE_TAG  // 16 byte tag
+          if (this.ALLOW_V3) msg += CONST.WHITESPACE_TAG_V3
+          if (this.ALLOW_V2) msg += CONST.WHITESPACE_TAG_V2
+        }
+        break
+      case CONST.MSGSTATE_FINISHED:
+        this.storedMgs.push({msg: msg, meta: meta})
+        this.notify('Message cannot be sent at this time.', 'warn')
+        return
+      case CONST.MSGSTATE_ENCRYPTED:
+        msg = this.prepareMsg(msg)
+        break
+      default:
+        throw new Error('Unknown message state.')
+    }
+
+    if (msg) this.io(msg, meta)
+  }
+
+  OTR.prototype.receiveMsg = function (msg) {
+
+    // parse type
+    msg = Parse.parseMsg(this, msg)
+
+    if (!msg) return
+
+    switch (msg.cls) {
+      case 'error':
+        this.notify(msg.msg)
+        return
+      case 'ake':
+        if ( msg.version === CONST.OTR_VERSION_3 &&
+          this.checkInstanceTags(msg.instance_tags)
+        ) {
+          this.notify(
+            'Received a message intended for a different session.', 'warn')
+          return  // ignore
+        }
+        this.ake.handleAKE(msg)
+        return
+      case 'data':
+        if ( msg.version === CONST.OTR_VERSION_3 &&
+          this.checkInstanceTags(msg.instance_tags)
+        ) {
+          this.notify(
+            'Received a message intended for a different session.', 'warn')
+          return  // ignore
+        }
+        msg.msg = this.handleDataMsg(msg)
+        msg.encrypted = true
+        break
+      case 'query':
+        if (this.msgstate === CONST.MSGSTATE_ENCRYPTED) this._akeInit()
+        this.doAKE(msg)
+        break
+      default:
+        // check for encrypted
+        if ( this.REQUIRE_ENCRYPTION ||
+             this.msgstate !== CONST.MSGSTATE_PLAINTEXT
+        ) this.notify('Received an unencrypted message.', 'warn')
+
+        // received a plaintext message
+        // stop sending the whitespace tag
+        this.receivedPlaintext = true
+
+        // received a whitespace tag
+        if (this.WHITESPACE_START_AKE && msg.ver.length > 0)
+          this.doAKE(msg)
+    }
+
+    if (msg.msg) this.trigger('ui', [msg.msg, !!msg.encrypted])
+  }
+
+  OTR.prototype.checkInstanceTags = function (it) {
+    var their_it = HLP.readLen(it.substr(0, 4))
+    var our_it = HLP.readLen(it.substr(4, 4))
+
+    if (our_it && our_it !== HLP.readLen(this.our_instance_tag))
+      return true
+
+    if (HLP.readLen(this.their_instance_tag)) {
+      if (HLP.readLen(this.their_instance_tag) !== their_it) return true
+    } else {
+      if (their_it < 100) return true
+      this.their_instance_tag = HLP.packINT(their_it)
+    }
+  }
+
+  OTR.prototype.doAKE = function (msg) {
+    if (this.ALLOW_V3 && ~msg.ver.indexOf(CONST.OTR_VERSION_3)) {
+      this.ake.initiateAKE(CONST.OTR_VERSION_3)
+    } else if (this.ALLOW_V2 && ~msg.ver.indexOf(CONST.OTR_VERSION_2)) {
+      this.ake.initiateAKE(CONST.OTR_VERSION_2)
+    } else {
+      this.notify('OTR conversation requested, ' +
+        'but no compatible protocol version found.', 'warn')
+    }
+  }
+
+  OTR.prototype.error = function (err) {
+    if (!this.debug) err = 'An OTR error has occurred.'
+    this.io('?OTR Error:' + err)
+    this.notify(err)
+  }
+
+  OTR.prototype.notify = function (err, severity) {
+    this.trigger('error', [err, severity || 'error'])
+  }
+
+  OTR.prototype.sendStored = function () {
+    var self = this
+    ;(this.storedMgs.splice(0)).forEach(function (elem) {
+      var msg = self.prepareMsg(elem.msg)
+      self.io(msg, elem.meta)
+    })
+  }
+
+  OTR.prototype.sendFile = function (filename) {
+    if (this.msgstate !== CONST.MSGSTATE_ENCRYPTED)
+      return this.notify('Not ready to encrypt.')
+
+    if (this.ake.otr_version !== CONST.OTR_VERSION_3)
+      return this.notify('Protocol v3 required.')
+
+    if (!filename) return this.notify('Please specify a filename.')
+
+    // utf8 filenames
+    var l1name = CryptoJS.enc.Utf8.parse(filename)
+    l1name = l1name.toString(CryptoJS.enc.Latin1)
+
+    if (l1name.length >= 65532) return this.notify('Filename is too long.')
+
+    var msg = '\x00'  // null byte
+    msg += '\x00\x08'  // type 8 tlv
+    msg += HLP.packSHORT(4 + l1name.length)  // length of value
+    msg += '\x00\x00\x00\x01'  // four bytes indicating file
+    msg += l1name
+
+    msg = this.prepareMsg(msg, filename)
+    this.io(msg)
+  }
+
+  OTR.prototype.endOtr = function (cb) {
+    if (this.msgstate === CONST.MSGSTATE_ENCRYPTED) {
+      if (typeof cb === 'function')
+        cb = new OTRCB(cb)
+      this.sendMsg('\x00\x00\x01\x00\x00', cb)
+      if (this.sm) {
+        if (this.smw) this.sm.worker.terminate()  // destroy webworker
+        this.sm = null
+      }
+    }
+    this.msgstate = CONST.MSGSTATE_PLAINTEXT
+    this.receivedPlaintext = false
+    this.trigger('status', [CONST.STATUS_END_OTR])
+  }
+
+  // attach methods
+
+  OTR.makeInstanceTag = function () {
+    var num = BigInt.randBigInt(32)
+    if (BigInt.greater(BigInt.str2bigInt('100', 16), num))
+      return OTR.makeInstanceTag()
+    return HLP.packINT(parseInt(BigInt.bigInt2str(num, 10), 10))
+  }
+
+}).call(this)
+
+
+  return {
+      OTR: this.OTR
+    , DSA: this.DSA
+  }
+
+}))
\ No newline at end of file
diff --git a/lib/otr/build/otr.min.js b/lib/otr/build/otr.min.js
new file mode 100644
index 0000000..2173bc9
--- /dev/null
+++ b/lib/otr/build/otr.min.js
@@ -0,0 +1,13 @@
+/*!
+
+  otr.js v0.2.13 - 2014-09-07
+  (c) 2014 - Arlo Breault <arlolra at gmail.com>
+  Freely distributed under the MPL v2.0 license.
+
+  This file is concatenated for the browser.
+  Please see: https://github.com/arlolra/otr
+
+*/
+
+!function(root,factory){"function"==typeof define&&define.amd?define(["bigint","crypto","eventemitter"],function(BigInt,CryptoJS,EventEmitter){var root={BigInt:BigInt,CryptoJS:CryptoJS,EventEmitter:EventEmitter,OTR:{},DSA:{}};return factory.call(root)}):(root.OTR={},root.DSA={},factory.call(root))}(this,function(){return function(){"use strict";var root=this,CONST={N:"FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF [...]
+this.sendMsg(HLP.packTLV(3,send))},SM.prototype.initiate=function(question){HLP.debug.call(this,"smp initiate"),this.smpstate!==CONST.SMPSTATE_EXPECT1&&this.abort(),this.makeG2s();var r2=HLP.randomExponent(),r3=HLP.randomExponent();this.c2=this.computeC(1,r2),this.c3=this.computeC(2,r3),this.d2=this.computeD(r2,this.a2,this.c2),this.d3=this.computeD(r3,this.a3,this.c3),this.smpstate=CONST.SMPSTATE_EXPECT2;var send="",type=2;question&&(send+=question,send+="\x00",type=7),send+=HLP.packINT [...]
\ No newline at end of file
diff --git a/lib/otr/build/sm-webworker.js b/lib/otr/build/sm-webworker.js
new file mode 100644
index 0000000..301843d
--- /dev/null
+++ b/lib/otr/build/sm-webworker.js
@@ -0,0 +1,60 @@
+;(function (root) {
+  "use strict";
+
+  root.OTR = {}
+  root.crypto = {
+    randomBytes: function () {
+      throw new Error("Haven't seeded yet.")
+    }
+  }
+
+  // default imports
+  var imports = [
+      'vendor/salsa20.js'
+    , 'vendor/bigint.js'
+    , 'vendor/crypto.js'
+    , 'vendor/eventemitter.js'
+    , 'lib/const.js'
+    , 'lib/helpers.js'
+    , 'lib/sm.js'
+  ]
+
+  function wrapPostMessage(method) {
+    return function () {
+      postMessage({
+          method: method
+        , args: Array.prototype.slice.call(arguments, 0)
+      })
+    }
+  }
+
+  var sm
+  onmessage = function (e) {
+    var data = e.data
+    switch (data.type) {
+      case 'seed':
+        if (data.imports) imports = data.imports
+        importScripts.apply(root, imports)
+
+        // use salsa20 since there's no prng in webworkers
+        var state = new root.Salsa20(
+          data.seed.slice(0, 32),
+          data.seed.slice(32)
+        )
+        root.crypto.randomBytes = function (n) {
+          return state.getBytes(n)
+        }
+        break
+      case 'init':
+        sm = new root.OTR.SM(data.reqs)
+        ;['trust','question', 'send', 'abort'].forEach(function (m) {
+          sm.on(m, wrapPostMessage(m));
+        })
+        break
+      case 'method':
+        sm[data.method].apply(sm, data.args)
+        break
+    }
+  }
+
+}(this))
\ No newline at end of file
diff --git a/lib/otr/lib/const.js b/lib/otr/lib/const.js
new file mode 100644
index 0000000..2fc3d75
--- /dev/null
+++ b/lib/otr/lib/const.js
@@ -0,0 +1,55 @@
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CONST = {
+
+    // diffie-heilman
+      N : 'FFFFFFFFFFFFFFFFC90FDAA22168C234C4C6628B80DC1CD129024E088A67CC74020BBEA63B139B22514A08798E3404DDEF9519B3CD3A431B302B0A6DF25F14374FE1356D6D51C245E485B576625E7EC6F44C42E9A637ED6B0BFF5CB6F406B7EDEE386BFB5A899FA5AE9F24117C4B1FE649286651ECE45B3DC2007CB8A163BF0598DA48361C55D39A69163FA8FD24CF5F83655D23DCA3AD961C62F356208552BB9ED529077096966D670C354E4ABC9804F1746C08CA237327FFFFFFFFFFFFFFFF'
+    , G : '2'
+
+    // otr message states
+    , MSGSTATE_PLAINTEXT : 0
+    , MSGSTATE_ENCRYPTED : 1
+    , MSGSTATE_FINISHED  : 2
+
+    // otr auth states
+    , AUTHSTATE_NONE               : 0
+    , AUTHSTATE_AWAITING_DHKEY     : 1
+    , AUTHSTATE_AWAITING_REVEALSIG : 2
+    , AUTHSTATE_AWAITING_SIG       : 3
+
+    // whitespace tags
+    , WHITESPACE_TAG    : '\x20\x09\x20\x20\x09\x09\x09\x09\x20\x09\x20\x09\x20\x09\x20\x20'
+    , WHITESPACE_TAG_V2 : '\x20\x20\x09\x09\x20\x20\x09\x20'
+    , WHITESPACE_TAG_V3 : '\x20\x20\x09\x09\x20\x20\x09\x09'
+
+    // otr tags
+    , OTR_TAG       : '?OTR'
+    , OTR_VERSION_1 : '\x00\x01'
+    , OTR_VERSION_2 : '\x00\x02'
+    , OTR_VERSION_3 : '\x00\x03'
+
+    // smp machine states
+    , SMPSTATE_EXPECT0 : 0
+    , SMPSTATE_EXPECT1 : 1
+    , SMPSTATE_EXPECT2 : 2
+    , SMPSTATE_EXPECT3 : 3
+    , SMPSTATE_EXPECT4 : 4
+
+    // unstandard status codes
+    , STATUS_SEND_QUERY  : 0
+    , STATUS_AKE_INIT    : 1
+    , STATUS_AKE_SUCCESS : 2
+    , STATUS_END_OTR     : 3
+
+  }
+
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = CONST
+  } else {
+    root.OTR.CONST = CONST
+  }
+
+}).call(this)
\ No newline at end of file
diff --git a/lib/otr/lib/dsa-webworker.js b/lib/otr/lib/dsa-webworker.js
new file mode 100644
index 0000000..d84ad17
--- /dev/null
+++ b/lib/otr/lib/dsa-webworker.js
@@ -0,0 +1,52 @@
+;(function (root) {
+  "use strict";
+
+  root.OTR = {}
+  root.DSA = {}
+  root.crypto = {
+    randomBytes: function () {
+      throw new Error("Haven't seeded yet.")
+    }
+  }
+
+  // default imports
+  var imports = [
+      'vendor/salsa20.js'
+    , 'vendor/bigint.js'
+    , 'vendor/crypto.js'
+    , 'vendor/eventemitter.js'
+    , 'lib/const.js'
+    , 'lib/helpers.js'
+    , 'lib/dsa.js'
+  ]
+
+  function sendMsg(type, val) {
+    postMessage({ type: type, val: val })
+  }
+
+  onmessage = function (e) {
+    var data = e.data;
+
+    if (data.imports) imports = data.imports
+    importScripts.apply(root, imports);
+
+    // use salsa20 since there's no prng in webworkers
+    var state = new root.Salsa20(data.seed.slice(0, 32), data.seed.slice(32))
+    root.crypto.randomBytes = function (n) {
+      return state.getBytes(n)
+    }
+
+    if (data.debug) sendMsg('debug', 'DSA key creation started')
+    var dsa
+    try {
+      dsa = new root.DSA()
+    } catch (e) {
+      if (data.debug) sendMsg('debug', e.toString())
+      return
+    }
+    if (data.debug) sendMsg('debug', 'DSA key creation finished')
+
+    sendMsg('data', dsa.packPrivate())
+  }
+
+}(this))
\ No newline at end of file
diff --git a/lib/otr/lib/dsa.js b/lib/otr/lib/dsa.js
new file mode 100644
index 0000000..10417f8
--- /dev/null
+++ b/lib/otr/lib/dsa.js
@@ -0,0 +1,405 @@
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var CryptoJS, BigInt, Worker, WWPath, HLP
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = DSA
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+    WWPath = require('path').join(__dirname, '/dsa-webworker.js')
+    HLP = require('./helpers.js')
+  } else {
+    // copy over and expose internals
+    Object.keys(root.DSA).forEach(function (k) {
+      DSA[k] = root.DSA[k]
+    })
+    root.DSA = DSA
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+    Worker = root.Worker
+    WWPath = 'dsa-webworker.js'
+    HLP = DSA.HLP
+  }
+
+  var ZERO = BigInt.str2bigInt('0', 10)
+    , ONE = BigInt.str2bigInt('1', 10)
+    , TWO = BigInt.str2bigInt('2', 10)
+    , KEY_TYPE = '\x00\x00'
+
+  var DEBUG = false
+  function timer() {
+    var start = (new Date()).getTime()
+    return function (s) {
+      if (!DEBUG || typeof console === 'undefined') return
+      var t = (new Date()).getTime()
+      console.log(s + ': ' + (t - start))
+      start = t
+    }
+  }
+
+  function makeRandom(min, max) {
+    var c = BigInt.randBigInt(BigInt.bitSize(max))
+    if (!HLP.between(c, min, max)) return makeRandom(min, max)
+    return c
+  }
+
+  // altered BigInt.randProbPrime()
+  // n rounds of Miller Rabin (after trial division with small primes)
+  var rpprb = []
+  function isProbPrime(k, n) {
+    var i, B = 30000, l = BigInt.bitSize(k)
+    var primes = BigInt.primes
+
+    if (primes.length === 0)
+      primes = BigInt.findPrimes(B)
+
+    if (rpprb.length != k.length)
+      rpprb = BigInt.dup(k)
+
+    // check ans for divisibility by small primes up to B
+    for (i = 0; (i < primes.length) && (primes[i] <= B); i++)
+      if (BigInt.modInt(k, primes[i]) === 0 && !BigInt.equalsInt(k, primes[i]))
+        return 0
+
+    // do n rounds of Miller Rabin, with random bases less than k
+    for (i = 0; i < n; i++) {
+      BigInt.randBigInt_(rpprb, l, 0)
+      while(!BigInt.greater(k, rpprb))  // pick a random rpprb that's < k
+        BigInt.randBigInt_(rpprb, l, 0)
+      if (!BigInt.millerRabin(k, rpprb))
+        return 0
+    }
+
+    return 1
+  }
+
+  var bit_lengths = {
+      '1024': { N: 160, repeat: 40 }  // 40x should give 2^-80 confidence
+    , '2048': { N: 224, repeat: 56 }
+  }
+
+  var primes = {}
+
+  // follows go lang http://golang.org/src/pkg/crypto/dsa/dsa.go
+  // fips version was removed in 0c99af0df3e7
+  function generatePrimes(bit_length) {
+
+    var t = timer()  // for debugging
+
+    // number of MR tests to perform
+    var repeat = bit_lengths[bit_length].repeat
+
+    var N = bit_lengths[bit_length].N
+
+    var LM1 = BigInt.twoToThe(bit_length - 1)
+    var bl4 = 4 * bit_length
+    var brk = false
+
+    var q, p, rem, counter
+    for (;;) {
+
+      q = BigInt.randBigInt(N, 1)
+      q[0] |= 1
+
+      if (!isProbPrime(q, repeat)) continue
+      t('q')
+
+      for (counter = 0; counter < bl4; counter++) {
+        p = BigInt.randBigInt(bit_length, 1)
+        p[0] |= 1
+
+        rem = BigInt.mod(p, q)
+        rem = BigInt.sub(rem, ONE)
+        p = BigInt.sub(p, rem)
+
+        if (BigInt.greater(LM1, p)) continue
+        if (!isProbPrime(p, repeat)) continue
+
+        t('p')
+        primes[bit_length] = { p: p, q: q }
+        brk = true
+        break
+      }
+
+      if (brk) break
+    }
+
+    var h = BigInt.dup(TWO)
+    var pm1 = BigInt.sub(p, ONE)
+    var e = BigInt.multMod(pm1, BigInt.inverseMod(q, p), p)
+
+    var g
+    for (;;) {
+      g = BigInt.powMod(h, e, p)
+      if (BigInt.equals(g, ONE)) {
+        h = BigInt.add(h, ONE)
+        continue
+      }
+      primes[bit_length].g = g
+      t('g')
+      return
+    }
+
+    throw new Error('Unreachable!')
+  }
+
+  function DSA(obj, opts) {
+    if (!(this instanceof DSA)) return new DSA(obj, opts)
+
+    // options
+    opts = opts || {}
+
+    // inherit
+    if (obj) {
+      var self = this
+      ;['p', 'q', 'g', 'y', 'x'].forEach(function (prop) {
+        self[prop] = obj[prop]
+      })
+      this.type = obj.type || KEY_TYPE
+      return
+    }
+
+    // default to 1024
+    var bit_length = parseInt(opts.bit_length ? opts.bit_length : 1024, 10)
+
+    if (!bit_lengths[bit_length])
+      throw new Error('Unsupported bit length.')
+
+    // set primes
+    if (!primes[bit_length])
+      generatePrimes(bit_length)
+
+    this.p = primes[bit_length].p
+    this.q = primes[bit_length].q
+    this.g = primes[bit_length].g
+
+    // key type
+    this.type = KEY_TYPE
+
+    // private key
+    this.x = makeRandom(ZERO, this.q)
+
+    // public keys (p, q, g, y)
+    this.y = BigInt.powMod(this.g, this.x, this.p)
+
+    // nocache?
+    if (opts.nocache) primes[bit_length] = null
+  }
+
+  DSA.prototype = {
+
+    constructor: DSA,
+
+    packPublic: function () {
+      var str = this.type
+      str += HLP.packMPI(this.p)
+      str += HLP.packMPI(this.q)
+      str += HLP.packMPI(this.g)
+      str += HLP.packMPI(this.y)
+      return str
+    },
+
+    packPrivate: function () {
+      var str = this.packPublic() + HLP.packMPI(this.x)
+      str = CryptoJS.enc.Latin1.parse(str)
+      return str.toString(CryptoJS.enc.Base64)
+    },
+
+    // http://www.imperialviolet.org/2013/06/15/suddendeathentropy.html
+    generateNonce: function (m) {
+      var priv = BigInt.bigInt2bits(BigInt.trim(this.x, 0))
+      var rand = BigInt.bigInt2bits(BigInt.randBigInt(256))
+
+      var sha256 = CryptoJS.algo.SHA256.create()
+      sha256.update(CryptoJS.enc.Latin1.parse(priv))
+      sha256.update(m)
+      sha256.update(CryptoJS.enc.Latin1.parse(rand))
+
+      var hash = sha256.finalize()
+      hash = HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+      BigInt.rightShift_(hash, 256 - BigInt.bitSize(this.q))
+
+      return HLP.between(hash, ZERO, this.q) ? hash : this.generateNonce(m)
+    },
+
+    sign: function (m) {
+      m = CryptoJS.enc.Latin1.parse(m)
+      var b = BigInt.str2bigInt(m.toString(CryptoJS.enc.Hex), 16)
+      var k, r = ZERO, s = ZERO
+      while (BigInt.isZero(s) || BigInt.isZero(r)) {
+        k = this.generateNonce(m)
+        r = BigInt.mod(BigInt.powMod(this.g, k, this.p), this.q)
+        if (BigInt.isZero(r)) continue
+        s = BigInt.inverseMod(k, this.q)
+        s = BigInt.mult(s, BigInt.add(b, BigInt.mult(this.x, r)))
+        s = BigInt.mod(s, this.q)
+      }
+      return [r, s]
+    },
+
+    fingerprint: function () {
+      var pk = this.packPublic()
+      if (this.type === KEY_TYPE) pk = pk.substring(2)
+      pk = CryptoJS.enc.Latin1.parse(pk)
+      return CryptoJS.SHA1(pk).toString(CryptoJS.enc.Hex)
+    }
+
+  }
+
+  DSA.parsePublic = function (str, priv) {
+    var fields = ['SHORT', 'MPI', 'MPI', 'MPI', 'MPI']
+    if (priv) fields.push('MPI')
+    str = HLP.splitype(fields, str)
+    var obj = {
+        type: str[0]
+      , p: HLP.readMPI(str[1])
+      , q: HLP.readMPI(str[2])
+      , g: HLP.readMPI(str[3])
+      , y: HLP.readMPI(str[4])
+    }
+    if (priv) obj.x = HLP.readMPI(str[5])
+    return new DSA(obj)
+  }
+
+  function tokenizeStr(str) {
+    var start, end
+
+    start = str.indexOf("(")
+    end = str.lastIndexOf(")")
+
+    if (start < 0 || end < 0)
+      throw new Error("Malformed S-Expression")
+
+    str = str.substring(start + 1, end)
+
+    var splt = str.search(/\s/)
+    var obj = {
+        type: str.substring(0, splt)
+      , val: []
+    }
+
+    str = str.substring(splt + 1, end)
+    start = str.indexOf("(")
+
+    if (start < 0) obj.val.push(str)
+    else {
+
+      var i, len, ss, es
+      while (start > -1) {
+        i = start + 1
+        len = str.length
+        for (ss = 1, es = 0; i < len && es < ss; i++) {
+          if (str[i] === "(") ss++
+          if (str[i] === ")") es++
+        }
+        obj.val.push(tokenizeStr(str.substring(start, ++i)))
+        str = str.substring(++i)
+        start = str.indexOf("(")
+      }
+
+    }
+    return obj
+  }
+
+  function parseLibotr(obj) {
+    if (!obj.type) throw new Error("Parse error.")
+
+    var o, val
+    if (obj.type === "privkeys") {
+      o = []
+      obj.val.forEach(function (i) {
+        o.push(parseLibotr(i))
+      })
+      return o
+    }
+
+    o = {}
+    obj.val.forEach(function (i) {
+
+      val = i.val[0]
+      if (typeof val === "string") {
+
+        if (val.indexOf("#") === 0) {
+          val = val.substring(1, val.lastIndexOf("#"))
+          val = BigInt.str2bigInt(val, 16)
+        }
+
+      } else {
+        val = parseLibotr(i)
+      }
+
+      o[i.type] = val
+    })
+
+    return o
+  }
+
+  DSA.parsePrivate = function (str, libotr) {
+    if (!libotr) {
+      str = CryptoJS.enc.Base64.parse(str)
+      str = str.toString(CryptoJS.enc.Latin1)
+      return DSA.parsePublic(str, true)
+    }
+    // only returning the first key found
+    return parseLibotr(tokenizeStr(str))[0]["private-key"].dsa
+  }
+
+  DSA.verify = function (key, m, r, s) {
+    if (!HLP.between(r, ZERO, key.q) || !HLP.between(s, ZERO, key.q))
+      return false
+
+    var hm = CryptoJS.enc.Latin1.parse(m)  // CryptoJS.SHA1(m)
+    hm = BigInt.str2bigInt(hm.toString(CryptoJS.enc.Hex), 16)
+
+    var w = BigInt.inverseMod(s, key.q)
+    var u1 = BigInt.multMod(hm, w, key.q)
+    var u2 = BigInt.multMod(r, w, key.q)
+
+    u1 = BigInt.powMod(key.g, u1, key.p)
+    u2 = BigInt.powMod(key.y, u2, key.p)
+
+    var v = BigInt.mod(BigInt.multMod(u1, u2, key.p), key.q)
+
+    return BigInt.equals(v, r)
+  }
+
+  DSA.createInWebWorker = function (options, cb) {
+    var opts = {
+        path: WWPath
+      , seed: BigInt.getSeed
+    }
+    if (options && typeof options === 'object')
+      Object.keys(options).forEach(function (k) {
+        opts[k] = options[k]
+      })
+
+    // load optional dep. in node
+    if (typeof module !== 'undefined' && module.exports)
+      Worker = require('webworker-threads').Worker
+
+    var worker = new Worker(opts.path)
+    worker.onmessage = function (e) {
+      var data = e.data
+      switch (data.type) {
+        case "debug":
+          if (!DEBUG || typeof console === 'undefined') return
+          console.log(data.val)
+          break;
+        case "data":
+          worker.terminate()
+          cb(DSA.parsePrivate(data.val))
+          break;
+        default:
+          throw new Error("Unrecognized type.")
+      }
+    }
+    worker.postMessage({
+        seed: opts.seed()
+      , imports: opts.imports
+      , debug: DEBUG
+    })
+  }
+
+}).call(this)
\ No newline at end of file
diff --git a/lib/otr/lib/helpers.js b/lib/otr/lib/helpers.js
new file mode 100644
index 0000000..e5a8507
--- /dev/null
+++ b/lib/otr/lib/helpers.js
@@ -0,0 +1,349 @@
+;(function () {
+  "use strict";
+
+  var root = this
+
+  var HLP = {}, CryptoJS, BigInt
+  if (typeof module !== 'undefined' && module.exports) {
+    module.exports = HLP = {}
+    CryptoJS = require('../vendor/crypto.js')
+    BigInt = require('../vendor/bigint.js')
+  } else {
+    if (root.OTR) root.OTR.HLP = HLP
+    if (root.DSA) root.DSA.HLP = HLP
+    CryptoJS = root.CryptoJS
+    BigInt = root.BigInt
+  }
+
+  // data types (byte lengths)
+  var DTS = {
+      BYTE  : 1
+    , SHORT : 2
+    , INT   : 4
+    , CTR   : 8
+    , MAC   : 20
+    , SIG   : 40
+  }
+
+  // otr message wrapper begin and end
+  var WRAPPER_BEGIN = "?OTR"
+    , WRAPPER_END   = "."
+
+  var TWO = BigInt.str2bigInt('2', 10)
+
+  HLP.debug = function (msg) {
+    // used as HLP.debug.call(ctx, msg)
+    if ( this.debug &&
+         typeof this.debug !== 'function' &&
+         typeof console !== 'undefined'
+    ) console.log(msg)
+  }
+
+  HLP.extend = function (child, parent) {
+    for (var key in parent) {
+      if (Object.hasOwnProperty.call(parent, key))
+        child[key] = parent[key]
+    }
+    function Ctor() { this.constructor = child }
+    Ctor.prototype = parent.prototype
+    child.prototype = new Ctor()
+    child.__super__ = parent.prototype
+  }
+
+  // assumes 32-bit
+  function intCompare(x, y) {
+    var z = ~(x ^ y)
+    z &= z >> 16
+    z &= z >> 8
+    z &= z >> 4
+    z &= z >> 2
+    z &= z >> 1
+    return z & 1
+  }
+
+  // constant-time string comparison
+  HLP.compare = function (str1, str2) {
+    if (str1.length !== str2.length)
+      return false
+    var i = 0, result = 0
+    for (; i < str1.length; i++)
+      result |= str1[i].charCodeAt(0) ^ str2[i].charCodeAt(0)
+    return intCompare(result, 0)
+  }
+
+  HLP.randomExponent = function () {
+    return BigInt.randBigInt(1536)
+  }
+
+  HLP.smpHash = function (version, fmpi, smpi) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packBytes(version, DTS.BYTE)))
+    sha256.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(fmpi)))
+    if (smpi) sha256.update(CryptoJS.enc.Latin1.parse(HLP.packMPI(smpi)))
+    var hash = sha256.finalize()
+    return HLP.bits2bigInt(hash.toString(CryptoJS.enc.Latin1))
+  }
+
+  HLP.makeMac = function (aesctr, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var mac = CryptoJS.HmacSHA256(CryptoJS.enc.Latin1.parse(aesctr), pass)
+    return HLP.mask(mac.toString(CryptoJS.enc.Latin1), 0, 160)
+  }
+
+  HLP.make1Mac = function (aesctr, m) {
+    var pass = CryptoJS.enc.Latin1.parse(m)
+    var mac = CryptoJS.HmacSHA1(CryptoJS.enc.Latin1.parse(aesctr), pass)
+    return mac.toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.encryptAes = function (msg, c, iv) {
+    var opts = {
+        mode: CryptoJS.mode.CTR
+      , iv: CryptoJS.enc.Latin1.parse(iv)
+      , padding: CryptoJS.pad.NoPadding
+    }
+    var aesctr = CryptoJS.AES.encrypt(
+        msg
+      , CryptoJS.enc.Latin1.parse(c)
+      , opts
+    )
+    var aesctr_decoded = CryptoJS.enc.Base64.parse(aesctr.toString())
+    return CryptoJS.enc.Latin1.stringify(aesctr_decoded)
+  }
+
+  HLP.decryptAes = function (msg, c, iv) {
+    msg = CryptoJS.enc.Latin1.parse(msg)
+    var opts = {
+        mode: CryptoJS.mode.CTR
+      , iv: CryptoJS.enc.Latin1.parse(iv)
+      , padding: CryptoJS.pad.NoPadding
+    }
+    return CryptoJS.AES.decrypt(
+        CryptoJS.enc.Base64.stringify(msg)
+      , CryptoJS.enc.Latin1.parse(c)
+      , opts
+    )
+  }
+
+  HLP.multPowMod = function (a, b, c, d, e) {
+    return BigInt.multMod(BigInt.powMod(a, b, e), BigInt.powMod(c, d, e), e)
+  }
+
+  HLP.ZKP = function (v, c, d, e) {
+    return BigInt.equals(c, HLP.smpHash(v, d, e))
+  }
+
+  // greater than, or equal
+  HLP.GTOE = function (a, b) {
+    return (BigInt.equals(a, b) || BigInt.greater(a, b))
+  }
+
+  HLP.between = function (x, a, b) {
+    return (BigInt.greater(x, a) && BigInt.greater(b, x))
+  }
+
+  HLP.checkGroup = function (g, N_MINUS_2) {
+    return HLP.GTOE(g, TWO) && HLP.GTOE(N_MINUS_2, g)
+  }
+
+  HLP.h1 = function (b, secbytes) {
+    var sha1 = CryptoJS.algo.SHA1.create()
+    sha1.update(CryptoJS.enc.Latin1.parse(b))
+    sha1.update(CryptoJS.enc.Latin1.parse(secbytes))
+    return (sha1.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.h2 = function (b, secbytes) {
+    var sha256 = CryptoJS.algo.SHA256.create()
+    sha256.update(CryptoJS.enc.Latin1.parse(b))
+    sha256.update(CryptoJS.enc.Latin1.parse(secbytes))
+    return (sha256.finalize()).toString(CryptoJS.enc.Latin1)
+  }
+
+  HLP.mask = function (bytes, start, n) {
+    return bytes.substr(start / 8, n / 8)
+  }
+
+  var _toString = String.fromCharCode;
+  HLP.packBytes = function (val, bytes) {
+    val = val.toString(16)
+    var nex, res = ''  // big-endian, unsigned long
+    for (; bytes > 0; bytes--) {
+      nex = val.length ? val.substr(-2, 2) : '0'
+      val = val.substr(0, val.length - 2)
+      res = _toString(parseInt(nex, 16)) + res
+    }
+    return res
+  }
+
+  HLP.packINT = function (d) {
+    return HLP.packBytes(d, DTS.INT)
+  }
+
+  HLP.packCtr = function (d) {
+    return HLP.padCtr(HLP.packBytes(d, DTS.CTR))
+  }
+
+  HLP.padCtr = function (ctr) {
+    return ctr + '\x00\x00\x00\x00\x00\x00\x00\x00'
+  }
+
+  HLP.unpackCtr = function (d) {
+    d = HLP.toByteArray(d.substring(0, 8))
+    return HLP.unpack(d)
+  }
+
+  HLP.unpack = function (arr) {
+    var val = 0, i = 0, len = arr.length
+    for (; i < len; i++) {
+      val = (val * 256) + arr[i]
+    }
+    return val
+  }
+
+  HLP.packData = function (d) {
+    return HLP.packINT(d.length) + d
+  }
+
+  HLP.bits2bigInt = function (bits) {
+    bits = HLP.toByteArray(bits)
+    return BigInt.ba2bigInt(bits)
+  }
+
+  HLP.packMPI = function (mpi) {
+    return HLP.packData(BigInt.bigInt2bits(BigInt.trim(mpi, 0)))
+  }
+
+  HLP.packSHORT = function (short) {
+    return HLP.packBytes(short, DTS.SHORT)
+  }
+
+  HLP.unpackSHORT = function (short) {
+    short = HLP.toByteArray(short)
+    return HLP.unpack(short)
+  }
+
+  HLP.packTLV = function (type, value) {
+    return HLP.packSHORT(type) + HLP.packSHORT(value.length) + value
+  }
+
+  HLP.readLen = function (msg) {
+    msg = HLP.toByteArray(msg.substring(0, 4))
+    return HLP.unpack(msg)
+  }
+
+  HLP.readData = function (data) {
+    var n = HLP.unpack(data.splice(0, 4))
+    return [n, data]
+  }
+
+  HLP.readMPI = function (data) {
+    data = HLP.toByteArray(data)
+    data = HLP.readData(data)
+    return BigInt.ba2bigInt(data[1])
+  }
+
+  HLP.packMPIs = function (arr) {
+    return arr.reduce(function (prv, cur) {
+      return prv + HLP.packMPI(cur)
+    }, '')
+  }
+
+  HLP.unpackMPIs = function (num, mpis) {
+    var i = 0, arr = []
+    for (; i < num; i++) arr.push('MPI')
+    return (HLP.splitype(arr, mpis)).map(function (m) {
+      return HLP.readMPI(m)
+    })
+  }
+
+  HLP.wrapMsg = function (msg, fs, v3, our_it, their_it) {
+    msg = CryptoJS.enc.Base64.stringify(CryptoJS.enc.Latin1.parse(msg))
+    msg = WRAPPER_BEGIN + ":" + msg + WRAPPER_END
+
+    var its
+    if (v3) {
+      its = '|'
+      its += (HLP.readLen(our_it)).toString(16)
+      its += '|'
+      its += (HLP.readLen(their_it)).toString(16)
+    }
+
+    if (!fs) return [null, msg]
+
+    var n = Math.ceil(msg.length / fs)
+    if (n > 65535) return ['Too many fragments']
+    if (n == 1) return [null, msg]
+
+    var k, bi, ei, frag, mf, mfs = []
+    for (k = 1; k <= n; k++) {
+      bi = (k - 1) * fs
+      ei = k * fs
+      frag = msg.slice(bi, ei)
+      mf = WRAPPER_BEGIN
+      if (v3) mf += its
+      mf += ',' + k + ','
+      mf += n + ','
+      mf += frag + ','
+      mfs.push(mf)
+    }
+
+    return [null, mfs]
+  }
+
+  HLP.splitype = function splitype(arr, msg) {
+    var data = []
+    arr.forEach(function (a) {
+      var str
+      switch (a) {
+        case 'PUBKEY':
+          str = splitype(['SHORT', 'MPI', 'MPI', 'MPI', 'MPI'], msg).join('')
+          break
+        case 'DATA':  // falls through
+        case 'MPI':
+          str = msg.substring(0, HLP.readLen(msg) + 4)
+          break
+        default:
+          str = msg.substring(0, DTS[a])
+      }
+      data.push(str)
+      msg = msg.substring(str.length)
+    })
+    return data
+  }
+
+  // https://github.com/msgpack/msgpack-javascript/blob/master/msgpack.js
+
+  var _bin2num = (function () {
+    var i = 0, _bin2num = {}
+    for (; i < 0x100; ++i) {
+      _bin2num[String.fromCharCode(i)] = i  // "\00" -> 0x00
+    }
+    for (i = 0x80; i < 0x100; ++i) {  // [Webkit][Gecko]
+      _bin2num[String.fromCharCode(0xf700 + i)] = i  // "\f780" -> 0x80
+    }
+    return _bin2num
+  }())
+
+  HLP.toByteArray = function (data) {
+    var rv = []
+      , ary = data.split("")
+      , i = -1
+      , iz = ary.length
+      , remain = iz % 8
+
+    while (remain--) {
+      ++i
+      rv[i] = _bin2num[ary[i]]
+    }
+    remain = iz >> 3
+    while (remain--) {
+      rv.push(_bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]],
+              _bin2num[ary[++i]], _bin2num[ary[++i]])
+    }
+    return rv
+  }
+
+}).call(this)
\ No newline at end of file
diff --git a/lib/otr/lib/sm-webworker.js b/lib/otr/lib/sm-webworker.js
new file mode 100644
index 0000000..301843d
--- /dev/null
+++ b/lib/otr/lib/sm-webworker.js
@@ -0,0 +1,60 @@
+;(function (root) {
+  "use strict";
+
+  root.OTR = {}
+  root.crypto = {
+    randomBytes: function () {
+      throw new Error("Haven't seeded yet.")
+    }
+  }
+
+  // default imports
+  var imports = [
+      'vendor/salsa20.js'
+    , 'vendor/bigint.js'
+    , 'vendor/crypto.js'
+    , 'vendor/eventemitter.js'
+    , 'lib/const.js'
+    , 'lib/helpers.js'
+    , 'lib/sm.js'
+  ]
+
+  function wrapPostMessage(method) {
+    return function () {
+      postMessage({
+          method: method
+        , args: Array.prototype.slice.call(arguments, 0)
+      })
+    }
+  }
+
+  var sm
+  onmessage = function (e) {
+    var data = e.data
+    switch (data.type) {
+      case 'seed':
+        if (data.imports) imports = data.imports
+        importScripts.apply(root, imports)
+
+        // use salsa20 since there's no prng in webworkers
+        var state = new root.Salsa20(
+          data.seed.slice(0, 32),
+          data.seed.slice(32)
+        )
+        root.crypto.randomBytes = function (n) {
+          return state.getBytes(n)
+        }
+        break
+      case 'init':
+        sm = new root.OTR.SM(data.reqs)
+        ;['trust','question', 'send', 'abort'].forEach(function (m) {
+          sm.on(m, wrapPostMessage(m));
+        })
+        break
+      case 'method':
+        sm[data.method].apply(sm, data.args)
+        break
+    }
+  }
+
+}(this))
\ No newline at end of file
diff --git a/lib/otr/vendor/bigint.js b/lib/otr/vendor/bigint.js
new file mode 100644
index 0000000..b7271da
--- /dev/null
+++ b/lib/otr/vendor/bigint.js
@@ -0,0 +1,1705 @@
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define(factory.bind(root, root.crypto || root.msCrypto))
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory(require('crypto'))
+  } else {
+    root.BigInt = factory(root.crypto || root.msCrypto)
+  }
+
+}(this, function (crypto) {
+
+  ////////////////////////////////////////////////////////////////////////////////////////
+  // Big Integer Library v. 5.5
+  // Created 2000, last modified 2013
+  // Leemon Baird
+  // www.leemon.com
+  //
+  // Version history:
+  // v 5.5  17 Mar 2013
+  //   - two lines of a form like "if (x<0) x+=n" had the "if" changed to "while" to
+  //     handle the case when x<-n. (Thanks to James Ansell for finding that bug)
+  // v 5.4  3 Oct 2009
+  //   - added "var i" to greaterShift() so i is not global. (Thanks to Péter Szabó for finding that bug)
+  //
+  // v 5.3  21 Sep 2009
+  //   - added randProbPrime(k) for probable primes
+  //   - unrolled loop in mont_ (slightly faster)
+  //   - millerRabin now takes a bigInt parameter rather than an int
+  //
+  // v 5.2  15 Sep 2009
+  //   - fixed capitalization in call to int2bigInt in randBigInt
+  //     (thanks to Emili Evripidou, Reinhold Behringer, and Samuel Macaleese for finding that bug)
+  //
+  // v 5.1  8 Oct 2007 
+  //   - renamed inverseModInt_ to inverseModInt since it doesn't change its parameters
+  //   - added functions GCD and randBigInt, which call GCD_ and randBigInt_
+  //   - fixed a bug found by Rob Visser (see comment with his name below)
+  //   - improved comments
+  //
+  // This file is public domain.   You can use it for any purpose without restriction.
+  // I do not guarantee that it is correct, so use it at your own risk.  If you use 
+  // it for something interesting, I'd appreciate hearing about it.  If you find 
+  // any bugs or make any improvements, I'd appreciate hearing about those too.
+  // It would also be nice if my name and URL were left in the comments.  But none 
+  // of that is required.
+  //
+  // This code defines a bigInt library for arbitrary-precision integers.
+  // A bigInt is an array of integers storing the value in chunks of bpe bits, 
+  // little endian (buff[0] is the least significant word).
+  // Negative bigInts are stored two's complement.  Almost all the functions treat
+  // bigInts as nonnegative.  The few that view them as two's complement say so
+  // in their comments.  Some functions assume their parameters have at least one 
+  // leading zero element. Functions with an underscore at the end of the name put
+  // their answer into one of the arrays passed in, and have unpredictable behavior 
+  // in case of overflow, so the caller must make sure the arrays are big enough to 
+  // hold the answer.  But the average user should never have to call any of the 
+  // underscored functions.  Each important underscored function has a wrapper function 
+  // of the same name without the underscore that takes care of the details for you.  
+  // For each underscored function where a parameter is modified, that same variable 
+  // must not be used as another argument too.  So, you cannot square x by doing 
+  // multMod_(x,x,n).  You must use squareMod_(x,n) instead, or do y=dup(x); multMod_(x,y,n).
+  // Or simply use the multMod(x,x,n) function without the underscore, where
+  // such issues never arise, because non-underscored functions never change
+  // their parameters; they always allocate new memory for the answer that is returned.
+  //
+  // These functions are designed to avoid frequent dynamic memory allocation in the inner loop.
+  // For most functions, if it needs a BigInt as a local variable it will actually use
+  // a global, and will only allocate to it only when it's not the right size.  This ensures
+  // that when a function is called repeatedly with same-sized parameters, it only allocates
+  // memory on the first call.
+  //
+  // Note that for cryptographic purposes, the calls to Math.random() must 
+  // be replaced with calls to a better pseudorandom number generator.
+  //
+  // In the following, "bigInt" means a bigInt with at least one leading zero element,
+  // and "integer" means a nonnegative integer less than radix.  In some cases, integer 
+  // can be negative.  Negative bigInts are 2s complement.
+  // 
+  // The following functions do not modify their inputs.
+  // Those returning a bigInt, string, or Array will dynamically allocate memory for that value.
+  // Those returning a boolean will return the integer 0 (false) or 1 (true).
+  // Those returning boolean or int will not allocate memory except possibly on the first 
+  // time they're called with a given parameter size.
+  // 
+  // bigInt  add(x,y)               //return (x+y) for bigInts x and y.  
+  // bigInt  addInt(x,n)            //return (x+n) where x is a bigInt and n is an integer.
+  // string  bigInt2str(x,base)     //return a string form of bigInt x in a given base, with 2 <= base <= 95
+  // int     bitSize(x)             //return how many bits long the bigInt x is, not counting leading zeros
+  // bigInt  dup(x)                 //return a copy of bigInt x
+  // boolean equals(x,y)            //is the bigInt x equal to the bigint y?
+  // boolean equalsInt(x,y)         //is bigint x equal to integer y?
+  // bigInt  expand(x,n)            //return a copy of x with at least n elements, adding leading zeros if needed
+  // Array   findPrimes(n)          //return array of all primes less than integer n
+  // bigInt  GCD(x,y)               //return greatest common divisor of bigInts x and y (each with same number of elements).
+  // boolean greater(x,y)           //is x>y?  (x and y are nonnegative bigInts)
+  // boolean greaterShift(x,y,shift)//is (x <<(shift*bpe)) > y?
+  // bigInt  int2bigInt(t,n,m)      //return a bigInt equal to integer t, with at least n bits and m array elements
+  // bigInt  inverseMod(x,n)        //return (x**(-1) mod n) for bigInts x and n.  If no inverse exists, it returns null
+  // int     inverseModInt(x,n)     //return x**(-1) mod n, for integers x and n.  Return 0 if there is no inverse
+  // boolean isZero(x)              //is the bigInt x equal to zero?
+  // boolean millerRabin(x,b)       //does one round of Miller-Rabin base integer b say that bigInt x is possibly prime? (b is bigInt, 1<b<x)
+  // boolean millerRabinInt(x,b)    //does one round of Miller-Rabin base integer b say that bigInt x is possibly prime? (b is int,    1<b<x)
+  // bigInt  mod(x,n)               //return a new bigInt equal to (x mod n) for bigInts x and n.
+  // int     modInt(x,n)            //return x mod n for bigInt x and integer n.
+  // bigInt  mult(x,y)              //return x*y for bigInts x and y. This is faster when y<x.
+  // bigInt  multMod(x,y,n)         //return (x*y mod n) for bigInts x,y,n.  For greater speed, let y<x.
+  // boolean negative(x)            //is bigInt x negative?
+  // bigInt  powMod(x,y,n)          //return (x**y mod n) where x,y,n are bigInts and ** is exponentiation.  0**0=1. Faster for odd n.
+  // bigInt  randBigInt(n,s)        //return an n-bit random BigInt (n>=1).  If s=1, then the most significant of those n bits is set to 1.
+  // bigInt  randTruePrime(k)       //return a new, random, k-bit, true prime bigInt using Maurer's algorithm.
+  // bigInt  randProbPrime(k)       //return a new, random, k-bit, probable prime bigInt (probability it's composite less than 2^-80).
+  // bigInt  str2bigInt(s,b,n,m)    //return a bigInt for number represented in string s in base b with at least n bits and m array elements
+  // bigInt  sub(x,y)               //return (x-y) for bigInts x and y.  Negative answers will be 2s complement
+  // bigInt  trim(x,k)              //return a copy of x with exactly k leading zero elements
+  //
+  //
+  // The following functions each have a non-underscored version, which most users should call instead.
+  // These functions each write to a single parameter, and the caller is responsible for ensuring the array 
+  // passed in is large enough to hold the result. 
+  //
+  // void    addInt_(x,n)          //do x=x+n where x is a bigInt and n is an integer
+  // void    add_(x,y)             //do x=x+y for bigInts x and y
+  // void    copy_(x,y)            //do x=y on bigInts x and y
+  // void    copyInt_(x,n)         //do x=n on bigInt x and integer n
+  // void    GCD_(x,y)             //set x to the greatest common divisor of bigInts x and y, (y is destroyed).  (This never overflows its array).
+  // boolean inverseMod_(x,n)      //do x=x**(-1) mod n, for bigInts x and n. Returns 1 (0) if inverse does (doesn't) exist
+  // void    mod_(x,n)             //do x=x mod n for bigInts x and n. (This never overflows its array).
+  // void    mult_(x,y)            //do x=x*y for bigInts x and y.
+  // void    multMod_(x,y,n)       //do x=x*y  mod n for bigInts x,y,n.
+  // void    powMod_(x,y,n)        //do x=x**y mod n, where x,y,n are bigInts (n is odd) and ** is exponentiation.  0**0=1.
+  // void    randBigInt_(b,n,s)    //do b = an n-bit random BigInt. if s=1, then nth bit (most significant bit) is set to 1. n>=1.
+  // void    randTruePrime_(ans,k) //do ans = a random k-bit true random prime (not just probable prime) with 1 in the msb.
+  // void    sub_(x,y)             //do x=x-y for bigInts x and y. Negative answers will be 2s complement.
+  //
+  // The following functions do NOT have a non-underscored version. 
+  // They each write a bigInt result to one or more parameters.  The caller is responsible for
+  // ensuring the arrays passed in are large enough to hold the results. 
+  //
+  // void addShift_(x,y,ys)       //do x=x+(y<<(ys*bpe))
+  // void carry_(x)               //do carries and borrows so each element of the bigInt x fits in bpe bits.
+  // void divide_(x,y,q,r)        //divide x by y giving quotient q and remainder r
+  // int  divInt_(x,n)            //do x=floor(x/n) for bigInt x and integer n, and return the remainder. (This never overflows its array).
+  // int  eGCD_(x,y,d,a,b)        //sets a,b,d to positive bigInts such that d = GCD_(x,y) = a*x-b*y
+  // void halve_(x)               //do x=floor(|x|/2)*sgn(x) for bigInt x in 2's complement.  (This never overflows its array).
+  // void leftShift_(x,n)         //left shift bigInt x by n bits.  n<bpe.
+  // void linComb_(x,y,a,b)       //do x=a*x+b*y for bigInts x and y and integers a and b
+  // void linCombShift_(x,y,b,ys) //do x=x+b*(y<<(ys*bpe)) for bigInts x and y, and integers b and ys
+  // void mont_(x,y,n,np)         //Montgomery multiplication (see comments where the function is defined)
+  // void multInt_(x,n)           //do x=x*n where x is a bigInt and n is an integer.
+  // void rightShift_(x,n)        //right shift bigInt x by n bits. (This never overflows its array).
+  // void squareMod_(x,n)         //do x=x*x  mod n for bigInts x,n
+  // void subShift_(x,y,ys)       //do x=x-(y<<(ys*bpe)). Negative answers will be 2s complement.
+  //
+  // The following functions are based on algorithms from the _Handbook of Applied Cryptography_
+  //    powMod_()           = algorithm 14.94, Montgomery exponentiation
+  //    eGCD_,inverseMod_() = algorithm 14.61, Binary extended GCD_
+  //    GCD_()              = algorothm 14.57, Lehmer's algorithm
+  //    mont_()             = algorithm 14.36, Montgomery multiplication
+  //    divide_()           = algorithm 14.20  Multiple-precision division
+  //    squareMod_()        = algorithm 14.16  Multiple-precision squaring
+  //    randTruePrime_()    = algorithm  4.62, Maurer's algorithm
+  //    millerRabin()       = algorithm  4.24, Miller-Rabin algorithm
+  //
+  // Profiling shows:
+  //     randTruePrime_() spends:
+  //         10% of its time in calls to powMod_()
+  //         85% of its time in calls to millerRabin()
+  //     millerRabin() spends:
+  //         99% of its time in calls to powMod_()   (always with a base of 2)
+  //     powMod_() spends:
+  //         94% of its time in calls to mont_()  (almost always with x==y)
+  //
+  // This suggests there are several ways to speed up this library slightly:
+  //     - convert powMod_ to use a Montgomery form of k-ary window (or maybe a Montgomery form of sliding window)
+  //         -- this should especially focus on being fast when raising 2 to a power mod n
+  //     - convert randTruePrime_() to use a minimum r of 1/3 instead of 1/2 with the appropriate change to the test
+  //     - tune the parameters in randTruePrime_(), including c, m, and recLimit
+  //     - speed up the single loop in mont_() that takes 95% of the runtime, perhaps by reducing checking
+  //       within the loop when all the parameters are the same length.
+  //
+  // There are several ideas that look like they wouldn't help much at all:
+  //     - replacing trial division in randTruePrime_() with a sieve (that speeds up something taking almost no time anyway)
+  //     - increase bpe from 15 to 30 (that would help if we had a 32*32->64 multiplier, but not with JavaScript's 32*32->32)
+  //     - speeding up mont_(x,y,n,np) when x==y by doing a non-modular, non-Montgomery square
+  //       followed by a Montgomery reduction.  The intermediate answer will be twice as long as x, so that
+  //       method would be slower.  This is unfortunate because the code currently spends almost all of its time
+  //       doing mont_(x,x,...), both for randTruePrime_() and powMod_().  A faster method for Montgomery squaring
+  //       would have a large impact on the speed of randTruePrime_() and powMod_().  HAC has a couple of poorly-worded
+  //       sentences that seem to imply it's faster to do a non-modular square followed by a single
+  //       Montgomery reduction, but that's obviously wrong.
+  ////////////////////////////////////////////////////////////////////////////////////////
+
+  //globals
+
+  // The number of significant bits in the fraction of a JavaScript
+  // floating-point number is 52, independent of platform.
+  // See: https://github.com/arlolra/otr/issues/41
+
+  var bpe = 26;          // bits stored per array element
+  var radix = 1 << bpe;  // equals 2^bpe
+  var mask = radix - 1;  // AND this with an array element to chop it down to bpe bits
+
+  //the digits for converting to different bases
+  var digitsStr='0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz_=!@#$%^&*()[]{}|;:,.<>/?`~ \\\'\"+-';
+
+  var one=int2bigInt(1,1,1);     //constant used in powMod_()
+
+  //the following global variables are scratchpad memory to 
+  //reduce dynamic memory allocation in the inner loop
+  var t=new Array(0);
+  var ss=t;       //used in mult_()
+  var s0=t;       //used in multMod_(), squareMod_()
+  var s1=t;       //used in powMod_(), multMod_(), squareMod_()
+  var s2=t;       //used in powMod_(), multMod_()
+  var s3=t;       //used in powMod_()
+  var s4=t, s5=t; //used in mod_()
+  var s6=t;       //used in bigInt2str()
+  var s7=t;       //used in powMod_()
+  var T=t;        //used in GCD_()
+  var sa=t;       //used in mont_()
+  var mr_x1=t, mr_r=t, mr_a=t;                                      //used in millerRabin()
+  var eg_v=t, eg_u=t, eg_A=t, eg_B=t, eg_C=t, eg_D=t;               //used in eGCD_(), inverseMod_()
+  var md_q1=t, md_q2=t, md_q3=t, md_r=t, md_r1=t, md_r2=t, md_tt=t; //used in mod_()
+
+  var primes=t, pows=t, s_i=t, s_i2=t, s_R=t, s_rm=t, s_q=t, s_n1=t;
+  var s_a=t, s_r2=t, s_n=t, s_b=t, s_d=t, s_x1=t, s_x2=t, s_aa=t; //used in randTruePrime_()
+    
+  var rpprb=t; //used in randProbPrimeRounds() (which also uses "primes")
+
+  ////////////////////////////////////////////////////////////////////////////////////////
+
+
+  //return array of all primes less than integer n
+  function findPrimes(n) {
+    var i,s,p,ans;
+    s=new Array(n);
+    for (i=0;i<n;i++)
+      s[i]=0;
+    s[0]=2;
+    p=0;    //first p elements of s are primes, the rest are a sieve
+    for(;s[p]<n;) {                  //s[p] is the pth prime
+      for(i=s[p]*s[p]; i<n; i+=s[p]) //mark multiples of s[p]
+        s[i]=1;
+      p++;
+      s[p]=s[p-1]+1;
+      for(; s[p]<n && s[s[p]]; s[p]++); //find next prime (where s[p]==0)
+    }
+    ans=new Array(p);
+    for(i=0;i<p;i++)
+      ans[i]=s[i];
+    return ans;
+  }
+
+
+  //does a single round of Miller-Rabin base b consider x to be a possible prime?
+  //x is a bigInt, and b is an integer, with b<x
+  function millerRabinInt(x,b) {
+    if (mr_x1.length!=x.length) {
+      mr_x1=dup(x);
+      mr_r=dup(x);
+      mr_a=dup(x);
+    }
+
+    copyInt_(mr_a,b);
+    return millerRabin(x,mr_a);
+  }
+
+  //does a single round of Miller-Rabin base b consider x to be a possible prime?
+  //x and b are bigInts with b<x
+  function millerRabin(x,b) {
+    var i,j,k,s;
+
+    if (mr_x1.length!=x.length) {
+      mr_x1=dup(x);
+      mr_r=dup(x);
+      mr_a=dup(x);
+    }
+
+    copy_(mr_a,b);
+    copy_(mr_r,x);
+    copy_(mr_x1,x);
+
+    addInt_(mr_r,-1);
+    addInt_(mr_x1,-1);
+
+    //s=the highest power of two that divides mr_r
+
+    /*
+    k=0;
+    for (i=0;i<mr_r.length;i++)
+      for (j=1;j<mask;j<<=1)
+        if (x[i] & j) {
+          s=(k<mr_r.length+bpe ? k : 0); 
+           i=mr_r.length;
+           j=mask;
+        } else
+          k++;
+    */
+
+    /* http://www.javascripter.net/math/primes/millerrabinbug-bigint54.htm */
+    if (isZero(mr_r)) return 0;
+    for (k=0; mr_r[k]==0; k++);
+    for (i=1,j=2; mr_r[k]%j==0; j*=2,i++ );
+    s = k*bpe + i - 1;
+    /* end */
+
+    if (s)                
+      rightShift_(mr_r,s);
+
+    powMod_(mr_a,mr_r,x);
+
+    if (!equalsInt(mr_a,1) && !equals(mr_a,mr_x1)) {
+      j=1;
+      while (j<=s-1 && !equals(mr_a,mr_x1)) {
+        squareMod_(mr_a,x);
+        if (equalsInt(mr_a,1)) {
+          return 0;
+        }
+        j++;
+      }
+      if (!equals(mr_a,mr_x1)) {
+        return 0;
+      }
+    }
+    return 1;  
+  }
+
+  //returns how many bits long the bigInt is, not counting leading zeros.
+  function bitSize(x) {
+    var j,z,w;
+    for (j=x.length-1; (x[j]==0) && (j>0); j--);
+    for (z=0,w=x[j]; w; (w>>=1),z++);
+    z+=bpe*j;
+    return z;
+  }
+
+  //return a copy of x with at least n elements, adding leading zeros if needed
+  function expand(x,n) {
+    var ans=int2bigInt(0,(x.length>n ? x.length : n)*bpe,0);
+    copy_(ans,x);
+    return ans;
+  }
+
+  //return a k-bit true random prime using Maurer's algorithm.
+  function randTruePrime(k) {
+    var ans=int2bigInt(0,k,0);
+    randTruePrime_(ans,k);
+    return trim(ans,1);
+  }
+
+  //return a k-bit random probable prime with probability of error < 2^-80
+  function randProbPrime(k) {
+    if (k>=600) return randProbPrimeRounds(k,2); //numbers from HAC table 4.3
+    if (k>=550) return randProbPrimeRounds(k,4);
+    if (k>=500) return randProbPrimeRounds(k,5);
+    if (k>=400) return randProbPrimeRounds(k,6);
+    if (k>=350) return randProbPrimeRounds(k,7);
+    if (k>=300) return randProbPrimeRounds(k,9);
+    if (k>=250) return randProbPrimeRounds(k,12); //numbers from HAC table 4.4
+    if (k>=200) return randProbPrimeRounds(k,15);
+    if (k>=150) return randProbPrimeRounds(k,18);
+    if (k>=100) return randProbPrimeRounds(k,27);
+                return randProbPrimeRounds(k,40); //number from HAC remark 4.26 (only an estimate)
+  }
+
+  //return a k-bit probable random prime using n rounds of Miller Rabin (after trial division with small primes)
+  function randProbPrimeRounds(k,n) {
+    var ans, i, divisible, B; 
+    B=30000;  //B is largest prime to use in trial division
+    ans=int2bigInt(0,k,0);
+    
+    //optimization: try larger and smaller B to find the best limit.
+    
+    if (primes.length==0)
+      primes=findPrimes(30000);  //check for divisibility by primes <=30000
+
+    if (rpprb.length!=ans.length)
+      rpprb=dup(ans);
+
+    for (;;) { //keep trying random values for ans until one appears to be prime
+      //optimization: pick a random number times L=2*3*5*...*p, plus a 
+      //   random element of the list of all numbers in [0,L) not divisible by any prime up to p.
+      //   This can reduce the amount of random number generation.
+      
+      randBigInt_(ans,k,0); //ans = a random odd number to check
+      ans[0] |= 1; 
+      divisible=0;
+    
+      //check ans for divisibility by small primes up to B
+      for (i=0; (i<primes.length) && (primes[i]<=B); i++)
+        if (modInt(ans,primes[i])==0 && !equalsInt(ans,primes[i])) {
+          divisible=1;
+          break;
+        }      
+      
+      //optimization: change millerRabin so the base can be bigger than the number being checked, then eliminate the while here.
+      
+      //do n rounds of Miller Rabin, with random bases less than ans
+      for (i=0; i<n && !divisible; i++) {
+        randBigInt_(rpprb,k,0);
+        while(!greater(ans,rpprb)) //pick a random rpprb that's < ans
+          randBigInt_(rpprb,k,0);
+        if (!millerRabin(ans,rpprb))
+          divisible=1;
+      }
+      
+      if(!divisible)
+        return ans;
+    }  
+  }
+
+  //return a new bigInt equal to (x mod n) for bigInts x and n.
+  function mod(x,n) {
+    var ans=dup(x);
+    mod_(ans,n);
+    return trim(ans,1);
+  }
+
+  //return (x+n) where x is a bigInt and n is an integer.
+  function addInt(x,n) {
+    var ans=expand(x,x.length+1);
+    addInt_(ans,n);
+    return trim(ans,1);
+  }
+
+  //return x*y for bigInts x and y. This is faster when y<x.
+  function mult(x,y) {
+    var ans=expand(x,x.length+y.length);
+    mult_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x**y mod n) where x,y,n are bigInts and ** is exponentiation.  0**0=1. Faster for odd n.
+  function powMod(x,y,n) {
+    var ans=expand(x,n.length);  
+    powMod_(ans,trim(y,2),trim(n,2),0);  //this should work without the trim, but doesn't
+    return trim(ans,1);
+  }
+
+  //return (x-y) for bigInts x and y.  Negative answers will be 2s complement
+  function sub(x,y) {
+    var ans=expand(x,(x.length>y.length ? x.length+1 : y.length+1)); 
+    sub_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x+y) for bigInts x and y.  
+  function add(x,y) {
+    var ans=expand(x,(x.length>y.length ? x.length+1 : y.length+1)); 
+    add_(ans,y);
+    return trim(ans,1);
+  }
+
+  //return (x**(-1) mod n) for bigInts x and n.  If no inverse exists, it returns null
+  function inverseMod(x,n) {
+    var ans=expand(x,n.length); 
+    var s;
+    s=inverseMod_(ans,n);
+    return s ? trim(ans,1) : null;
+  }
+
+  //return (x*y mod n) for bigInts x,y,n.  For greater speed, let y<x.
+  function multMod(x,y,n) {
+    var ans=expand(x,n.length);
+    multMod_(ans,y,n);
+    return trim(ans,1);
+  }
+
+  //generate a k-bit true random prime using Maurer's algorithm,
+  //and put it into ans.  The bigInt ans must be large enough to hold it.
+  function randTruePrime_(ans,k) {
+    var c,w,m,pm,dd,j,r,B,divisible,z,zz,recSize,recLimit;
+
+    if (primes.length==0)
+      primes=findPrimes(30000);  //check for divisibility by primes <=30000
+
+    if (pows.length==0) {
+      pows=new Array(512);
+      for (j=0;j<512;j++) {
+        pows[j]=Math.pow(2,j/511.0-1.0);
+      }
+    }
+
+    //c and m should be tuned for a particular machine and value of k, to maximize speed
+    c=0.1;  //c=0.1 in HAC
+    m=20;   //generate this k-bit number by first recursively generating a number that has between k/2 and k-m bits
+    recLimit=20; //stop recursion when k <=recLimit.  Must have recLimit >= 2
+
+    if (s_i2.length!=ans.length) {
+      s_i2=dup(ans);
+      s_R =dup(ans);
+      s_n1=dup(ans);
+      s_r2=dup(ans);
+      s_d =dup(ans);
+      s_x1=dup(ans);
+      s_x2=dup(ans);
+      s_b =dup(ans);
+      s_n =dup(ans);
+      s_i =dup(ans);
+      s_rm=dup(ans);
+      s_q =dup(ans);
+      s_a =dup(ans);
+      s_aa=dup(ans);
+    }
+
+    if (k <= recLimit) {  //generate small random primes by trial division up to its square root
+      pm=(1<<((k+2)>>1))-1; //pm is binary number with all ones, just over sqrt(2^k)
+      copyInt_(ans,0);
+      for (dd=1;dd;) {
+        dd=0;
+        ans[0]= 1 | (1<<(k-1)) | randomBitInt(k);  //random, k-bit, odd integer, with msb 1
+        for (j=1;(j<primes.length) && ((primes[j]&pm)==primes[j]);j++) { //trial division by all primes 3...sqrt(2^k)
+          if (0==(ans[0]%primes[j])) {
+            dd=1;
+            break;
+          }
+        }
+      }
+      carry_(ans);
+      return;
+    }
+
+    B=c*k*k;    //try small primes up to B (or all the primes[] array if the largest is less than B).
+    if (k>2*m)  //generate this k-bit number by first recursively generating a number that has between k/2 and k-m bits
+      for (r=1; k-k*r<=m; )
+        r=pows[randomBitInt(9)];   //r=Math.pow(2,Math.random()-1);
+    else
+      r=0.5;
+
+    //simulation suggests the more complex algorithm using r=.333 is only slightly faster.
+
+    recSize=Math.floor(r*k)+1;
+
+    randTruePrime_(s_q,recSize);
+    copyInt_(s_i2,0);
+    s_i2[Math.floor((k-2)/bpe)] |= (1<<((k-2)%bpe));   //s_i2=2^(k-2)
+    divide_(s_i2,s_q,s_i,s_rm);                        //s_i=floor((2^(k-1))/(2q))
+
+    z=bitSize(s_i);
+
+    for (;;) {
+      for (;;) {  //generate z-bit numbers until one falls in the range [0,s_i-1]
+        randBigInt_(s_R,z,0);
+        if (greater(s_i,s_R))
+          break;
+      }                //now s_R is in the range [0,s_i-1]
+      addInt_(s_R,1);  //now s_R is in the range [1,s_i]
+      add_(s_R,s_i);   //now s_R is in the range [s_i+1,2*s_i]
+
+      copy_(s_n,s_q);
+      mult_(s_n,s_R); 
+      multInt_(s_n,2);
+      addInt_(s_n,1);    //s_n=2*s_R*s_q+1
+      
+      copy_(s_r2,s_R);
+      multInt_(s_r2,2);  //s_r2=2*s_R
+
+      //check s_n for divisibility by small primes up to B
+      for (divisible=0,j=0; (j<primes.length) && (primes[j]<B); j++)
+        if (modInt(s_n,primes[j])==0 && !equalsInt(s_n,primes[j])) {
+          divisible=1;
+          break;
+        }      
+
+      if (!divisible)    //if it passes small primes check, then try a single Miller-Rabin base 2
+        if (!millerRabinInt(s_n,2)) //this line represents 75% of the total runtime for randTruePrime_ 
+          divisible=1;
+
+      if (!divisible) {  //if it passes that test, continue checking s_n
+        addInt_(s_n,-3);
+        for (j=s_n.length-1;(s_n[j]==0) && (j>0); j--);  //strip leading zeros
+        for (zz=0,w=s_n[j]; w; (w>>=1),zz++);
+        zz+=bpe*j;                             //zz=number of bits in s_n, ignoring leading zeros
+        for (;;) {  //generate z-bit numbers until one falls in the range [0,s_n-1]
+          randBigInt_(s_a,zz,0);
+          if (greater(s_n,s_a))
+            break;
+        }                //now s_a is in the range [0,s_n-1]
+        addInt_(s_n,3);  //now s_a is in the range [0,s_n-4]
+        addInt_(s_a,2);  //now s_a is in the range [2,s_n-2]
+        copy_(s_b,s_a);
+        copy_(s_n1,s_n);
+        addInt_(s_n1,-1);
+        powMod_(s_b,s_n1,s_n);   //s_b=s_a^(s_n-1) modulo s_n
+        addInt_(s_b,-1);
+        if (isZero(s_b)) {
+          copy_(s_b,s_a);
+          powMod_(s_b,s_r2,s_n);
+          addInt_(s_b,-1);
+          copy_(s_aa,s_n);
+          copy_(s_d,s_b);
+          GCD_(s_d,s_n);  //if s_b and s_n are relatively prime, then s_n is a prime
+          if (equalsInt(s_d,1)) {
+            copy_(ans,s_aa);
+            return;     //if we've made it this far, then s_n is absolutely guaranteed to be prime
+          }
+        }
+      }
+    }
+  }
+
+  //Return an n-bit random BigInt (n>=1).  If s=1, then the most significant of those n bits is set to 1.
+  function randBigInt(n,s) {
+    var a,b;
+    a=Math.floor((n-1)/bpe)+2; //# array elements to hold the BigInt with a leading 0 element
+    b=int2bigInt(0,0,a);
+    randBigInt_(b,n,s);
+    return b;
+  }
+
+  //Set b to an n-bit random BigInt.  If s=1, then the most significant of those n bits is set to 1.
+  //Array b must be big enough to hold the result. Must have n>=1
+  function randBigInt_(b,n,s) {
+    var i,a;
+    for (i=0;i<b.length;i++)
+      b[i]=0;
+    a=Math.floor((n-1)/bpe)+1; //# array elements to hold the BigInt
+    for (i=0;i<a;i++) {
+      b[i]=randomBitInt(bpe);
+    }
+    b[a-1] &= (2<<((n-1)%bpe))-1;
+    if (s==1)
+      b[a-1] |= (1<<((n-1)%bpe));
+  }
+
+  //Return the greatest common divisor of bigInts x and y (each with same number of elements).
+  function GCD(x,y) {
+    var xc,yc;
+    xc=dup(x);
+    yc=dup(y);
+    GCD_(xc,yc);
+    return xc;
+  }
+
+  //set x to the greatest common divisor of bigInts x and y (each with same number of elements).
+  //y is destroyed.
+  function GCD_(x,y) {
+    var i,xp,yp,A,B,C,D,q,sing,qp;
+    if (T.length!=x.length)
+      T=dup(x);
+
+    sing=1;
+    while (sing) { //while y has nonzero elements other than y[0]
+      sing=0;
+      for (i=1;i<y.length;i++) //check if y has nonzero elements other than 0
+        if (y[i]) {
+          sing=1;
+          break;
+        }
+      if (!sing) break; //quit when y all zero elements except possibly y[0]
+
+      for (i=x.length;!x[i] && i>=0;i--);  //find most significant element of x
+      xp=x[i];
+      yp=y[i];
+      A=1; B=0; C=0; D=1;
+      while ((yp+C) && (yp+D)) {
+        q =Math.floor((xp+A)/(yp+C));
+        qp=Math.floor((xp+B)/(yp+D));
+        if (q!=qp)
+          break;
+        t= A-q*C;   A=C;   C=t;    //  do (A,B,xp, C,D,yp) = (C,D,yp, A,B,xp) - q*(0,0,0, C,D,yp)      
+        t= B-q*D;   B=D;   D=t;
+        t=xp-q*yp; xp=yp; yp=t;
+      }
+      if (B) {
+        copy_(T,x);
+        linComb_(x,y,A,B); //x=A*x+B*y
+        linComb_(y,T,D,C); //y=D*y+C*T
+      } else {
+        mod_(x,y);
+        copy_(T,x);
+        copy_(x,y);
+        copy_(y,T);
+      } 
+    }
+    if (y[0]==0)
+      return;
+    t=modInt(x,y[0]);
+    copyInt_(x,y[0]);
+    y[0]=t;
+    while (y[0]) {
+      x[0]%=y[0];
+      t=x[0]; x[0]=y[0]; y[0]=t;
+    }
+  }
+
+  //do x=x**(-1) mod n, for bigInts x and n.
+  //If no inverse exists, it sets x to zero and returns 0, else it returns 1.
+  //The x array must be at least as large as the n array.
+  function inverseMod_(x,n) {
+    var k=1+2*Math.max(x.length,n.length);
+
+    if(!(x[0]&1)  && !(n[0]&1)) {  //if both inputs are even, then inverse doesn't exist
+      copyInt_(x,0);
+      return 0;
+    }
+
+    if (eg_u.length!=k) {
+      eg_u=new Array(k);
+      eg_v=new Array(k);
+      eg_A=new Array(k);
+      eg_B=new Array(k);
+      eg_C=new Array(k);
+      eg_D=new Array(k);
+    }
+
+    copy_(eg_u,x);
+    copy_(eg_v,n);
+    copyInt_(eg_A,1);
+    copyInt_(eg_B,0);
+    copyInt_(eg_C,0);
+    copyInt_(eg_D,1);
+    for (;;) {
+      while(!(eg_u[0]&1)) {  //while eg_u is even
+        halve_(eg_u);
+        if (!(eg_A[0]&1) && !(eg_B[0]&1)) { //if eg_A==eg_B==0 mod 2
+          halve_(eg_A);
+          halve_(eg_B);      
+        } else {
+          add_(eg_A,n);  halve_(eg_A);
+          sub_(eg_B,x);  halve_(eg_B);
+        }
+      }
+
+      while (!(eg_v[0]&1)) {  //while eg_v is even
+        halve_(eg_v);
+        if (!(eg_C[0]&1) && !(eg_D[0]&1)) { //if eg_C==eg_D==0 mod 2
+          halve_(eg_C);
+          halve_(eg_D);      
+        } else {
+          add_(eg_C,n);  halve_(eg_C);
+          sub_(eg_D,x);  halve_(eg_D);
+        }
+      }
+
+      if (!greater(eg_v,eg_u)) { //eg_v <= eg_u
+        sub_(eg_u,eg_v);
+        sub_(eg_A,eg_C);
+        sub_(eg_B,eg_D);
+      } else {                   //eg_v > eg_u
+        sub_(eg_v,eg_u);
+        sub_(eg_C,eg_A);
+        sub_(eg_D,eg_B);
+      }
+
+      if (equalsInt(eg_u,0)) {
+        while (negative(eg_C)) //make sure answer is nonnegative
+          add_(eg_C,n);
+        copy_(x,eg_C);
+
+        if (!equalsInt(eg_v,1)) { //if GCD_(x,n)!=1, then there is no inverse
+          copyInt_(x,0);
+          return 0;
+        }
+        return 1;
+      }
+    }
+  }
+
+  //return x**(-1) mod n, for integers x and n.  Return 0 if there is no inverse
+  function inverseModInt(x,n) {
+    var a=1,b=0,t;
+    for (;;) {
+      if (x==1) return a;
+      if (x==0) return 0;
+      b-=a*Math.floor(n/x);
+      n%=x;
+
+      if (n==1) return b; //to avoid negatives, change this b to n-b, and each -= to +=
+      if (n==0) return 0;
+      a-=b*Math.floor(x/n);
+      x%=n;
+    }
+  }
+
+  //this deprecated function is for backward compatibility only. 
+  function inverseModInt_(x,n) {
+     return inverseModInt(x,n);
+  }
+
+
+  //Given positive bigInts x and y, change the bigints v, a, and b to positive bigInts such that:
+  //     v = GCD_(x,y) = a*x-b*y
+  //The bigInts v, a, b, must have exactly as many elements as the larger of x and y.
+  function eGCD_(x,y,v,a,b) {
+    var g=0;
+    var k=Math.max(x.length,y.length);
+    if (eg_u.length!=k) {
+      eg_u=new Array(k);
+      eg_A=new Array(k);
+      eg_B=new Array(k);
+      eg_C=new Array(k);
+      eg_D=new Array(k);
+    }
+    while(!(x[0]&1)  && !(y[0]&1)) {  //while x and y both even
+      halve_(x);
+      halve_(y);
+      g++;
+    }
+    copy_(eg_u,x);
+    copy_(v,y);
+    copyInt_(eg_A,1);
+    copyInt_(eg_B,0);
+    copyInt_(eg_C,0);
+    copyInt_(eg_D,1);
+    for (;;) {
+      while(!(eg_u[0]&1)) {  //while u is even
+        halve_(eg_u);
+        if (!(eg_A[0]&1) && !(eg_B[0]&1)) { //if A==B==0 mod 2
+          halve_(eg_A);
+          halve_(eg_B);      
+        } else {
+          add_(eg_A,y);  halve_(eg_A);
+          sub_(eg_B,x);  halve_(eg_B);
+        }
+      }
+
+      while (!(v[0]&1)) {  //while v is even
+        halve_(v);
+        if (!(eg_C[0]&1) && !(eg_D[0]&1)) { //if C==D==0 mod 2
+          halve_(eg_C);
+          halve_(eg_D);      
+        } else {
+          add_(eg_C,y);  halve_(eg_C);
+          sub_(eg_D,x);  halve_(eg_D);
+        }
+      }
+
+      if (!greater(v,eg_u)) { //v<=u
+        sub_(eg_u,v);
+        sub_(eg_A,eg_C);
+        sub_(eg_B,eg_D);
+      } else {                //v>u
+        sub_(v,eg_u);
+        sub_(eg_C,eg_A);
+        sub_(eg_D,eg_B);
+      }
+      if (equalsInt(eg_u,0)) {
+        while (negative(eg_C)) {   //make sure a (C) is nonnegative
+          add_(eg_C,y);
+          sub_(eg_D,x);
+        }
+        multInt_(eg_D,-1);  ///make sure b (D) is nonnegative
+        copy_(a,eg_C);
+        copy_(b,eg_D);
+        leftShift_(v,g);
+        return;
+      }
+    }
+  }
+
+
+  //is bigInt x negative?
+  function negative(x) {
+    return ((x[x.length-1]>>(bpe-1))&1);
+  }
+
+
+  //is (x << (shift*bpe)) > y?
+  //x and y are nonnegative bigInts
+  //shift is a nonnegative integer
+  function greaterShift(x,y,shift) {
+    var i, kx=x.length, ky=y.length;
+    var k=((kx+shift)<ky) ? (kx+shift) : ky;
+    for (i=ky-1-shift; i<kx && i>=0; i++) 
+      if (x[i]>0)
+        return 1; //if there are nonzeros in x to the left of the first column of y, then x is bigger
+    for (i=kx-1+shift; i<ky; i++)
+      if (y[i]>0)
+        return 0; //if there are nonzeros in y to the left of the first column of x, then x is not bigger
+    for (i=k-1; i>=shift; i--)
+      if      (x[i-shift]>y[i]) return 1;
+      else if (x[i-shift]<y[i]) return 0;
+    return 0;
+  }
+
+  //is x > y? (x and y both nonnegative)
+  function greater(x,y) {
+    var i;
+    var k=(x.length<y.length) ? x.length : y.length;
+
+    for (i=x.length;i<y.length;i++)
+      if (y[i])
+        return 0;  //y has more digits
+
+    for (i=y.length;i<x.length;i++)
+      if (x[i])
+        return 1;  //x has more digits
+
+    for (i=k-1;i>=0;i--)
+      if (x[i]>y[i])
+        return 1;
+      else if (x[i]<y[i])
+        return 0;
+    return 0;
+  }
+
+  //divide x by y giving quotient q and remainder r.  (q=floor(x/y),  r=x mod y).  All 4 are bigints.
+  //x must have at least one leading zero element.
+  //y must be nonzero.
+  //q and r must be arrays that are exactly the same length as x. (Or q can have more).
+  //Must have x.length >= y.length >= 2.
+  function divide_(x,y,q,r) {
+    var kx, ky;
+    var i,j,y1,y2,c,a,b;
+    copy_(r,x);
+    for (ky=y.length;y[ky-1]==0;ky--); //ky is number of elements in y, not including leading zeros
+
+    //normalize: ensure the most significant element of y has its highest bit set  
+    b=y[ky-1];
+    for (a=0; b; a++)
+      b>>=1;  
+    a=bpe-a;  //a is how many bits to shift so that the high order bit of y is leftmost in its array element
+    leftShift_(y,a);  //multiply both by 1<<a now, then divide both by that at the end
+    leftShift_(r,a);
+
+    //Rob Visser discovered a bug: the following line was originally just before the normalization.
+    for (kx=r.length;r[kx-1]==0 && kx>ky;kx--); //kx is number of elements in normalized x, not including leading zeros
+
+    copyInt_(q,0);                      // q=0
+    while (!greaterShift(y,r,kx-ky)) {  // while (leftShift_(y,kx-ky) <= r) {
+      subShift_(r,y,kx-ky);             //   r=r-leftShift_(y,kx-ky)
+      q[kx-ky]++;                       //   q[kx-ky]++;
+    }                                   // }
+
+    for (i=kx-1; i>=ky; i--) {
+      if (r[i]==y[ky-1])
+        q[i-ky]=mask;
+      else
+        q[i-ky]=Math.floor((r[i]*radix+r[i-1])/y[ky-1]);
+
+      //The following for(;;) loop is equivalent to the commented while loop, 
+      //except that the uncommented version avoids overflow.
+      //The commented loop comes from HAC, which assumes r[-1]==y[-1]==0
+      //  while (q[i-ky]*(y[ky-1]*radix+y[ky-2]) > r[i]*radix*radix+r[i-1]*radix+r[i-2])
+      //    q[i-ky]--;    
+      for (;;) {
+        y2=(ky>1 ? y[ky-2] : 0)*q[i-ky];
+        c=y2;
+        y2=y2 & mask;
+        c = (c - y2) / radix;
+        y1=c+q[i-ky]*y[ky-1];
+        c=y1;
+        y1=y1 & mask;
+        c = (c - y1) / radix;
+
+        if (c==r[i] ? y1==r[i-1] ? y2>(i>1 ? r[i-2] : 0) : y1>r[i-1] : c>r[i]) 
+          q[i-ky]--;
+        else
+          break;
+      }
+
+      linCombShift_(r,y,-q[i-ky],i-ky);    //r=r-q[i-ky]*leftShift_(y,i-ky)
+      if (negative(r)) {
+        addShift_(r,y,i-ky);         //r=r+leftShift_(y,i-ky)
+        q[i-ky]--;
+      }
+    }
+
+    rightShift_(y,a);  //undo the normalization step
+    rightShift_(r,a);  //undo the normalization step
+  }
+
+  //do carries and borrows so each element of the bigInt x fits in bpe bits.
+  function carry_(x) {
+    var i,k,c,b;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i];
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+    }
+  }
+
+  //return x mod n for bigInt x and integer n.
+  function modInt(x,n) {
+    var i,c=0;
+    for (i=x.length-1; i>=0; i--)
+      c=(c*radix+x[i])%n;
+    return c;
+  }
+
+  //convert the integer t into a bigInt with at least the given number of bits.
+  //the returned array stores the bigInt in bpe-bit chunks, little endian (buff[0] is least significant word)
+  //Pad the array with leading zeros so that it has at least minSize elements.
+  //There will always be at least one leading 0 element.
+  function int2bigInt(t,bits,minSize) {   
+    var i,k, buff;
+    k=Math.ceil(bits/bpe)+1;
+    k=minSize>k ? minSize : k;
+    buff=new Array(k);
+    copyInt_(buff,t);
+    return buff;
+  }
+
+  //return the bigInt given a string representation in a given base.  
+  //Pad the array with leading zeros so that it has at least minSize elements.
+  //If base=-1, then it reads in a space-separated list of array elements in decimal.
+  //The array will always have at least one leading zero, unless base=-1.
+  function str2bigInt(s,base,minSize) {
+    var d, i, j, x, y, kk;
+    var k=s.length;
+    if (base==-1) { //comma-separated list of array elements in decimal
+      x=new Array(0);
+      for (;;) {
+        y=new Array(x.length+1);
+        for (i=0;i<x.length;i++)
+          y[i+1]=x[i];
+        y[0]=parseInt(s,10);
+        x=y;
+        d=s.indexOf(',',0);
+        if (d<1) 
+          break;
+        s=s.substring(d+1);
+        if (s.length==0)
+          break;
+      }
+      if (x.length<minSize) {
+        y=new Array(minSize);
+        copy_(y,x);
+        return y;
+      }
+      return x;
+    }
+
+    // log2(base)*k
+    var bb = base, p = 0;
+    var b = base == 1 ? k : 0;
+    while (bb > 1) {
+      if (bb & 1) p = 1;
+      b += k;
+      bb >>= 1;
+    }
+    b += p*k;
+
+    x=int2bigInt(0,b,0);
+    for (i=0;i<k;i++) {
+      d=digitsStr.indexOf(s.substring(i,i+1),0);
+      if (base<=36 && d>=36)  //convert lowercase to uppercase if base<=36
+        d-=26;
+      if (d>=base || d<0) {   //stop at first illegal character
+        break;
+      }
+      multInt_(x,base);
+      addInt_(x,d);
+    }
+
+    for (k=x.length;k>0 && !x[k-1];k--); //strip off leading zeros
+    k=minSize>k+1 ? minSize : k+1;
+    y=new Array(k);
+    kk=k<x.length ? k : x.length;
+    for (i=0;i<kk;i++)
+      y[i]=x[i];
+    for (;i<k;i++)
+      y[i]=0;
+    return y;
+  }
+
+  //is bigint x equal to integer y?
+  //y must have less than bpe bits
+  function equalsInt(x,y) {
+    var i;
+    if (x[0]!=y)
+      return 0;
+    for (i=1;i<x.length;i++)
+      if (x[i])
+        return 0;
+    return 1;
+  }
+
+  //are bigints x and y equal?
+  //this works even if x and y are different lengths and have arbitrarily many leading zeros
+  function equals(x,y) {
+    var i;
+    var k=x.length<y.length ? x.length : y.length;
+    for (i=0;i<k;i++)
+      if (x[i]!=y[i])
+        return 0;
+    if (x.length>y.length) {
+      for (;i<x.length;i++)
+        if (x[i])
+          return 0;
+    } else {
+      for (;i<y.length;i++)
+        if (y[i])
+          return 0;
+    }
+    return 1;
+  }
+
+  //is the bigInt x equal to zero?
+  function isZero(x) {
+    var i;
+    for (i=0;i<x.length;i++)
+      if (x[i])
+        return 0;
+    return 1;
+  }
+
+  //convert a bigInt into a string in a given base, from base 2 up to base 95.
+  //Base -1 prints the contents of the array representing the number.
+  function bigInt2str(x,base) {
+    var i,t,s="";
+
+    if (s6.length!=x.length) 
+      s6=dup(x);
+    else
+      copy_(s6,x);
+
+    if (base==-1) { //return the list of array contents
+      for (i=x.length-1;i>0;i--)
+        s+=x[i]+',';
+      s+=x[0];
+    }
+    else { //return it in the given base
+      while (!isZero(s6)) {
+        t=divInt_(s6,base);  //t=s6 % base; s6=floor(s6/base);
+        s=digitsStr.substring(t,t+1)+s;
+      }
+    }
+    if (s.length==0)
+      s="0";
+    return s;
+  }
+
+  //returns a duplicate of bigInt x
+  function dup(x) {
+    var i, buff;
+    buff=new Array(x.length);
+    copy_(buff,x);
+    return buff;
+  }
+
+  //do x=y on bigInts x and y.  x must be an array at least as big as y (not counting the leading zeros in y).
+  function copy_(x,y) {
+    var i;
+    var k=x.length<y.length ? x.length : y.length;
+    for (i=0;i<k;i++)
+      x[i]=y[i];
+    for (i=k;i<x.length;i++)
+      x[i]=0;
+  }
+
+  //do x=y on bigInt x and integer y.  
+  function copyInt_(x,n) {
+    var i,c;
+    for (c=n,i=0;i<x.length;i++) {
+      x[i]=c & mask;
+      c>>=bpe;
+    }
+  }
+
+  //do x=x+n where x is a bigInt and n is an integer.
+  //x must be large enough to hold the result.
+  function addInt_(x,n) {
+    var i,k,c,b;
+    x[0]+=n;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i];
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+      if (!c) return; //stop carrying as soon as the carry is zero
+    }
+  }
+
+  //right shift bigInt x by n bits.
+  function rightShift_(x,n) {
+    var i;
+    var k=Math.floor(n/bpe);
+    if (k) {
+      for (i=0;i<x.length-k;i++) //right shift x by k elements
+        x[i]=x[i+k];
+      for (;i<x.length;i++)
+        x[i]=0;
+      n%=bpe;
+    }
+    for (i=0;i<x.length-1;i++) {
+      x[i]=mask & ((x[i+1]<<(bpe-n)) | (x[i]>>n));
+    }
+    x[i]>>=n;
+  }
+
+  //do x=floor(|x|/2)*sgn(x) for bigInt x in 2's complement
+  function halve_(x) {
+    var i;
+    for (i=0;i<x.length-1;i++) {
+      x[i]=mask & ((x[i+1]<<(bpe-1)) | (x[i]>>1));
+    }
+    x[i]=(x[i]>>1) | (x[i] & (radix>>1));  //most significant bit stays the same
+  }
+
+  //left shift bigInt x by n bits.
+  function leftShift_(x,n) {
+    var i;
+    var k=Math.floor(n/bpe);
+    if (k) {
+      for (i=x.length; i>=k; i--) //left shift x by k elements
+        x[i]=x[i-k];
+      for (;i>=0;i--)
+        x[i]=0;  
+      n%=bpe;
+    }
+    if (!n)
+      return;
+    for (i=x.length-1;i>0;i--) {
+      x[i]=mask & ((x[i]<<n) | (x[i-1]>>(bpe-n)));
+    }
+    x[i]=mask & (x[i]<<n);
+  }
+
+  //do x=x*n where x is a bigInt and n is an integer.
+  //x must be large enough to hold the result.
+  function multInt_(x,n) {
+    var i,k,c,b;
+    if (!n)
+      return;
+    k=x.length;
+    c=0;
+    for (i=0;i<k;i++) {
+      c+=x[i]*n;
+      b=0;
+      if (c<0) {
+        b = c & mask;
+        b = -((c - b) / radix);
+        c+=b*radix;
+      }
+      x[i]=c & mask;
+      c = ((c - x[i]) / radix) - b;
+    }
+  }
+
+  //do x=floor(x/n) for bigInt x and integer n, and return the remainder
+  function divInt_(x,n) {
+    var i,r=0,s;
+    for (i=x.length-1;i>=0;i--) {
+      s=r*radix+x[i];
+      x[i]=Math.floor(s/n);
+      r=s%n;
+    }
+    return r;
+  }
+
+  //do the linear combination x=a*x+b*y for bigInts x and y, and integers a and b.
+  //x must be large enough to hold the answer.
+  function linComb_(x,y,a,b) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    kk=x.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=a*x[i]+b*y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;i<kk;i++) {
+      c+=a*x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do the linear combination x=a*x+b*(y<<(ys*bpe)) for bigInts x and y, and integers a, b and ys.
+  //x must be large enough to hold the answer.
+  function linCombShift_(x,y,b,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]+b*y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x+(y<<(ys*bpe)) for bigInts x and y, and integers a,b and ys.
+  //x must be large enough to hold the answer.
+  function addShift_(x,y,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]+y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x-(y<<(ys*bpe)) for bigInts x and y, and integers a,b and ys.
+  //x must be large enough to hold the answer.
+  function subShift_(x,y,ys) {
+    var i,c,k,kk;
+    k=x.length<ys+y.length ? x.length : ys+y.length;
+    kk=x.length;
+    for (c=0,i=ys;i<k;i++) {
+      c+=x[i]-y[i-ys];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<kk;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x-y for bigInts x and y.
+  //x must be large enough to hold the answer.
+  //negative answers will be 2s complement
+  function sub_(x,y) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=x[i]-y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<x.length;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x+y for bigInts x and y.
+  //x must be large enough to hold the answer.
+  function add_(x,y) {
+    var i,c,k,kk;
+    k=x.length<y.length ? x.length : y.length;
+    for (c=0,i=0;i<k;i++) {
+      c+=x[i]+y[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+    for (i=k;c && i<x.length;i++) {
+      c+=x[i];
+      x[i]=c & mask;
+      c = (c - x[i]) / radix;
+    }
+  }
+
+  //do x=x*y for bigInts x and y.  This is faster when y<x.
+  function mult_(x,y) {
+    var i;
+    if (ss.length!=2*x.length)
+      ss=new Array(2*x.length);
+    copyInt_(ss,0);
+    for (i=0;i<y.length;i++)
+      if (y[i])
+        linCombShift_(ss,x,y[i],i);   //ss=1*ss+y[i]*(x<<(i*bpe))
+    copy_(x,ss);
+  }
+
+  //do x=x mod n for bigInts x and n.
+  function mod_(x,n) {
+    if (s4.length!=x.length)
+      s4=dup(x);
+    else
+      copy_(s4,x);
+    if (s5.length!=x.length)
+      s5=dup(x);  
+    divide_(s4,n,s5,x);  //x = remainder of s4 / n
+  }
+
+  //do x=x*y mod n for bigInts x,y,n.
+  //for greater speed, let y<x.
+  function multMod_(x,y,n) {
+    var i;
+    if (s0.length!=2*x.length)
+      s0=new Array(2*x.length);
+    copyInt_(s0,0);
+    for (i=0;i<y.length;i++)
+      if (y[i])
+        linCombShift_(s0,x,y[i],i);   //s0=1*s0+y[i]*(x<<(i*bpe))
+    mod_(s0,n);
+    copy_(x,s0);
+  }
+
+  //do x=x*x mod n for bigInts x,n.
+  function squareMod_(x,n) {
+    var i,j,d,c,kx,kn,k;
+    for (kx=x.length; kx>0 && !x[kx-1]; kx--);  //ignore leading zeros in x
+    k=kx>n.length ? 2*kx : 2*n.length; //k=# elements in the product, which is twice the elements in the larger of x and n
+    if (s0.length!=k) 
+      s0=new Array(k);
+    copyInt_(s0,0);
+    for (i=0;i<kx;i++) {
+      c=s0[2*i]+x[i]*x[i];
+      s0[2*i]=c & mask;
+      c = (c - s0[2*i]) / radix;
+      for (j=i+1;j<kx;j++) {
+        c=s0[i+j]+2*x[i]*x[j]+c;
+        s0[i+j]=(c & mask);
+        c = (c - s0[i+j]) / radix;
+      }
+      s0[i+kx]=c;
+    }
+    mod_(s0,n);
+    copy_(x,s0);
+  }
+
+  //return x with exactly k leading zero elements
+  function trim(x,k) {
+    var i,y;
+    for (i=x.length; i>0 && !x[i-1]; i--);
+    y=new Array(i+k);
+    copy_(y,x);
+    return y;
+  }
+
+  //do x=x**y mod n, where x,y,n are bigInts and ** is exponentiation.  0**0=1.
+  //this is faster when n is odd.  x usually needs to have as many elements as n.
+  function powMod_(x,y,n) {
+    var k1,k2,kn,np;
+    if(s7.length!=n.length)
+      s7=dup(n);
+
+    //for even modulus, use a simple square-and-multiply algorithm,
+    //rather than using the more complex Montgomery algorithm.
+    if ((n[0]&1)==0) {
+      copy_(s7,x);
+      copyInt_(x,1);
+      while(!equalsInt(y,0)) {
+        if (y[0]&1)
+          multMod_(x,s7,n);
+        divInt_(y,2);
+        squareMod_(s7,n); 
+      }
+      return;
+    }
+
+    //calculate np from n for the Montgomery multiplications
+    copyInt_(s7,0);
+    for (kn=n.length;kn>0 && !n[kn-1];kn--);
+    np=radix-inverseModInt(modInt(n,radix),radix);
+    s7[kn]=1;
+    multMod_(x ,s7,n);   // x = x * 2**(kn*bp) mod n
+
+    if (s3.length!=x.length)
+      s3=dup(x);
+    else
+      copy_(s3,x);
+
+    for (k1=y.length-1;k1>0 & !y[k1]; k1--);  //k1=first nonzero element of y
+    if (y[k1]==0) {  //anything to the 0th power is 1
+      copyInt_(x,1);
+      return;
+    }
+    for (k2=1<<(bpe-1);k2 && !(y[k1] & k2); k2>>=1);  //k2=position of first 1 bit in y[k1]
+    for (;;) {
+      if (!(k2>>=1)) {  //look at next bit of y
+        k1--;
+        if (k1<0) {
+          mont_(x,one,n,np);
+          return;
+        }
+        k2=1<<(bpe-1);
+      }    
+      mont_(x,x,n,np);
+
+      if (k2 & y[k1]) //if next bit is a 1
+        mont_(x,s3,n,np);
+    }
+  }
+
+
+  //do x=x*y*Ri mod n for bigInts x,y,n, 
+  //  where Ri = 2**(-kn*bpe) mod n, and kn is the 
+  //  number of elements in the n array, not 
+  //  counting leading zeros.  
+  //x array must have at least as many elemnts as the n array
+  //It's OK if x and y are the same variable.
+  //must have:
+  //  x,y < n
+  //  n is odd
+  //  np = -(n^(-1)) mod radix
+  function mont_(x,y,n,np) {
+    var i,j,c,ui,t,t2,ks;
+    var kn=n.length;
+    var ky=y.length;
+
+    if (sa.length!=kn)
+      sa=new Array(kn);
+      
+    copyInt_(sa,0);
+
+    for (;kn>0 && n[kn-1]==0;kn--); //ignore leading zeros of n
+    for (;ky>0 && y[ky-1]==0;ky--); //ignore leading zeros of y
+    ks=sa.length-1; //sa will never have more than this many nonzero elements.  
+
+    //the following loop consumes 95% of the runtime for randTruePrime_() and powMod_() for large numbers
+    for (i=0; i<kn; i++) {
+      t=sa[0]+x[i]*y[0];
+      ui=((t & mask) * np) & mask;  //the inner "& mask" was needed on Safari (but not MSIE) at one time
+      c=(t+ui*n[0]);
+      c = (c - (c & mask)) / radix;
+      t=x[i];
+      
+      //do sa=(sa+x[i]*y+ui*n)/b   where b=2**bpe.  Loop is unrolled 5-fold for speed
+      j=1;
+      for (;j<ky-4;) {
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<ky;)   {
+        c+=sa[j]+ui*n[j]+t*y[j]; t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<kn-4;) {
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<kn;)   {
+        c+=sa[j]+ui*n[j];        t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      for (;j<ks;)   {
+        c+=sa[j];                t2=sa[j-1]=c & mask; c=(c-t2)/radix; j++;
+      }
+      sa[j-1]=c & mask;
+    }
+
+    if (!greater(n,sa))
+      sub_(sa,n);
+    copy_(x,sa);
+  }
+
+
+  // otr.js additions
+
+
+  // computes num / den mod n
+  function divMod(num, den, n) {
+    return multMod(num, inverseMod(den, n), n)
+  }
+
+  // computes one - two mod n
+  function subMod(one, two, n) {
+    one = mod(one, n)
+    two = mod(two, n)
+    if (greater(two, one)) one = add(one, n)
+    return sub(one, two)
+  }
+
+  // computes 2^m as a bigInt
+  function twoToThe(m) {
+    var b = Math.floor(m / bpe) + 2
+    var t = new Array(b)
+    for (var i = 0; i < b; i++) t[i] = 0
+    t[b - 2] = 1 << (m % bpe)
+    return t
+  }
+
+  // cache these results for faster lookup
+  var _num2bin = (function () {
+    var i = 0, _num2bin= {}
+    for (; i < 0x100; ++i) {
+      _num2bin[i] = String.fromCharCode(i)  // 0 -> "\00"
+    }
+    return _num2bin
+  }())
+
+  // serialize a bigInt to an ascii string
+  // padded up to pad length
+  function bigInt2bits(bi, pad) {
+    pad || (pad = 0)
+    bi = dup(bi)
+    var ba = ''
+    while (!isZero(bi)) {
+      ba = _num2bin[bi[0] & 0xff] + ba
+      rightShift_(bi, 8)
+    }
+    while (ba.length < pad) {
+      ba = '\x00' + ba
+    }
+    return ba
+  }
+
+  // converts a byte array to a bigInt
+  function ba2bigInt(data) {
+    var mpi = str2bigInt('0', 10, data.length)
+    data.forEach(function (d, i) {
+      if (i) leftShift_(mpi, 8)
+      mpi[0] |= d
+    })
+    return mpi
+  }
+
+  // returns a function that returns an array of n bytes
+  var randomBytes = (function () {
+
+    // in node
+    if ( typeof crypto !== 'undefined' &&
+      typeof crypto.randomBytes === 'function' ) {
+      return function (n) {
+        try {
+          var buf = crypto.randomBytes(n)
+        } catch (e) { throw e }
+        return Array.prototype.slice.call(buf, 0)
+      }
+    }
+
+    // in browser
+    else if ( typeof crypto !== 'undefined' &&
+      typeof crypto.getRandomValues === 'function' ) {
+      return function (n) {
+        var buf = new Uint8Array(n)
+        crypto.getRandomValues(buf)
+        return Array.prototype.slice.call(buf, 0)
+      }
+    }
+
+    // err
+    else {
+      throw new Error('Keys should not be generated without CSPRNG.')
+    }
+
+  }())
+
+  // Salsa 20 in webworker needs a 40 byte seed
+  function getSeed() {
+    return randomBytes(40)
+  }
+
+  // returns a single random byte
+  function randomByte() {
+    return randomBytes(1)[0]
+  }
+
+  // returns a k-bit random integer
+  function randomBitInt(k) {
+    if (k > 31) throw new Error("Too many bits.")
+    var i = 0, r = 0
+    var b = Math.floor(k / 8)
+    var mask = (1 << (k % 8)) - 1
+    if (mask) r = randomByte() & mask
+    for (; i < b; i++)
+      r = (256 * r) + randomByte()
+    return r
+  }
+
+  return {
+      str2bigInt    : str2bigInt
+    , bigInt2str    : bigInt2str
+    , int2bigInt    : int2bigInt
+    , multMod       : multMod
+    , powMod        : powMod
+    , inverseMod    : inverseMod
+    , randBigInt    : randBigInt
+    , randBigInt_   : randBigInt_
+    , equals        : equals
+    , equalsInt     : equalsInt
+    , sub           : sub
+    , mod           : mod
+    , modInt        : modInt
+    , mult          : mult
+    , divInt_       : divInt_
+    , rightShift_   : rightShift_
+    , dup           : dup
+    , greater       : greater
+    , add           : add
+    , isZero        : isZero
+    , bitSize       : bitSize
+    , millerRabin   : millerRabin
+    , divide_       : divide_
+    , trim          : trim
+    , primes        : primes
+    , findPrimes    : findPrimes
+    , getSeed       : getSeed
+    , divMod        : divMod
+    , subMod        : subMod
+    , twoToThe      : twoToThe
+    , bigInt2bits   : bigInt2bits
+    , ba2bigInt     : ba2bigInt
+  }
+
+}))
\ No newline at end of file
diff --git a/lib/otr/vendor/crypto.js b/lib/otr/vendor/crypto.js
new file mode 100644
index 0000000..d34731b
--- /dev/null
+++ b/lib/otr/vendor/crypto.js
@@ -0,0 +1,2434 @@
+;(function (root, factory) {
+
+  if (typeof define === "function" && define.amd) {
+    define(factory)
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory()
+  } else {
+    root.CryptoJS = factory()
+  }
+
+}(this, function () {
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * CryptoJS core components.
+ */
+var CryptoJS = CryptoJS || (function (Math, undefined) {
+    /**
+     * CryptoJS namespace.
+     */
+    var C = {};
+
+    /**
+     * Library namespace.
+     */
+    var C_lib = C.lib = {};
+
+    /**
+     * Base object for prototypal inheritance.
+     */
+    var Base = C_lib.Base = (function () {
+        function F() {}
+
+        return {
+            /**
+             * Creates a new object that inherits from this object.
+             *
+             * @param {Object} overrides Properties to copy into the new object.
+             *
+             * @return {Object} The new object.
+             *
+             * @static
+             *
+             * @example
+             *
+             *     var MyType = CryptoJS.lib.Base.extend({
+             *         field: 'value',
+             *
+             *         method: function () {
+             *         }
+             *     });
+             */
+            extend: function (overrides) {
+                // Spawn
+                F.prototype = this;
+                var subtype = new F();
+
+                // Augment
+                if (overrides) {
+                    subtype.mixIn(overrides);
+                }
+
+                // Create default initializer
+                if (!subtype.hasOwnProperty('init')) {
+                    subtype.init = function () {
+                        subtype.$super.init.apply(this, arguments);
+                    };
+                }
+
+                // Initializer's prototype is the subtype object
+                subtype.init.prototype = subtype;
+
+                // Reference supertype
+                subtype.$super = this;
+
+                return subtype;
+            },
+
+            /**
+             * Extends this object and runs the init method.
+             * Arguments to create() will be passed to init().
+             *
+             * @return {Object} The new object.
+             *
+             * @static
+             *
+             * @example
+             *
+             *     var instance = MyType.create();
+             */
+            create: function () {
+                var instance = this.extend();
+                instance.init.apply(instance, arguments);
+
+                return instance;
+            },
+
+            /**
+             * Initializes a newly created object.
+             * Override this method to add some logic when your objects are created.
+             *
+             * @example
+             *
+             *     var MyType = CryptoJS.lib.Base.extend({
+             *         init: function () {
+             *             // ...
+             *         }
+             *     });
+             */
+            init: function () {
+            },
+
+            /**
+             * Copies properties into this object.
+             *
+             * @param {Object} properties The properties to mix in.
+             *
+             * @example
+             *
+             *     MyType.mixIn({
+             *         field: 'value'
+             *     });
+             */
+            mixIn: function (properties) {
+                for (var propertyName in properties) {
+                    if (properties.hasOwnProperty(propertyName)) {
+                        this[propertyName] = properties[propertyName];
+                    }
+                }
+
+                // IE won't copy toString using the loop above
+                if (properties.hasOwnProperty('toString')) {
+                    this.toString = properties.toString;
+                }
+            },
+
+            /**
+             * Creates a copy of this object.
+             *
+             * @return {Object} The clone.
+             *
+             * @example
+             *
+             *     var clone = instance.clone();
+             */
+            clone: function () {
+                return this.init.prototype.extend(this);
+            }
+        };
+    }());
+
+    /**
+     * An array of 32-bit words.
+     *
+     * @property {Array} words The array of 32-bit words.
+     * @property {number} sigBytes The number of significant bytes in this word array.
+     */
+    var WordArray = C_lib.WordArray = Base.extend({
+        /**
+         * Initializes a newly created word array.
+         *
+         * @param {Array} words (Optional) An array of 32-bit words.
+         * @param {number} sigBytes (Optional) The number of significant bytes in the words.
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.lib.WordArray.create();
+         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607]);
+         *     var wordArray = CryptoJS.lib.WordArray.create([0x00010203, 0x04050607], 6);
+         */
+        init: function (words, sigBytes) {
+            words = this.words = words || [];
+
+            if (sigBytes != undefined) {
+                this.sigBytes = sigBytes;
+            } else {
+                this.sigBytes = words.length * 4;
+            }
+        },
+
+        /**
+         * Converts this word array to a string.
+         *
+         * @param {Encoder} encoder (Optional) The encoding strategy to use. Default: CryptoJS.enc.Hex
+         *
+         * @return {string} The stringified word array.
+         *
+         * @example
+         *
+         *     var string = wordArray + '';
+         *     var string = wordArray.toString();
+         *     var string = wordArray.toString(CryptoJS.enc.Utf8);
+         */
+        toString: function (encoder) {
+            return (encoder || Hex).stringify(this);
+        },
+
+        /**
+         * Concatenates a word array to this word array.
+         *
+         * @param {WordArray} wordArray The word array to append.
+         *
+         * @return {WordArray} This word array.
+         *
+         * @example
+         *
+         *     wordArray1.concat(wordArray2);
+         */
+        concat: function (wordArray) {
+            // Shortcuts
+            var thisWords = this.words;
+            var thatWords = wordArray.words;
+            var thisSigBytes = this.sigBytes;
+            var thatSigBytes = wordArray.sigBytes;
+
+            // Clamp excess bits
+            this.clamp();
+
+            // Concat
+            if (thisSigBytes % 4) {
+                // Copy one byte at a time
+                for (var i = 0; i < thatSigBytes; i++) {
+                    var thatByte = (thatWords[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                    thisWords[(thisSigBytes + i) >>> 2] |= thatByte << (24 - ((thisSigBytes + i) % 4) * 8);
+                }
+            } else if (thatWords.length > 0xffff) {
+                // Copy one word at a time
+                for (var i = 0; i < thatSigBytes; i += 4) {
+                    thisWords[(thisSigBytes + i) >>> 2] = thatWords[i >>> 2];
+                }
+            } else {
+                // Copy all words at once
+                thisWords.push.apply(thisWords, thatWords);
+            }
+            this.sigBytes += thatSigBytes;
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Removes insignificant bits.
+         *
+         * @example
+         *
+         *     wordArray.clamp();
+         */
+        clamp: function () {
+            // Shortcuts
+            var words = this.words;
+            var sigBytes = this.sigBytes;
+
+            // Clamp
+            words[sigBytes >>> 2] &= 0xffffffff << (32 - (sigBytes % 4) * 8);
+            words.length = Math.ceil(sigBytes / 4);
+        },
+
+        /**
+         * Creates a copy of this word array.
+         *
+         * @return {WordArray} The clone.
+         *
+         * @example
+         *
+         *     var clone = wordArray.clone();
+         */
+        clone: function () {
+            var clone = Base.clone.call(this);
+            clone.words = this.words.slice(0);
+
+            return clone;
+        },
+
+        /**
+         * Creates a word array filled with random bytes.
+         *
+         * @param {number} nBytes The number of random bytes to generate.
+         *
+         * @return {WordArray} The random word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.lib.WordArray.random(16);
+         */
+        random: function (nBytes) {
+            var words = [];
+            for (var i = 0; i < nBytes; i += 4) {
+                words.push((Math.random() * 0x100000000) | 0);
+            }
+
+            return new WordArray.init(words, nBytes);
+        }
+    });
+
+    /**
+     * Encoder namespace.
+     */
+    var C_enc = C.enc = {};
+
+    /**
+     * Hex encoding strategy.
+     */
+    var Hex = C_enc.Hex = {
+        /**
+         * Converts a word array to a hex string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The hex string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var hexString = CryptoJS.enc.Hex.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+
+            // Convert
+            var hexChars = [];
+            for (var i = 0; i < sigBytes; i++) {
+                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                hexChars.push((bite >>> 4).toString(16));
+                hexChars.push((bite & 0x0f).toString(16));
+            }
+
+            return hexChars.join('');
+        },
+
+        /**
+         * Converts a hex string to a word array.
+         *
+         * @param {string} hexStr The hex string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Hex.parse(hexString);
+         */
+        parse: function (hexStr) {
+            // Shortcut
+            var hexStrLength = hexStr.length;
+
+            // Convert
+            var words = [];
+            for (var i = 0; i < hexStrLength; i += 2) {
+                words[i >>> 3] |= parseInt(hexStr.substr(i, 2), 16) << (24 - (i % 8) * 4);
+            }
+
+            return new WordArray.init(words, hexStrLength / 2);
+        }
+    };
+
+    /**
+     * Latin1 encoding strategy.
+     */
+    var Latin1 = C_enc.Latin1 = {
+        /**
+         * Converts a word array to a Latin1 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The Latin1 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var latin1String = CryptoJS.enc.Latin1.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+
+            // Convert
+            var latin1Chars = [];
+            for (var i = 0; i < sigBytes; i++) {
+                var bite = (words[i >>> 2] >>> (24 - (i % 4) * 8)) & 0xff;
+                latin1Chars.push(String.fromCharCode(bite));
+            }
+
+            return latin1Chars.join('');
+        },
+
+        /**
+         * Converts a Latin1 string to a word array.
+         *
+         * @param {string} latin1Str The Latin1 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Latin1.parse(latin1String);
+         */
+        parse: function (latin1Str) {
+            // Shortcut
+            var latin1StrLength = latin1Str.length;
+
+            // Convert
+            var words = [];
+            for (var i = 0; i < latin1StrLength; i++) {
+                words[i >>> 2] |= (latin1Str.charCodeAt(i) & 0xff) << (24 - (i % 4) * 8);
+            }
+
+            return new WordArray.init(words, latin1StrLength);
+        }
+    };
+
+    /**
+     * UTF-8 encoding strategy.
+     */
+    var Utf8 = C_enc.Utf8 = {
+        /**
+         * Converts a word array to a UTF-8 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The UTF-8 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var utf8String = CryptoJS.enc.Utf8.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            try {
+                return decodeURIComponent(escape(Latin1.stringify(wordArray)));
+            } catch (e) {
+                throw new Error('Malformed UTF-8 data');
+            }
+        },
+
+        /**
+         * Converts a UTF-8 string to a word array.
+         *
+         * @param {string} utf8Str The UTF-8 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Utf8.parse(utf8String);
+         */
+        parse: function (utf8Str) {
+            return Latin1.parse(unescape(encodeURIComponent(utf8Str)));
+        }
+    };
+
+    /**
+     * Abstract buffered block algorithm template.
+     *
+     * The property blockSize must be implemented in a concrete subtype.
+     *
+     * @property {number} _minBufferSize The number of blocks that should be kept unprocessed in the buffer. Default: 0
+     */
+    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm = Base.extend({
+        /**
+         * Resets this block algorithm's data buffer to its initial state.
+         *
+         * @example
+         *
+         *     bufferedBlockAlgorithm.reset();
+         */
+        reset: function () {
+            // Initial values
+            this._data = new WordArray.init();
+            this._nDataBytes = 0;
+        },
+
+        /**
+         * Adds new data to this block algorithm's buffer.
+         *
+         * @param {WordArray|string} data The data to append. Strings are converted to a WordArray using UTF-8.
+         *
+         * @example
+         *
+         *     bufferedBlockAlgorithm._append('data');
+         *     bufferedBlockAlgorithm._append(wordArray);
+         */
+        _append: function (data) {
+            // Convert string to WordArray, else assume WordArray already
+            if (typeof data == 'string') {
+                data = Utf8.parse(data);
+            }
+
+            // Append
+            this._data.concat(data);
+            this._nDataBytes += data.sigBytes;
+        },
+
+        /**
+         * Processes available data blocks.
+         *
+         * This method invokes _doProcessBlock(offset), which must be implemented by a concrete subtype.
+         *
+         * @param {boolean} doFlush Whether all blocks and partial blocks should be processed.
+         *
+         * @return {WordArray} The processed data.
+         *
+         * @example
+         *
+         *     var processedData = bufferedBlockAlgorithm._process();
+         *     var processedData = bufferedBlockAlgorithm._process(!!'flush');
+         */
+        _process: function (doFlush) {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+            var dataSigBytes = data.sigBytes;
+            var blockSize = this.blockSize;
+            var blockSizeBytes = blockSize * 4;
+
+            // Count blocks ready
+            var nBlocksReady = dataSigBytes / blockSizeBytes;
+            if (doFlush) {
+                // Round up to include partial blocks
+                nBlocksReady = Math.ceil(nBlocksReady);
+            } else {
+                // Round down to include only full blocks,
+                // less the number of blocks that must remain in the buffer
+                nBlocksReady = Math.max((nBlocksReady | 0) - this._minBufferSize, 0);
+            }
+
+            // Count words ready
+            var nWordsReady = nBlocksReady * blockSize;
+
+            // Count bytes ready
+            var nBytesReady = Math.min(nWordsReady * 4, dataSigBytes);
+
+            // Process blocks
+            if (nWordsReady) {
+                for (var offset = 0; offset < nWordsReady; offset += blockSize) {
+                    // Perform concrete-algorithm logic
+                    this._doProcessBlock(dataWords, offset);
+                }
+
+                // Remove processed words
+                var processedWords = dataWords.splice(0, nWordsReady);
+                data.sigBytes -= nBytesReady;
+            }
+
+            // Return processed words
+            return new WordArray.init(processedWords, nBytesReady);
+        },
+
+        /**
+         * Creates a copy of this object.
+         *
+         * @return {Object} The clone.
+         *
+         * @example
+         *
+         *     var clone = bufferedBlockAlgorithm.clone();
+         */
+        clone: function () {
+            var clone = Base.clone.call(this);
+            clone._data = this._data.clone();
+
+            return clone;
+        },
+
+        _minBufferSize: 0
+    });
+
+    /**
+     * Abstract hasher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this hasher operates on. Default: 16 (512 bits)
+     */
+    var Hasher = C_lib.Hasher = BufferedBlockAlgorithm.extend({
+        /**
+         * Configuration options.
+         */
+        cfg: Base.extend(),
+
+        /**
+         * Initializes a newly created hasher.
+         *
+         * @param {Object} cfg (Optional) The configuration options to use for this hash computation.
+         *
+         * @example
+         *
+         *     var hasher = CryptoJS.algo.SHA256.create();
+         */
+        init: function (cfg) {
+            // Apply config defaults
+            this.cfg = this.cfg.extend(cfg);
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this hasher to its initial state.
+         *
+         * @example
+         *
+         *     hasher.reset();
+         */
+        reset: function () {
+            // Reset data buffer
+            BufferedBlockAlgorithm.reset.call(this);
+
+            // Perform concrete-hasher logic
+            this._doReset();
+        },
+
+        /**
+         * Updates this hasher with a message.
+         *
+         * @param {WordArray|string} messageUpdate The message to append.
+         *
+         * @return {Hasher} This hasher.
+         *
+         * @example
+         *
+         *     hasher.update('message');
+         *     hasher.update(wordArray);
+         */
+        update: function (messageUpdate) {
+            // Append
+            this._append(messageUpdate);
+
+            // Update the hash
+            this._process();
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Finalizes the hash computation.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} messageUpdate (Optional) A final message update.
+         *
+         * @return {WordArray} The hash.
+         *
+         * @example
+         *
+         *     var hash = hasher.finalize();
+         *     var hash = hasher.finalize('message');
+         *     var hash = hasher.finalize(wordArray);
+         */
+        finalize: function (messageUpdate) {
+            // Final message update
+            if (messageUpdate) {
+                this._append(messageUpdate);
+            }
+
+            // Perform concrete-hasher logic
+            var hash = this._doFinalize();
+
+            return hash;
+        },
+
+        blockSize: 512/32,
+
+        /**
+         * Creates a shortcut function to a hasher's object interface.
+         *
+         * @param {Hasher} hasher The hasher to create a helper for.
+         *
+         * @return {Function} The shortcut function.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var SHA256 = CryptoJS.lib.Hasher._createHelper(CryptoJS.algo.SHA256);
+         */
+        _createHelper: function (hasher) {
+            return function (message, cfg) {
+                return new hasher.init(cfg).finalize(message);
+            };
+        },
+
+        /**
+         * Creates a shortcut function to the HMAC's object interface.
+         *
+         * @param {Hasher} hasher The hasher to use in this HMAC helper.
+         *
+         * @return {Function} The shortcut function.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var HmacSHA256 = CryptoJS.lib.Hasher._createHmacHelper(CryptoJS.algo.SHA256);
+         */
+        _createHmacHelper: function (hasher) {
+            return function (message, key) {
+                return new C_algo.HMAC.init(hasher, key).finalize(message);
+            };
+        }
+    });
+
+    /**
+     * Algorithm namespace.
+     */
+    var C_algo = C.algo = {};
+
+    return C;
+}(Math));
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var C_enc = C.enc;
+
+    /**
+     * Base64 encoding strategy.
+     */
+    var Base64 = C_enc.Base64 = {
+        /**
+         * Converts a word array to a Base64 string.
+         *
+         * @param {WordArray} wordArray The word array.
+         *
+         * @return {string} The Base64 string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var base64String = CryptoJS.enc.Base64.stringify(wordArray);
+         */
+        stringify: function (wordArray) {
+            // Shortcuts
+            var words = wordArray.words;
+            var sigBytes = wordArray.sigBytes;
+            var map = this._map;
+
+            // Clamp excess bits
+            wordArray.clamp();
+
+            // Convert
+            var base64Chars = [];
+            for (var i = 0; i < sigBytes; i += 3) {
+                var byte1 = (words[i >>> 2]       >>> (24 - (i % 4) * 8))       & 0xff;
+                var byte2 = (words[(i + 1) >>> 2] >>> (24 - ((i + 1) % 4) * 8)) & 0xff;
+                var byte3 = (words[(i + 2) >>> 2] >>> (24 - ((i + 2) % 4) * 8)) & 0xff;
+
+                var triplet = (byte1 << 16) | (byte2 << 8) | byte3;
+
+                for (var j = 0; (j < 4) && (i + j * 0.75 < sigBytes); j++) {
+                    base64Chars.push(map.charAt((triplet >>> (6 * (3 - j))) & 0x3f));
+                }
+            }
+
+            // Add padding
+            var paddingChar = map.charAt(64);
+            if (paddingChar) {
+                while (base64Chars.length % 4) {
+                    base64Chars.push(paddingChar);
+                }
+            }
+
+            return base64Chars.join('');
+        },
+
+        /**
+         * Converts a Base64 string to a word array.
+         *
+         * @param {string} base64Str The Base64 string.
+         *
+         * @return {WordArray} The word array.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var wordArray = CryptoJS.enc.Base64.parse(base64String);
+         */
+        parse: function (base64Str) {
+            // Shortcuts
+            var base64StrLength = base64Str.length;
+            var map = this._map;
+
+            // Ignore padding
+            var paddingChar = map.charAt(64);
+            if (paddingChar) {
+                var paddingIndex = base64Str.indexOf(paddingChar);
+                if (paddingIndex != -1) {
+                    base64StrLength = paddingIndex;
+                }
+            }
+
+            // Convert
+            var words = [];
+            var nBytes = 0;
+            for (var i = 0; i < base64StrLength; i++) {
+                if (i % 4) {
+                    var bits1 = map.indexOf(base64Str.charAt(i - 1)) << ((i % 4) * 2);
+                    var bits2 = map.indexOf(base64Str.charAt(i)) >>> (6 - (i % 4) * 2);
+                    words[nBytes >>> 2] |= (bits1 | bits2) << (24 - (nBytes % 4) * 8);
+                    nBytes++;
+                }
+            }
+
+            return WordArray.create(words, nBytes);
+        },
+
+        _map: 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/='
+    };
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * Cipher core components.
+ */
+CryptoJS.lib.Cipher || (function (undefined) {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var Base = C_lib.Base;
+    var WordArray = C_lib.WordArray;
+    var BufferedBlockAlgorithm = C_lib.BufferedBlockAlgorithm;
+    var C_enc = C.enc;
+    var Utf8 = C_enc.Utf8;
+    var Base64 = C_enc.Base64;
+    var C_algo = C.algo;
+    var EvpKDF = C_algo.EvpKDF;
+
+    /**
+     * Abstract base cipher template.
+     *
+     * @property {number} keySize This cipher's key size. Default: 4 (128 bits)
+     * @property {number} ivSize This cipher's IV size. Default: 4 (128 bits)
+     * @property {number} _ENC_XFORM_MODE A constant representing encryption mode.
+     * @property {number} _DEC_XFORM_MODE A constant representing decryption mode.
+     */
+    var Cipher = C_lib.Cipher = BufferedBlockAlgorithm.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {WordArray} iv The IV to use for this operation.
+         */
+        cfg: Base.extend(),
+
+        /**
+         * Creates this cipher in encryption mode.
+         *
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {Cipher} A cipher instance.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.createEncryptor(keyWordArray, { iv: ivWordArray });
+         */
+        createEncryptor: function (key, cfg) {
+            return this.create(this._ENC_XFORM_MODE, key, cfg);
+        },
+
+        /**
+         * Creates this cipher in decryption mode.
+         *
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {Cipher} A cipher instance.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.createDecryptor(keyWordArray, { iv: ivWordArray });
+         */
+        createDecryptor: function (key, cfg) {
+            return this.create(this._DEC_XFORM_MODE, key, cfg);
+        },
+
+        /**
+         * Initializes a newly created cipher.
+         *
+         * @param {number} xformMode Either the encryption or decryption transormation mode constant.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @example
+         *
+         *     var cipher = CryptoJS.algo.AES.create(CryptoJS.algo.AES._ENC_XFORM_MODE, keyWordArray, { iv: ivWordArray });
+         */
+        init: function (xformMode, key, cfg) {
+            // Apply config defaults
+            this.cfg = this.cfg.extend(cfg);
+
+            // Store transform mode and key
+            this._xformMode = xformMode;
+            this._key = key;
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this cipher to its initial state.
+         *
+         * @example
+         *
+         *     cipher.reset();
+         */
+        reset: function () {
+            // Reset data buffer
+            BufferedBlockAlgorithm.reset.call(this);
+
+            // Perform concrete-cipher logic
+            this._doReset();
+        },
+
+        /**
+         * Adds data to be encrypted or decrypted.
+         *
+         * @param {WordArray|string} dataUpdate The data to encrypt or decrypt.
+         *
+         * @return {WordArray} The data after processing.
+         *
+         * @example
+         *
+         *     var encrypted = cipher.process('data');
+         *     var encrypted = cipher.process(wordArray);
+         */
+        process: function (dataUpdate) {
+            // Append
+            this._append(dataUpdate);
+
+            // Process available blocks
+            return this._process();
+        },
+
+        /**
+         * Finalizes the encryption or decryption process.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} dataUpdate The final data to encrypt or decrypt.
+         *
+         * @return {WordArray} The data after final processing.
+         *
+         * @example
+         *
+         *     var encrypted = cipher.finalize();
+         *     var encrypted = cipher.finalize('data');
+         *     var encrypted = cipher.finalize(wordArray);
+         */
+        finalize: function (dataUpdate) {
+            // Final data update
+            if (dataUpdate) {
+                this._append(dataUpdate);
+            }
+
+            // Perform concrete-cipher logic
+            var finalProcessedData = this._doFinalize();
+
+            return finalProcessedData;
+        },
+
+        keySize: 128/32,
+
+        ivSize: 128/32,
+
+        _ENC_XFORM_MODE: 1,
+
+        _DEC_XFORM_MODE: 2,
+
+        /**
+         * Creates shortcut functions to a cipher's object interface.
+         *
+         * @param {Cipher} cipher The cipher to create a helper for.
+         *
+         * @return {Object} An object with encrypt and decrypt shortcut functions.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var AES = CryptoJS.lib.Cipher._createHelper(CryptoJS.algo.AES);
+         */
+        _createHelper: (function () {
+            function selectCipherStrategy(key) {
+                if (typeof key == 'string') {
+                    return PasswordBasedCipher;
+                } else {
+                    return SerializableCipher;
+                }
+            }
+
+            return function (cipher) {
+                return {
+                    encrypt: function (message, key, cfg) {
+                        return selectCipherStrategy(key).encrypt(cipher, message, key, cfg);
+                    },
+
+                    decrypt: function (ciphertext, key, cfg) {
+                        return selectCipherStrategy(key).decrypt(cipher, ciphertext, key, cfg);
+                    }
+                };
+            };
+        }())
+    });
+
+    /**
+     * Abstract base stream cipher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 1 (32 bits)
+     */
+    var StreamCipher = C_lib.StreamCipher = Cipher.extend({
+        _doFinalize: function () {
+            // Process partial blocks
+            var finalProcessedBlocks = this._process(!!'flush');
+
+            return finalProcessedBlocks;
+        },
+
+        blockSize: 1
+    });
+
+    /**
+     * Mode namespace.
+     */
+    var C_mode = C.mode = {};
+
+    /**
+     * Abstract base block cipher mode template.
+     */
+    var BlockCipherMode = C_lib.BlockCipherMode = Base.extend({
+        /**
+         * Creates this mode for encryption.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.createEncryptor(cipher, iv.words);
+         */
+        createEncryptor: function (cipher, iv) {
+            return this.Encryptor.create(cipher, iv);
+        },
+
+        /**
+         * Creates this mode for decryption.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.createDecryptor(cipher, iv.words);
+         */
+        createDecryptor: function (cipher, iv) {
+            return this.Decryptor.create(cipher, iv);
+        },
+
+        /**
+         * Initializes a newly created mode.
+         *
+         * @param {Cipher} cipher A block cipher instance.
+         * @param {Array} iv The IV words.
+         *
+         * @example
+         *
+         *     var mode = CryptoJS.mode.CBC.Encryptor.create(cipher, iv.words);
+         */
+        init: function (cipher, iv) {
+            this._cipher = cipher;
+            this._iv = iv;
+        }
+    });
+
+    /**
+     * Cipher Block Chaining mode.
+     */
+    var CBC = C_mode.CBC = (function () {
+        /**
+         * Abstract base CBC mode.
+         */
+        var CBC = BlockCipherMode.extend();
+
+        /**
+         * CBC encryptor.
+         */
+        CBC.Encryptor = CBC.extend({
+            /**
+             * Processes the data block at offset.
+             *
+             * @param {Array} words The data words to operate on.
+             * @param {number} offset The offset where the block starts.
+             *
+             * @example
+             *
+             *     mode.processBlock(data.words, offset);
+             */
+            processBlock: function (words, offset) {
+                // Shortcuts
+                var cipher = this._cipher;
+                var blockSize = cipher.blockSize;
+
+                // XOR and encrypt
+                xorBlock.call(this, words, offset, blockSize);
+                cipher.encryptBlock(words, offset);
+
+                // Remember this block to use with next block
+                this._prevBlock = words.slice(offset, offset + blockSize);
+            }
+        });
+
+        /**
+         * CBC decryptor.
+         */
+        CBC.Decryptor = CBC.extend({
+            /**
+             * Processes the data block at offset.
+             *
+             * @param {Array} words The data words to operate on.
+             * @param {number} offset The offset where the block starts.
+             *
+             * @example
+             *
+             *     mode.processBlock(data.words, offset);
+             */
+            processBlock: function (words, offset) {
+                // Shortcuts
+                var cipher = this._cipher;
+                var blockSize = cipher.blockSize;
+
+                // Remember this block to use with next block
+                var thisBlock = words.slice(offset, offset + blockSize);
+
+                // Decrypt and XOR
+                cipher.decryptBlock(words, offset);
+                xorBlock.call(this, words, offset, blockSize);
+
+                // This block becomes the previous block
+                this._prevBlock = thisBlock;
+            }
+        });
+
+        function xorBlock(words, offset, blockSize) {
+            // Shortcut
+            var iv = this._iv;
+
+            // Choose mixing block
+            if (iv) {
+                var block = iv;
+
+                // Remove IV for subsequent blocks
+                this._iv = undefined;
+            } else {
+                var block = this._prevBlock;
+            }
+
+            // XOR blocks
+            for (var i = 0; i < blockSize; i++) {
+                words[offset + i] ^= block[i];
+            }
+        }
+
+        return CBC;
+    }());
+
+    /**
+     * Padding namespace.
+     */
+    var C_pad = C.pad = {};
+
+    /**
+     * PKCS #5/7 padding strategy.
+     */
+    var Pkcs7 = C_pad.Pkcs7 = {
+        /**
+         * Pads data using the algorithm defined in PKCS #5/7.
+         *
+         * @param {WordArray} data The data to pad.
+         * @param {number} blockSize The multiple that the data should be padded to.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     CryptoJS.pad.Pkcs7.pad(wordArray, 4);
+         */
+        pad: function (data, blockSize) {
+            // Shortcut
+            var blockSizeBytes = blockSize * 4;
+
+            // Count padding bytes
+            var nPaddingBytes = blockSizeBytes - data.sigBytes % blockSizeBytes;
+
+            // Create padding word
+            var paddingWord = (nPaddingBytes << 24) | (nPaddingBytes << 16) | (nPaddingBytes << 8) | nPaddingBytes;
+
+            // Create padding
+            var paddingWords = [];
+            for (var i = 0; i < nPaddingBytes; i += 4) {
+                paddingWords.push(paddingWord);
+            }
+            var padding = WordArray.create(paddingWords, nPaddingBytes);
+
+            // Add padding
+            data.concat(padding);
+        },
+
+        /**
+         * Unpads data that had been padded using the algorithm defined in PKCS #5/7.
+         *
+         * @param {WordArray} data The data to unpad.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     CryptoJS.pad.Pkcs7.unpad(wordArray);
+         */
+        unpad: function (data) {
+            // Get number of padding bytes from last byte
+            var nPaddingBytes = data.words[(data.sigBytes - 1) >>> 2] & 0xff;
+
+            // Remove padding
+            data.sigBytes -= nPaddingBytes;
+        }
+    };
+
+    /**
+     * Abstract base block cipher template.
+     *
+     * @property {number} blockSize The number of 32-bit words this cipher operates on. Default: 4 (128 bits)
+     */
+    var BlockCipher = C_lib.BlockCipher = Cipher.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {Mode} mode The block mode to use. Default: CBC
+         * @property {Padding} padding The padding strategy to use. Default: Pkcs7
+         */
+        cfg: Cipher.cfg.extend({
+            mode: CBC,
+            padding: Pkcs7
+        }),
+
+        reset: function () {
+            // Reset cipher
+            Cipher.reset.call(this);
+
+            // Shortcuts
+            var cfg = this.cfg;
+            var iv = cfg.iv;
+            var mode = cfg.mode;
+
+            // Reset block mode
+            if (this._xformMode == this._ENC_XFORM_MODE) {
+                var modeCreator = mode.createEncryptor;
+            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
+                var modeCreator = mode.createDecryptor;
+
+                // Keep at least one block in the buffer for unpadding
+                this._minBufferSize = 1;
+            }
+            this._mode = modeCreator.call(mode, this, iv && iv.words);
+        },
+
+        _doProcessBlock: function (words, offset) {
+            this._mode.processBlock(words, offset);
+        },
+
+        _doFinalize: function () {
+            // Shortcut
+            var padding = this.cfg.padding;
+
+            // Finalize
+            if (this._xformMode == this._ENC_XFORM_MODE) {
+                // Pad data
+                padding.pad(this._data, this.blockSize);
+
+                // Process final blocks
+                var finalProcessedBlocks = this._process(!!'flush');
+            } else /* if (this._xformMode == this._DEC_XFORM_MODE) */ {
+                // Process final blocks
+                var finalProcessedBlocks = this._process(!!'flush');
+
+                // Unpad data
+                padding.unpad(finalProcessedBlocks);
+            }
+
+            return finalProcessedBlocks;
+        },
+
+        blockSize: 128/32
+    });
+
+    /**
+     * A collection of cipher parameters.
+     *
+     * @property {WordArray} ciphertext The raw ciphertext.
+     * @property {WordArray} key The key to this ciphertext.
+     * @property {WordArray} iv The IV used in the ciphering operation.
+     * @property {WordArray} salt The salt used with a key derivation function.
+     * @property {Cipher} algorithm The cipher algorithm.
+     * @property {Mode} mode The block mode used in the ciphering operation.
+     * @property {Padding} padding The padding scheme used in the ciphering operation.
+     * @property {number} blockSize The block size of the cipher.
+     * @property {Format} formatter The default formatting strategy to convert this cipher params object to a string.
+     */
+    var CipherParams = C_lib.CipherParams = Base.extend({
+        /**
+         * Initializes a newly created cipher params object.
+         *
+         * @param {Object} cipherParams An object with any of the possible cipher parameters.
+         *
+         * @example
+         *
+         *     var cipherParams = CryptoJS.lib.CipherParams.create({
+         *         ciphertext: ciphertextWordArray,
+         *         key: keyWordArray,
+         *         iv: ivWordArray,
+         *         salt: saltWordArray,
+         *         algorithm: CryptoJS.algo.AES,
+         *         mode: CryptoJS.mode.CBC,
+         *         padding: CryptoJS.pad.PKCS7,
+         *         blockSize: 4,
+         *         formatter: CryptoJS.format.OpenSSL
+         *     });
+         */
+        init: function (cipherParams) {
+            this.mixIn(cipherParams);
+        },
+
+        /**
+         * Converts this cipher params object to a string.
+         *
+         * @param {Format} formatter (Optional) The formatting strategy to use.
+         *
+         * @return {string} The stringified cipher params.
+         *
+         * @throws Error If neither the formatter nor the default formatter is set.
+         *
+         * @example
+         *
+         *     var string = cipherParams + '';
+         *     var string = cipherParams.toString();
+         *     var string = cipherParams.toString(CryptoJS.format.OpenSSL);
+         */
+        toString: function (formatter) {
+            return (formatter || this.formatter).stringify(this);
+        }
+    });
+
+    /**
+     * Format namespace.
+     */
+    var C_format = C.format = {};
+
+    /**
+     * OpenSSL formatting strategy.
+     */
+    var OpenSSLFormatter = C_format.OpenSSL = {
+        /**
+         * Converts a cipher params object to an OpenSSL-compatible string.
+         *
+         * @param {CipherParams} cipherParams The cipher params object.
+         *
+         * @return {string} The OpenSSL-compatible string.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var openSSLString = CryptoJS.format.OpenSSL.stringify(cipherParams);
+         */
+        stringify: function (cipherParams) {
+            // Shortcuts
+            var ciphertext = cipherParams.ciphertext;
+            var salt = cipherParams.salt;
+
+            // Format
+            if (salt) {
+                var wordArray = WordArray.create([0x53616c74, 0x65645f5f]).concat(salt).concat(ciphertext);
+            } else {
+                var wordArray = ciphertext;
+            }
+
+            return wordArray.toString(Base64);
+        },
+
+        /**
+         * Converts an OpenSSL-compatible string to a cipher params object.
+         *
+         * @param {string} openSSLStr The OpenSSL-compatible string.
+         *
+         * @return {CipherParams} The cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var cipherParams = CryptoJS.format.OpenSSL.parse(openSSLString);
+         */
+        parse: function (openSSLStr) {
+            // Parse base64
+            var ciphertext = Base64.parse(openSSLStr);
+
+            // Shortcut
+            var ciphertextWords = ciphertext.words;
+
+            // Test for salt
+            if (ciphertextWords[0] == 0x53616c74 && ciphertextWords[1] == 0x65645f5f) {
+                // Extract salt
+                var salt = WordArray.create(ciphertextWords.slice(2, 4));
+
+                // Remove salt from ciphertext
+                ciphertextWords.splice(0, 4);
+                ciphertext.sigBytes -= 16;
+            }
+
+            return CipherParams.create({ ciphertext: ciphertext, salt: salt });
+        }
+    };
+
+    /**
+     * A cipher wrapper that returns ciphertext as a serializable cipher params object.
+     */
+    var SerializableCipher = C_lib.SerializableCipher = Base.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {Formatter} format The formatting strategy to convert cipher param objects to and from a string. Default: OpenSSL
+         */
+        cfg: Base.extend({
+            format: OpenSSLFormatter
+        }),
+
+        /**
+         * Encrypts a message.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {WordArray|string} message The message to encrypt.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {CipherParams} A cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key);
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv });
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher.encrypt(CryptoJS.algo.AES, message, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         */
+        encrypt: function (cipher, message, key, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Encrypt
+            var encryptor = cipher.createEncryptor(key, cfg);
+            var ciphertext = encryptor.finalize(message);
+
+            // Shortcut
+            var cipherCfg = encryptor.cfg;
+
+            // Create and return serializable cipher params
+            return CipherParams.create({
+                ciphertext: ciphertext,
+                key: key,
+                iv: cipherCfg.iv,
+                algorithm: cipher,
+                mode: cipherCfg.mode,
+                padding: cipherCfg.padding,
+                blockSize: cipher.blockSize,
+                formatter: cfg.format
+            });
+        },
+
+        /**
+         * Decrypts serialized ciphertext.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
+         * @param {WordArray} key The key.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {WordArray} The plaintext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         *     var plaintext = CryptoJS.lib.SerializableCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, key, { iv: iv, format: CryptoJS.format.OpenSSL });
+         */
+        decrypt: function (cipher, ciphertext, key, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Convert string to CipherParams
+            ciphertext = this._parse(ciphertext, cfg.format);
+
+            // Decrypt
+            var plaintext = cipher.createDecryptor(key, cfg).finalize(ciphertext.ciphertext);
+
+            return plaintext;
+        },
+
+        /**
+         * Converts serialized ciphertext to CipherParams,
+         * else assumed CipherParams already and returns ciphertext unchanged.
+         *
+         * @param {CipherParams|string} ciphertext The ciphertext.
+         * @param {Formatter} format The formatting strategy to use to parse serialized ciphertext.
+         *
+         * @return {CipherParams} The unserialized ciphertext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.SerializableCipher._parse(ciphertextStringOrParams, format);
+         */
+        _parse: function (ciphertext, format) {
+            if (typeof ciphertext == 'string') {
+                return format.parse(ciphertext, this);
+            } else {
+                return ciphertext;
+            }
+        }
+    });
+
+    /**
+     * Key derivation function namespace.
+     */
+    var C_kdf = C.kdf = {};
+
+    /**
+     * OpenSSL key derivation function.
+     */
+    var OpenSSLKdf = C_kdf.OpenSSL = {
+        /**
+         * Derives a key and IV from a password.
+         *
+         * @param {string} password The password to derive from.
+         * @param {number} keySize The size in words of the key to generate.
+         * @param {number} ivSize The size in words of the IV to generate.
+         * @param {WordArray|string} salt (Optional) A 64-bit salt to use. If omitted, a salt will be generated randomly.
+         *
+         * @return {CipherParams} A cipher params object with the key, IV, and salt.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32);
+         *     var derivedParams = CryptoJS.kdf.OpenSSL.execute('Password', 256/32, 128/32, 'saltsalt');
+         */
+        execute: function (password, keySize, ivSize, salt) {
+            // Generate random salt
+            if (!salt) {
+                salt = WordArray.random(64/8);
+            }
+
+            // Derive key and IV
+            var key = EvpKDF.create({ keySize: keySize + ivSize }).compute(password, salt);
+
+            // Separate key and IV
+            var iv = WordArray.create(key.words.slice(keySize), ivSize * 4);
+            key.sigBytes = keySize * 4;
+
+            // Return params
+            return CipherParams.create({ key: key, iv: iv, salt: salt });
+        }
+    };
+
+    /**
+     * A serializable cipher wrapper that derives the key from a password,
+     * and returns ciphertext as a serializable cipher params object.
+     */
+    var PasswordBasedCipher = C_lib.PasswordBasedCipher = SerializableCipher.extend({
+        /**
+         * Configuration options.
+         *
+         * @property {KDF} kdf The key derivation function to use to generate a key and IV from a password. Default: OpenSSL
+         */
+        cfg: SerializableCipher.cfg.extend({
+            kdf: OpenSSLKdf
+        }),
+
+        /**
+         * Encrypts a message using a password.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {WordArray|string} message The message to encrypt.
+         * @param {string} password The password.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {CipherParams} A cipher params object.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password');
+         *     var ciphertextParams = CryptoJS.lib.PasswordBasedCipher.encrypt(CryptoJS.algo.AES, message, 'password', { format: CryptoJS.format.OpenSSL });
+         */
+        encrypt: function (cipher, message, password, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Derive key and other params
+            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize);
+
+            // Add IV to config
+            cfg.iv = derivedParams.iv;
+
+            // Encrypt
+            var ciphertext = SerializableCipher.encrypt.call(this, cipher, message, derivedParams.key, cfg);
+
+            // Mix in derived params
+            ciphertext.mixIn(derivedParams);
+
+            return ciphertext;
+        },
+
+        /**
+         * Decrypts serialized ciphertext using a password.
+         *
+         * @param {Cipher} cipher The cipher algorithm to use.
+         * @param {CipherParams|string} ciphertext The ciphertext to decrypt.
+         * @param {string} password The password.
+         * @param {Object} cfg (Optional) The configuration options to use for this operation.
+         *
+         * @return {WordArray} The plaintext.
+         *
+         * @static
+         *
+         * @example
+         *
+         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, formattedCiphertext, 'password', { format: CryptoJS.format.OpenSSL });
+         *     var plaintext = CryptoJS.lib.PasswordBasedCipher.decrypt(CryptoJS.algo.AES, ciphertextParams, 'password', { format: CryptoJS.format.OpenSSL });
+         */
+        decrypt: function (cipher, ciphertext, password, cfg) {
+            // Apply config defaults
+            cfg = this.cfg.extend(cfg);
+
+            // Convert string to CipherParams
+            ciphertext = this._parse(ciphertext, cfg.format);
+
+            // Derive key and other params
+            var derivedParams = cfg.kdf.execute(password, cipher.keySize, cipher.ivSize, ciphertext.salt);
+
+            // Add IV to config
+            cfg.iv = derivedParams.iv;
+
+            // Decrypt
+            var plaintext = SerializableCipher.decrypt.call(this, cipher, ciphertext, derivedParams.key, cfg);
+
+            return plaintext;
+        }
+    });
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var BlockCipher = C_lib.BlockCipher;
+    var C_algo = C.algo;
+
+    // Lookup tables
+    var SBOX = [];
+    var INV_SBOX = [];
+    var SUB_MIX_0 = [];
+    var SUB_MIX_1 = [];
+    var SUB_MIX_2 = [];
+    var SUB_MIX_3 = [];
+    var INV_SUB_MIX_0 = [];
+    var INV_SUB_MIX_1 = [];
+    var INV_SUB_MIX_2 = [];
+    var INV_SUB_MIX_3 = [];
+
+    // Compute lookup tables
+    (function () {
+        // Compute double table
+        var d = [];
+        for (var i = 0; i < 256; i++) {
+            if (i < 128) {
+                d[i] = i << 1;
+            } else {
+                d[i] = (i << 1) ^ 0x11b;
+            }
+        }
+
+        // Walk GF(2^8)
+        var x = 0;
+        var xi = 0;
+        for (var i = 0; i < 256; i++) {
+            // Compute sbox
+            var sx = xi ^ (xi << 1) ^ (xi << 2) ^ (xi << 3) ^ (xi << 4);
+            sx = (sx >>> 8) ^ (sx & 0xff) ^ 0x63;
+            SBOX[x] = sx;
+            INV_SBOX[sx] = x;
+
+            // Compute multiplication
+            var x2 = d[x];
+            var x4 = d[x2];
+            var x8 = d[x4];
+
+            // Compute sub bytes, mix columns tables
+            var t = (d[sx] * 0x101) ^ (sx * 0x1010100);
+            SUB_MIX_0[x] = (t << 24) | (t >>> 8);
+            SUB_MIX_1[x] = (t << 16) | (t >>> 16);
+            SUB_MIX_2[x] = (t << 8)  | (t >>> 24);
+            SUB_MIX_3[x] = t;
+
+            // Compute inv sub bytes, inv mix columns tables
+            var t = (x8 * 0x1010101) ^ (x4 * 0x10001) ^ (x2 * 0x101) ^ (x * 0x1010100);
+            INV_SUB_MIX_0[sx] = (t << 24) | (t >>> 8);
+            INV_SUB_MIX_1[sx] = (t << 16) | (t >>> 16);
+            INV_SUB_MIX_2[sx] = (t << 8)  | (t >>> 24);
+            INV_SUB_MIX_3[sx] = t;
+
+            // Compute next counter
+            if (!x) {
+                x = xi = 1;
+            } else {
+                x = x2 ^ d[d[d[x8 ^ x2]]];
+                xi ^= d[d[xi]];
+            }
+        }
+    }());
+
+    // Precomputed Rcon lookup
+    var RCON = [0x00, 0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36];
+
+    /**
+     * AES block cipher algorithm.
+     */
+    var AES = C_algo.AES = BlockCipher.extend({
+        _doReset: function () {
+            // Shortcuts
+            var key = this._key;
+            var keyWords = key.words;
+            var keySize = key.sigBytes / 4;
+
+            // Compute number of rounds
+            var nRounds = this._nRounds = keySize + 6
+
+            // Compute number of key schedule rows
+            var ksRows = (nRounds + 1) * 4;
+
+            // Compute key schedule
+            var keySchedule = this._keySchedule = [];
+            for (var ksRow = 0; ksRow < ksRows; ksRow++) {
+                if (ksRow < keySize) {
+                    keySchedule[ksRow] = keyWords[ksRow];
+                } else {
+                    var t = keySchedule[ksRow - 1];
+
+                    if (!(ksRow % keySize)) {
+                        // Rot word
+                        t = (t << 8) | (t >>> 24);
+
+                        // Sub word
+                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
+
+                        // Mix Rcon
+                        t ^= RCON[(ksRow / keySize) | 0] << 24;
+                    } else if (keySize > 6 && ksRow % keySize == 4) {
+                        // Sub word
+                        t = (SBOX[t >>> 24] << 24) | (SBOX[(t >>> 16) & 0xff] << 16) | (SBOX[(t >>> 8) & 0xff] << 8) | SBOX[t & 0xff];
+                    }
+
+                    keySchedule[ksRow] = keySchedule[ksRow - keySize] ^ t;
+                }
+            }
+
+            // Compute inv key schedule
+            var invKeySchedule = this._invKeySchedule = [];
+            for (var invKsRow = 0; invKsRow < ksRows; invKsRow++) {
+                var ksRow = ksRows - invKsRow;
+
+                if (invKsRow % 4) {
+                    var t = keySchedule[ksRow];
+                } else {
+                    var t = keySchedule[ksRow - 4];
+                }
+
+                if (invKsRow < 4 || ksRow <= 4) {
+                    invKeySchedule[invKsRow] = t;
+                } else {
+                    invKeySchedule[invKsRow] = INV_SUB_MIX_0[SBOX[t >>> 24]] ^ INV_SUB_MIX_1[SBOX[(t >>> 16) & 0xff]] ^
+                                               INV_SUB_MIX_2[SBOX[(t >>> 8) & 0xff]] ^ INV_SUB_MIX_3[SBOX[t & 0xff]];
+                }
+            }
+        },
+
+        encryptBlock: function (M, offset) {
+            this._doCryptBlock(M, offset, this._keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX);
+        },
+
+        decryptBlock: function (M, offset) {
+            // Swap 2nd and 4th rows
+            var t = M[offset + 1];
+            M[offset + 1] = M[offset + 3];
+            M[offset + 3] = t;
+
+            this._doCryptBlock(M, offset, this._invKeySchedule, INV_SUB_MIX_0, INV_SUB_MIX_1, INV_SUB_MIX_2, INV_SUB_MIX_3, INV_SBOX);
+
+            // Inv swap 2nd and 4th rows
+            var t = M[offset + 1];
+            M[offset + 1] = M[offset + 3];
+            M[offset + 3] = t;
+        },
+
+        _doCryptBlock: function (M, offset, keySchedule, SUB_MIX_0, SUB_MIX_1, SUB_MIX_2, SUB_MIX_3, SBOX) {
+            // Shortcut
+            var nRounds = this._nRounds;
+
+            // Get input, add round key
+            var s0 = M[offset]     ^ keySchedule[0];
+            var s1 = M[offset + 1] ^ keySchedule[1];
+            var s2 = M[offset + 2] ^ keySchedule[2];
+            var s3 = M[offset + 3] ^ keySchedule[3];
+
+            // Key schedule row counter
+            var ksRow = 4;
+
+            // Rounds
+            for (var round = 1; round < nRounds; round++) {
+                // Shift rows, sub bytes, mix columns, add round key
+                var t0 = SUB_MIX_0[s0 >>> 24] ^ SUB_MIX_1[(s1 >>> 16) & 0xff] ^ SUB_MIX_2[(s2 >>> 8) & 0xff] ^ SUB_MIX_3[s3 & 0xff] ^ keySchedule[ksRow++];
+                var t1 = SUB_MIX_0[s1 >>> 24] ^ SUB_MIX_1[(s2 >>> 16) & 0xff] ^ SUB_MIX_2[(s3 >>> 8) & 0xff] ^ SUB_MIX_3[s0 & 0xff] ^ keySchedule[ksRow++];
+                var t2 = SUB_MIX_0[s2 >>> 24] ^ SUB_MIX_1[(s3 >>> 16) & 0xff] ^ SUB_MIX_2[(s0 >>> 8) & 0xff] ^ SUB_MIX_3[s1 & 0xff] ^ keySchedule[ksRow++];
+                var t3 = SUB_MIX_0[s3 >>> 24] ^ SUB_MIX_1[(s0 >>> 16) & 0xff] ^ SUB_MIX_2[(s1 >>> 8) & 0xff] ^ SUB_MIX_3[s2 & 0xff] ^ keySchedule[ksRow++];
+
+                // Update state
+                s0 = t0;
+                s1 = t1;
+                s2 = t2;
+                s3 = t3;
+            }
+
+            // Shift rows, sub bytes, add round key
+            var t0 = ((SBOX[s0 >>> 24] << 24) | (SBOX[(s1 >>> 16) & 0xff] << 16) | (SBOX[(s2 >>> 8) & 0xff] << 8) | SBOX[s3 & 0xff]) ^ keySchedule[ksRow++];
+            var t1 = ((SBOX[s1 >>> 24] << 24) | (SBOX[(s2 >>> 16) & 0xff] << 16) | (SBOX[(s3 >>> 8) & 0xff] << 8) | SBOX[s0 & 0xff]) ^ keySchedule[ksRow++];
+            var t2 = ((SBOX[s2 >>> 24] << 24) | (SBOX[(s3 >>> 16) & 0xff] << 16) | (SBOX[(s0 >>> 8) & 0xff] << 8) | SBOX[s1 & 0xff]) ^ keySchedule[ksRow++];
+            var t3 = ((SBOX[s3 >>> 24] << 24) | (SBOX[(s0 >>> 16) & 0xff] << 16) | (SBOX[(s1 >>> 8) & 0xff] << 8) | SBOX[s2 & 0xff]) ^ keySchedule[ksRow++];
+
+            // Set output
+            M[offset]     = t0;
+            M[offset + 1] = t1;
+            M[offset + 2] = t2;
+            M[offset + 3] = t3;
+        },
+
+        keySize: 256/32
+    });
+
+    /**
+     * Shortcut functions to the cipher's object interface.
+     *
+     * @example
+     *
+     *     var ciphertext = CryptoJS.AES.encrypt(message, key, cfg);
+     *     var plaintext  = CryptoJS.AES.decrypt(ciphertext, key, cfg);
+     */
+    C.AES = BlockCipher._createHelper(AES);
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var Hasher = C_lib.Hasher;
+    var C_algo = C.algo;
+
+    // Reusable object
+    var W = [];
+
+    /**
+     * SHA-1 hash algorithm.
+     */
+    var SHA1 = C_algo.SHA1 = Hasher.extend({
+        _doReset: function () {
+            this._hash = new WordArray.init([
+                0x67452301, 0xefcdab89,
+                0x98badcfe, 0x10325476,
+                0xc3d2e1f0
+            ]);
+        },
+
+        _doProcessBlock: function (M, offset) {
+            // Shortcut
+            var H = this._hash.words;
+
+            // Working variables
+            var a = H[0];
+            var b = H[1];
+            var c = H[2];
+            var d = H[3];
+            var e = H[4];
+
+            // Computation
+            for (var i = 0; i < 80; i++) {
+                if (i < 16) {
+                    W[i] = M[offset + i] | 0;
+                } else {
+                    var n = W[i - 3] ^ W[i - 8] ^ W[i - 14] ^ W[i - 16];
+                    W[i] = (n << 1) | (n >>> 31);
+                }
+
+                var t = ((a << 5) | (a >>> 27)) + e + W[i];
+                if (i < 20) {
+                    t += ((b & c) | (~b & d)) + 0x5a827999;
+                } else if (i < 40) {
+                    t += (b ^ c ^ d) + 0x6ed9eba1;
+                } else if (i < 60) {
+                    t += ((b & c) | (b & d) | (c & d)) - 0x70e44324;
+                } else /* if (i < 80) */ {
+                    t += (b ^ c ^ d) - 0x359d3e2a;
+                }
+
+                e = d;
+                d = c;
+                c = (b << 30) | (b >>> 2);
+                b = a;
+                a = t;
+            }
+
+            // Intermediate hash value
+            H[0] = (H[0] + a) | 0;
+            H[1] = (H[1] + b) | 0;
+            H[2] = (H[2] + c) | 0;
+            H[3] = (H[3] + d) | 0;
+            H[4] = (H[4] + e) | 0;
+        },
+
+        _doFinalize: function () {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+
+            var nBitsTotal = this._nDataBytes * 8;
+            var nBitsLeft = data.sigBytes * 8;
+
+            // Add padding
+            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
+            data.sigBytes = dataWords.length * 4;
+
+            // Hash final blocks
+            this._process();
+
+            // Return final computed hash
+            return this._hash;
+        },
+
+        clone: function () {
+            var clone = Hasher.clone.call(this);
+            clone._hash = this._hash.clone();
+
+            return clone;
+        }
+    });
+
+    /**
+     * Shortcut function to the hasher's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     *
+     * @return {WordArray} The hash.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hash = CryptoJS.SHA1('message');
+     *     var hash = CryptoJS.SHA1(wordArray);
+     */
+    C.SHA1 = Hasher._createHelper(SHA1);
+
+    /**
+     * Shortcut function to the HMAC's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     * @param {WordArray|string} key The secret key.
+     *
+     * @return {WordArray} The HMAC.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hmac = CryptoJS.HmacSHA1(message, key);
+     */
+    C.HmacSHA1 = Hasher._createHmacHelper(SHA1);
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function (Math) {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var WordArray = C_lib.WordArray;
+    var Hasher = C_lib.Hasher;
+    var C_algo = C.algo;
+
+    // Initialization and round constants tables
+    var H = [];
+    var K = [];
+
+    // Compute constants
+    (function () {
+        function isPrime(n) {
+            var sqrtN = Math.sqrt(n);
+            for (var factor = 2; factor <= sqrtN; factor++) {
+                if (!(n % factor)) {
+                    return false;
+                }
+            }
+
+            return true;
+        }
+
+        function getFractionalBits(n) {
+            return ((n - (n | 0)) * 0x100000000) | 0;
+        }
+
+        var n = 2;
+        var nPrime = 0;
+        while (nPrime < 64) {
+            if (isPrime(n)) {
+                if (nPrime < 8) {
+                    H[nPrime] = getFractionalBits(Math.pow(n, 1 / 2));
+                }
+                K[nPrime] = getFractionalBits(Math.pow(n, 1 / 3));
+
+                nPrime++;
+            }
+
+            n++;
+        }
+    }());
+
+    // Reusable object
+    var W = [];
+
+    /**
+     * SHA-256 hash algorithm.
+     */
+    var SHA256 = C_algo.SHA256 = Hasher.extend({
+        _doReset: function () {
+            this._hash = new WordArray.init(H.slice(0));
+        },
+
+        _doProcessBlock: function (M, offset) {
+            // Shortcut
+            var H = this._hash.words;
+
+            // Working variables
+            var a = H[0];
+            var b = H[1];
+            var c = H[2];
+            var d = H[3];
+            var e = H[4];
+            var f = H[5];
+            var g = H[6];
+            var h = H[7];
+
+            // Computation
+            for (var i = 0; i < 64; i++) {
+                if (i < 16) {
+                    W[i] = M[offset + i] | 0;
+                } else {
+                    var gamma0x = W[i - 15];
+                    var gamma0  = ((gamma0x << 25) | (gamma0x >>> 7))  ^
+                                  ((gamma0x << 14) | (gamma0x >>> 18)) ^
+                                   (gamma0x >>> 3);
+
+                    var gamma1x = W[i - 2];
+                    var gamma1  = ((gamma1x << 15) | (gamma1x >>> 17)) ^
+                                  ((gamma1x << 13) | (gamma1x >>> 19)) ^
+                                   (gamma1x >>> 10);
+
+                    W[i] = gamma0 + W[i - 7] + gamma1 + W[i - 16];
+                }
+
+                var ch  = (e & f) ^ (~e & g);
+                var maj = (a & b) ^ (a & c) ^ (b & c);
+
+                var sigma0 = ((a << 30) | (a >>> 2)) ^ ((a << 19) | (a >>> 13)) ^ ((a << 10) | (a >>> 22));
+                var sigma1 = ((e << 26) | (e >>> 6)) ^ ((e << 21) | (e >>> 11)) ^ ((e << 7)  | (e >>> 25));
+
+                var t1 = h + sigma1 + ch + K[i] + W[i];
+                var t2 = sigma0 + maj;
+
+                h = g;
+                g = f;
+                f = e;
+                e = (d + t1) | 0;
+                d = c;
+                c = b;
+                b = a;
+                a = (t1 + t2) | 0;
+            }
+
+            // Intermediate hash value
+            H[0] = (H[0] + a) | 0;
+            H[1] = (H[1] + b) | 0;
+            H[2] = (H[2] + c) | 0;
+            H[3] = (H[3] + d) | 0;
+            H[4] = (H[4] + e) | 0;
+            H[5] = (H[5] + f) | 0;
+            H[6] = (H[6] + g) | 0;
+            H[7] = (H[7] + h) | 0;
+        },
+
+        _doFinalize: function () {
+            // Shortcuts
+            var data = this._data;
+            var dataWords = data.words;
+
+            var nBitsTotal = this._nDataBytes * 8;
+            var nBitsLeft = data.sigBytes * 8;
+
+            // Add padding
+            dataWords[nBitsLeft >>> 5] |= 0x80 << (24 - nBitsLeft % 32);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 14] = Math.floor(nBitsTotal / 0x100000000);
+            dataWords[(((nBitsLeft + 64) >>> 9) << 4) + 15] = nBitsTotal;
+            data.sigBytes = dataWords.length * 4;
+
+            // Hash final blocks
+            this._process();
+
+            // Return final computed hash
+            return this._hash;
+        },
+
+        clone: function () {
+            var clone = Hasher.clone.call(this);
+            clone._hash = this._hash.clone();
+
+            return clone;
+        }
+    });
+
+    /**
+     * Shortcut function to the hasher's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     *
+     * @return {WordArray} The hash.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hash = CryptoJS.SHA256('message');
+     *     var hash = CryptoJS.SHA256(wordArray);
+     */
+    C.SHA256 = Hasher._createHelper(SHA256);
+
+    /**
+     * Shortcut function to the HMAC's object interface.
+     *
+     * @param {WordArray|string} message The message to hash.
+     * @param {WordArray|string} key The secret key.
+     *
+     * @return {WordArray} The HMAC.
+     *
+     * @static
+     *
+     * @example
+     *
+     *     var hmac = CryptoJS.HmacSHA256(message, key);
+     */
+    C.HmacSHA256 = Hasher._createHmacHelper(SHA256);
+}(Math));
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+(function () {
+    // Shortcuts
+    var C = CryptoJS;
+    var C_lib = C.lib;
+    var Base = C_lib.Base;
+    var C_enc = C.enc;
+    var Utf8 = C_enc.Utf8;
+    var C_algo = C.algo;
+
+    /**
+     * HMAC algorithm.
+     */
+    var HMAC = C_algo.HMAC = Base.extend({
+        /**
+         * Initializes a newly created HMAC.
+         *
+         * @param {Hasher} hasher The hash algorithm to use.
+         * @param {WordArray|string} key The secret key.
+         *
+         * @example
+         *
+         *     var hmacHasher = CryptoJS.algo.HMAC.create(CryptoJS.algo.SHA256, key);
+         */
+        init: function (hasher, key) {
+            // Init hasher
+            hasher = this._hasher = new hasher.init();
+
+            // Convert string to WordArray, else assume WordArray already
+            if (typeof key == 'string') {
+                key = Utf8.parse(key);
+            }
+
+            // Shortcuts
+            var hasherBlockSize = hasher.blockSize;
+            var hasherBlockSizeBytes = hasherBlockSize * 4;
+
+            // Allow arbitrary length keys
+            if (key.sigBytes > hasherBlockSizeBytes) {
+                key = hasher.finalize(key);
+            }
+
+            // Clamp excess bits
+            key.clamp();
+
+            // Clone key for inner and outer pads
+            var oKey = this._oKey = key.clone();
+            var iKey = this._iKey = key.clone();
+
+            // Shortcuts
+            var oKeyWords = oKey.words;
+            var iKeyWords = iKey.words;
+
+            // XOR keys with pad constants
+            for (var i = 0; i < hasherBlockSize; i++) {
+                oKeyWords[i] ^= 0x5c5c5c5c;
+                iKeyWords[i] ^= 0x36363636;
+            }
+            oKey.sigBytes = iKey.sigBytes = hasherBlockSizeBytes;
+
+            // Set initial values
+            this.reset();
+        },
+
+        /**
+         * Resets this HMAC to its initial state.
+         *
+         * @example
+         *
+         *     hmacHasher.reset();
+         */
+        reset: function () {
+            // Shortcut
+            var hasher = this._hasher;
+
+            // Reset
+            hasher.reset();
+            hasher.update(this._iKey);
+        },
+
+        /**
+         * Updates this HMAC with a message.
+         *
+         * @param {WordArray|string} messageUpdate The message to append.
+         *
+         * @return {HMAC} This HMAC instance.
+         *
+         * @example
+         *
+         *     hmacHasher.update('message');
+         *     hmacHasher.update(wordArray);
+         */
+        update: function (messageUpdate) {
+            this._hasher.update(messageUpdate);
+
+            // Chainable
+            return this;
+        },
+
+        /**
+         * Finalizes the HMAC computation.
+         * Note that the finalize operation is effectively a destructive, read-once operation.
+         *
+         * @param {WordArray|string} messageUpdate (Optional) A final message update.
+         *
+         * @return {WordArray} The HMAC.
+         *
+         * @example
+         *
+         *     var hmac = hmacHasher.finalize();
+         *     var hmac = hmacHasher.finalize('message');
+         *     var hmac = hmacHasher.finalize(wordArray);
+         */
+        finalize: function (messageUpdate) {
+            // Shortcut
+            var hasher = this._hasher;
+
+            // Compute HMAC
+            var innerHash = hasher.finalize(messageUpdate);
+            hasher.reset();
+            var hmac = hasher.finalize(this._oKey.clone().concat(innerHash));
+
+            return hmac;
+        }
+    });
+}());
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * A noop padding strategy.
+ */
+CryptoJS.pad.NoPadding = {
+    pad: function () {
+    },
+
+    unpad: function () {
+    }
+};
+
+/*
+CryptoJS v3.1.2
+code.google.com/p/crypto-js
+(c) 2009-2013 by Jeff Mott. All rights reserved.
+code.google.com/p/crypto-js/wiki/License
+*/
+/**
+ * Counter block mode.
+ */
+CryptoJS.mode.CTR = (function () {
+    var CTR = CryptoJS.lib.BlockCipherMode.extend();
+
+    var Encryptor = CTR.Encryptor = CTR.extend({
+        processBlock: function (words, offset) {
+            // Shortcuts
+            var cipher = this._cipher
+            var blockSize = cipher.blockSize;
+            var iv = this._iv;
+            var counter = this._counter;
+
+            // Generate keystream
+            if (iv) {
+                counter = this._counter = iv.slice(0);
+
+                // Remove IV for subsequent blocks
+                this._iv = undefined;
+            }
+            var keystream = counter.slice(0);
+            cipher.encryptBlock(keystream, 0);
+
+            // Increment counter
+            counter[blockSize - 1] = (counter[blockSize - 1] + 1) | 0
+
+            // Encrypt
+            for (var i = 0; i < blockSize; i++) {
+                words[offset + i] ^= keystream[i];
+            }
+        }
+    });
+
+    CTR.Decryptor = Encryptor;
+
+    return CTR;
+}());
+
+
+  return CryptoJS
+
+}))
\ No newline at end of file
diff --git a/lib/otr/vendor/eventemitter.js b/lib/otr/vendor/eventemitter.js
new file mode 100644
index 0000000..0443d1f
--- /dev/null
+++ b/lib/otr/vendor/eventemitter.js
@@ -0,0 +1,455 @@
+/*!
+ * EventEmitter v4.2.3 - git.io/ee
+ * Oliver Caldwell
+ * MIT license
+ * @preserve
+ */
+
+(function () {
+	'use strict';
+
+	/**
+	 * Class for managing events.
+	 * Can be extended to provide event functionality in other classes.
+	 *
+	 * @class EventEmitter Manages event registering and emitting.
+	 */
+	function EventEmitter() {}
+
+	// Shortcuts to improve speed and size
+
+	// Easy access to the prototype
+	var proto = EventEmitter.prototype;
+
+	/**
+	 * Finds the index of the listener for the event in it's storage array.
+	 *
+	 * @param {Function[]} listeners Array of listeners to search through.
+	 * @param {Function} listener Method to look for.
+	 * @return {Number} Index of the specified listener, -1 if not found
+	 * @api private
+	 */
+	function indexOfListener(listeners, listener) {
+		var i = listeners.length;
+		while (i--) {
+			if (listeners[i].listener === listener) {
+				return i;
+			}
+		}
+
+		return -1;
+	}
+
+	/**
+	 * Alias a method while keeping the context correct, to allow for overwriting of target method.
+	 *
+	 * @param {String} name The name of the target method.
+	 * @return {Function} The aliased method
+	 * @api private
+	 */
+	function alias(name) {
+		return function aliasClosure() {
+			return this[name].apply(this, arguments);
+		};
+	}
+
+	/**
+	 * Returns the listener array for the specified event.
+	 * Will initialise the event object and listener arrays if required.
+	 * Will return an object if you use a regex search. The object contains keys for each matched event. So /ba[rz]/ might return an object containing bar and baz. But only if you have either defined them with defineEvent or added some listeners to them.
+	 * Each property in the object response is an array of listener functions.
+	 *
+	 * @param {String|RegExp} evt Name of the event to return the listeners from.
+	 * @return {Function[]|Object} All listener functions for the event.
+	 */
+	proto.getListeners = function getListeners(evt) {
+		var events = this._getEvents();
+		var response;
+		var key;
+
+		// Return a concatenated array of all matching events if
+		// the selector is a regular expression.
+		if (typeof evt === 'object') {
+			response = {};
+			for (key in events) {
+				if (events.hasOwnProperty(key) && evt.test(key)) {
+					response[key] = events[key];
+				}
+			}
+		}
+		else {
+			response = events[evt] || (events[evt] = []);
+		}
+
+		return response;
+	};
+
+	/**
+	 * Takes a list of listener objects and flattens it into a list of listener functions.
+	 *
+	 * @param {Object[]} listeners Raw listener objects.
+	 * @return {Function[]} Just the listener functions.
+	 */
+	proto.flattenListeners = function flattenListeners(listeners) {
+		var flatListeners = [];
+		var i;
+
+		for (i = 0; i < listeners.length; i += 1) {
+			flatListeners.push(listeners[i].listener);
+		}
+
+		return flatListeners;
+	};
+
+	/**
+	 * Fetches the requested listeners via getListeners but will always return the results inside an object. This is mainly for internal use but others may find it useful.
+	 *
+	 * @param {String|RegExp} evt Name of the event to return the listeners from.
+	 * @return {Object} All listener functions for an event in an object.
+	 */
+	proto.getListenersAsObject = function getListenersAsObject(evt) {
+		var listeners = this.getListeners(evt);
+		var response;
+
+		if (listeners instanceof Array) {
+			response = {};
+			response[evt] = listeners;
+		}
+
+		return response || listeners;
+	};
+
+	/**
+	 * Adds a listener function to the specified event.
+	 * The listener will not be added if it is a duplicate.
+	 * If the listener returns true then it will be removed after it is called.
+	 * If you pass a regular expression as the event name then the listener will be added to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to attach the listener to.
+	 * @param {Function} listener Method to be called when the event is emitted. If the function returns true then it will be removed after calling.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addListener = function addListener(evt, listener) {
+		var listeners = this.getListenersAsObject(evt);
+		var listenerIsWrapped = typeof listener === 'object';
+		var key;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key) && indexOfListener(listeners[key], listener) === -1) {
+				listeners[key].push(listenerIsWrapped ? listener : {
+					listener: listener,
+					once: false
+				});
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of addListener
+	 */
+	proto.on = alias('addListener');
+
+	/**
+	 * Semi-alias of addListener. It will add a listener that will be
+	 * automatically removed after it's first execution.
+	 *
+	 * @param {String|RegExp} evt Name of the event to attach the listener to.
+	 * @param {Function} listener Method to be called when the event is emitted. If the function returns true then it will be removed after calling.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addOnceListener = function addOnceListener(evt, listener) {
+		return this.addListener(evt, {
+			listener: listener,
+			once: true
+		});
+	};
+
+	/**
+	 * Alias of addOnceListener.
+	 */
+	proto.once = alias('addOnceListener');
+
+	/**
+	 * Defines an event name. This is required if you want to use a regex to add a listener to multiple events at once. If you don't do this then how do you expect it to know what event to add to? Should it just add to every possible match for a regex? No. That is scary and bad.
+	 * You need to tell it what event names should be matched by a regex.
+	 *
+	 * @param {String} evt Name of the event to create.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.defineEvent = function defineEvent(evt) {
+		this.getListeners(evt);
+		return this;
+	};
+
+	/**
+	 * Uses defineEvent to define multiple events.
+	 *
+	 * @param {String[]} evts An array of event names to define.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.defineEvents = function defineEvents(evts) {
+		for (var i = 0; i < evts.length; i += 1) {
+			this.defineEvent(evts[i]);
+		}
+		return this;
+	};
+
+	/**
+	 * Removes a listener function from the specified event.
+	 * When passed a regular expression as the event name, it will remove the listener from all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to remove the listener from.
+	 * @param {Function} listener Method to remove from the event.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeListener = function removeListener(evt, listener) {
+		var listeners = this.getListenersAsObject(evt);
+		var index;
+		var key;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key)) {
+				index = indexOfListener(listeners[key], listener);
+
+				if (index !== -1) {
+					listeners[key].splice(index, 1);
+				}
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of removeListener
+	 */
+	proto.off = alias('removeListener');
+
+	/**
+	 * Adds listeners in bulk using the manipulateListeners method.
+	 * If you pass an object as the second argument you can add to multiple events at once. The object should contain key value pairs of events and listeners or listener arrays. You can also pass it an event name and an array of listeners to be added.
+	 * You can also pass it a regular expression to add the array of listeners to all events that match it.
+	 * Yeah, this function does quite a bit. That's probably a bad thing.
+	 *
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to add to multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to add.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.addListeners = function addListeners(evt, listeners) {
+		// Pass through to manipulateListeners
+		return this.manipulateListeners(false, evt, listeners);
+	};
+
+	/**
+	 * Removes listeners in bulk using the manipulateListeners method.
+	 * If you pass an object as the second argument you can remove from multiple events at once. The object should contain key value pairs of events and listeners or listener arrays.
+	 * You can also pass it an event name and an array of listeners to be removed.
+	 * You can also pass it a regular expression to remove the listeners from all events that match it.
+	 *
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to remove from multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to remove.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeListeners = function removeListeners(evt, listeners) {
+		// Pass through to manipulateListeners
+		return this.manipulateListeners(true, evt, listeners);
+	};
+
+	/**
+	 * Edits listeners in bulk. The addListeners and removeListeners methods both use this to do their job. You should really use those instead, this is a little lower level.
+	 * The first argument will determine if the listeners are removed (true) or added (false).
+	 * If you pass an object as the second argument you can add/remove from multiple events at once. The object should contain key value pairs of events and listeners or listener arrays.
+	 * You can also pass it an event name and an array of listeners to be added/removed.
+	 * You can also pass it a regular expression to manipulate the listeners of all events that match it.
+	 *
+	 * @param {Boolean} remove True if you want to remove listeners, false if you want to add.
+	 * @param {String|Object|RegExp} evt An event name if you will pass an array of listeners next. An object if you wish to add/remove from multiple events at once.
+	 * @param {Function[]} [listeners] An optional array of listener functions to add/remove.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.manipulateListeners = function manipulateListeners(remove, evt, listeners) {
+		var i;
+		var value;
+		var single = remove ? this.removeListener : this.addListener;
+		var multiple = remove ? this.removeListeners : this.addListeners;
+
+		// If evt is an object then pass each of it's properties to this method
+		if (typeof evt === 'object' && !(evt instanceof RegExp)) {
+			for (i in evt) {
+				if (evt.hasOwnProperty(i) && (value = evt[i])) {
+					// Pass the single listener straight through to the singular method
+					if (typeof value === 'function') {
+						single.call(this, i, value);
+					}
+					else {
+						// Otherwise pass back to the multiple function
+						multiple.call(this, i, value);
+					}
+				}
+			}
+		}
+		else {
+			// So evt must be a string
+			// And listeners must be an array of listeners
+			// Loop over it and pass each one to the multiple method
+			i = listeners.length;
+			while (i--) {
+				single.call(this, evt, listeners[i]);
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Removes all listeners from a specified event.
+	 * If you do not specify an event then all listeners will be removed.
+	 * That means every event will be emptied.
+	 * You can also pass a regex to remove all events that match it.
+	 *
+	 * @param {String|RegExp} [evt] Optional name of the event to remove all listeners for. Will remove from every event if not passed.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.removeEvent = function removeEvent(evt) {
+		var type = typeof evt;
+		var events = this._getEvents();
+		var key;
+
+		// Remove different things depending on the state of evt
+		if (type === 'string') {
+			// Remove all listeners for the specified event
+			delete events[evt];
+		}
+		else if (type === 'object') {
+			// Remove all events matching the regex.
+			for (key in events) {
+				if (events.hasOwnProperty(key) && evt.test(key)) {
+					delete events[key];
+				}
+			}
+		}
+		else {
+			// Remove all listeners in all events
+			delete this._events;
+		}
+
+		return this;
+	};
+
+	/**
+	 * Emits an event of your choice.
+	 * When emitted, every listener attached to that event will be executed.
+	 * If you pass the optional argument array then those arguments will be passed to every listener upon execution.
+	 * Because it uses `apply`, your array of arguments will be passed as if you wrote them out separately.
+	 * So they will not arrive within the array on the other side, they will be separate.
+	 * You can also pass a regular expression to emit to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to emit and execute listeners for.
+	 * @param {Array} [args] Optional array of arguments to be passed to each listener.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.emitEvent = function emitEvent(evt, args) {
+		var listeners = this.getListenersAsObject(evt);
+		var listener;
+		var i;
+		var key;
+		var response;
+
+		for (key in listeners) {
+			if (listeners.hasOwnProperty(key)) {
+				i = listeners[key].length;
+
+				while (i--) {
+					// If the listener returns true then it shall be removed from the event
+					// The function is executed either with a basic call or an apply if there is an args array
+					listener = listeners[key][i];
+
+					if (listener.once === true) {
+						this.removeListener(evt, listener.listener);
+					}
+
+					response = listener.listener.apply(this, args || []);
+
+					if (response === this._getOnceReturnValue()) {
+						this.removeListener(evt, listener.listener);
+					}
+				}
+			}
+		}
+
+		return this;
+	};
+
+	/**
+	 * Alias of emitEvent
+	 */
+	proto.trigger = alias('emitEvent');
+
+	/**
+	 * Subtly different from emitEvent in that it will pass its arguments on to the listeners, as opposed to taking a single array of arguments to pass on.
+	 * As with emitEvent, you can pass a regex in place of the event name to emit to all events that match it.
+	 *
+	 * @param {String|RegExp} evt Name of the event to emit and execute listeners for.
+	 * @param {...*} Optional additional arguments to be passed to each listener.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.emit = function emit(evt) {
+		var args = Array.prototype.slice.call(arguments, 1);
+		return this.emitEvent(evt, args);
+	};
+
+	/**
+	 * Sets the current value to check against when executing listeners. If a
+	 * listeners return value matches the one set here then it will be removed
+	 * after execution. This value defaults to true.
+	 *
+	 * @param {*} value The new value to check for when executing listeners.
+	 * @return {Object} Current instance of EventEmitter for chaining.
+	 */
+	proto.setOnceReturnValue = function setOnceReturnValue(value) {
+		this._onceReturnValue = value;
+		return this;
+	};
+
+	/**
+	 * Fetches the current value to check against when executing listeners. If
+	 * the listeners return value matches this one then it should be removed
+	 * automatically. It will return true by default.
+	 *
+	 * @return {*|Boolean} The current value to check for or the default, true.
+	 * @api private
+	 */
+	proto._getOnceReturnValue = function _getOnceReturnValue() {
+		if (this.hasOwnProperty('_onceReturnValue')) {
+			return this._onceReturnValue;
+		}
+		else {
+			return true;
+		}
+	};
+
+	/**
+	 * Fetches the events object and creates one if required.
+	 *
+	 * @return {Object} The events storage object.
+	 * @api private
+	 */
+	proto._getEvents = function _getEvents() {
+		return this._events || (this._events = {});
+	};
+
+	// Expose the class either via AMD, CommonJS or the global object
+	if (typeof define === 'function' && define.amd) {
+		define(function () {
+			return EventEmitter;
+		});
+	}
+	else if (typeof module === 'object' && module.exports){
+		module.exports = EventEmitter;
+	}
+	else {
+		this.EventEmitter = EventEmitter;
+	}
+}.call(this));
diff --git a/lib/otr/vendor/salsa20.js b/lib/otr/vendor/salsa20.js
new file mode 100644
index 0000000..485314b
--- /dev/null
+++ b/lib/otr/vendor/salsa20.js
@@ -0,0 +1,254 @@
+// Salsa20 implementation
+// Contributed to Cryptocat by Dmitry Chestnykh
+// 21-01-2013
+
+;(function (root, factory) {
+
+  if (typeof define === 'function' && define.amd) {
+    define(factory)
+  } else if (typeof module !== 'undefined' && module.exports) {
+    module.exports = factory()
+  } else {
+    root.Salsa20 = factory()
+  }
+
+}(this, function () {
+
+    function Salsa20(key, nonce) {
+        // Constants.
+        this.rounds = 20; // number of Salsa rounds
+        this.sigmaWords = [0x61707865, 0x3320646e, 0x79622d32, 0x6b206574];
+
+        // State.
+        this.keyWords = [];           // key words
+        this.nonceWords = [0, 0];     // nonce words
+        this.counterWords = [0, 0];   // block counter words
+
+        // Output buffer.
+        this.block = [];        // output block of 64 bytes
+        this.blockUsed = 64;     // number of block bytes used
+
+        this.setKey(key);
+        this.setNonce(nonce);
+    }
+
+    // setKey sets the key to the given 32-byte array.
+    Salsa20.prototype.setKey = function(key) {
+        for (var i = 0, j = 0; i < 8; i++, j += 4) {
+            this.keyWords[i] = (key[j] & 0xff)        |
+                              ((key[j+1] & 0xff)<<8)  |
+                              ((key[j+2] & 0xff)<<16) |
+                              ((key[j+3] & 0xff)<<24);
+        }
+        this._reset();
+    };
+
+    // setNonce sets the nonce to the given 8-byte array.
+    Salsa20.prototype.setNonce = function(nonce) {
+        this.nonceWords[0] = (nonce[0] & 0xff)      |
+                            ((nonce[1] & 0xff)<<8)  |
+                            ((nonce[2] & 0xff)<<16) |
+                            ((nonce[3] & 0xff)<<24);
+        this.nonceWords[1] = (nonce[4] & 0xff)      |
+                            ((nonce[5] & 0xff)<<8)  |
+                            ((nonce[6] & 0xff)<<16) |
+                            ((nonce[7] & 0xff)<<24);
+        this._reset();
+    };
+
+    // getBytes returns the next numberOfBytes bytes of stream.
+    Salsa20.prototype.getBytes = function(numberOfBytes) {
+        var out = new Array(numberOfBytes);
+        for (var i = 0; i < numberOfBytes; i++) {
+            if (this.blockUsed == 64) {
+                this._generateBlock();
+                this._incrementCounter();
+                this.blockUsed = 0;
+            }
+            out[i] = this.block[this.blockUsed];
+            this.blockUsed++;
+        }
+        return out;
+    };
+
+    Salsa20.prototype.getHexString = function(numberOfBytes) {
+        var hex=['0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f'];
+        var out = [];
+        var bytes = this.getBytes(numberOfBytes);
+        for(var i = 0; i < bytes.length; i++) {
+            out.push(hex[(bytes[i] >> 4) & 15]);
+            out.push(hex[bytes[i] & 15]);
+        }
+        return out.join('');
+    };
+
+    // Private methods.
+
+    Salsa20.prototype._reset = function() {
+        this.counterWords[0] = 0;
+        this.counterWords[1] = 0;
+        this.blockUsed = 64;
+    };
+
+    // _incrementCounter increments block counter.
+    Salsa20.prototype._incrementCounter = function() {
+        // Note: maximum 2^64 blocks.
+        this.counterWords[0] = (this.counterWords[0] + 1) & 0xffffffff;
+        if (this.counterWords[0] == 0) {
+            this.counterWords[1] = (this.counterWords[1] + 1) & 0xffffffff;
+        }
+    };
+
+    // _generateBlock generates 64 bytes from key, nonce, and counter,
+    // and puts the result into this.block.
+    Salsa20.prototype._generateBlock = function() {
+        var j0 = this.sigmaWords[0],
+            j1 = this.keyWords[0],
+            j2 = this.keyWords[1],
+            j3 = this.keyWords[2],
+            j4 = this.keyWords[3],
+            j5 = this.sigmaWords[1],
+            j6 = this.nonceWords[0],
+            j7 = this.nonceWords[1],
+            j8 = this.counterWords[0],
+            j9 = this.counterWords[1],
+            j10 = this.sigmaWords[2],
+            j11 = this.keyWords[4],
+            j12 = this.keyWords[5],
+            j13 = this.keyWords[6],
+            j14 = this.keyWords[7],
+            j15 = this.sigmaWords[3];
+
+            var x0 = j0, x1 = j1, x2 = j2, x3 = j3, x4 = j4, x5 = j5, x6 = j6, x7 = j7,
+                x8 = j8, x9 = j9, x10 = j10, x11 = j11, x12 = j12, x13 = j13, x14 = j14, x15 = j15;
+
+            var u;
+
+            for (var i = 0; i < this.rounds; i += 2) {
+                u = x0 + x12;
+                x4 ^= (u<<7) | (u>>>(32-7));
+                u = x4 + x0;
+                x8 ^= (u<<9) | (u>>>(32-9));
+                u = x8 + x4;
+                x12 ^= (u<<13) | (u>>>(32-13));
+                u = x12 + x8;
+                x0 ^= (u<<18) | (u>>>(32-18));
+
+                u = x5 + x1;
+                x9 ^= (u<<7) | (u>>>(32-7));
+                u = x9 + x5;
+                x13 ^= (u<<9) | (u>>>(32-9));
+                u = x13 + x9;
+                x1 ^= (u<<13) | (u>>>(32-13));
+                u = x1 + x13;
+                x5 ^= (u<<18) | (u>>>(32-18));
+
+                u = x10 + x6;
+                x14 ^= (u<<7) | (u>>>(32-7));
+                u = x14 + x10;
+                x2 ^= (u<<9) | (u>>>(32-9));
+                u = x2 + x14;
+                x6 ^= (u<<13) | (u>>>(32-13));
+                u = x6 + x2;
+                x10 ^= (u<<18) | (u>>>(32-18));
+
+                u = x15 + x11;
+                x3 ^= (u<<7) | (u>>>(32-7));
+                u = x3 + x15;
+                x7 ^= (u<<9) | (u>>>(32-9));
+                u = x7 + x3;
+                x11 ^= (u<<13) | (u>>>(32-13));
+                u = x11 + x7;
+                x15 ^= (u<<18) | (u>>>(32-18));
+
+                u = x0 + x3;
+                x1 ^= (u<<7) | (u>>>(32-7));
+                u = x1 + x0;
+                x2 ^= (u<<9) | (u>>>(32-9));
+                u = x2 + x1;
+                x3 ^= (u<<13) | (u>>>(32-13));
+                u = x3 + x2;
+                x0 ^= (u<<18) | (u>>>(32-18));
+
+                u = x5 + x4;
+                x6 ^= (u<<7) | (u>>>(32-7));
+                u = x6 + x5;
+                x7 ^= (u<<9) | (u>>>(32-9));
+                u = x7 + x6;
+                x4 ^= (u<<13) | (u>>>(32-13));
+                u = x4 + x7;
+                x5 ^= (u<<18) | (u>>>(32-18));
+
+                u = x10 + x9;
+                x11 ^= (u<<7) | (u>>>(32-7));
+                u = x11 + x10;
+                x8 ^= (u<<9) | (u>>>(32-9));
+                u = x8 + x11;
+                x9 ^= (u<<13) | (u>>>(32-13));
+                u = x9 + x8;
+                x10 ^= (u<<18) | (u>>>(32-18));
+
+                u = x15 + x14;
+                x12 ^= (u<<7) | (u>>>(32-7));
+                u = x12 + x15;
+                x13 ^= (u<<9) | (u>>>(32-9));
+                u = x13 + x12;
+                x14 ^= (u<<13) | (u>>>(32-13));
+                u = x14 + x13;
+                x15 ^= (u<<18) | (u>>>(32-18));
+            }
+
+            x0 += j0;
+            x1 += j1;
+            x2 += j2;
+            x3 += j3;
+            x4 += j4;
+            x5 += j5;
+            x6 += j6;
+            x7 += j7;
+            x8 += j8;
+            x9 += j9;
+            x10 += j10;
+            x11 += j11;
+            x12 += j12;
+            x13 += j13;
+            x14 += j14;
+            x15 += j15;
+
+            this.block[ 0] = ( x0 >>>  0) & 0xff; this.block[ 1] = ( x0 >>>  8) & 0xff;
+            this.block[ 2] = ( x0 >>> 16) & 0xff; this.block[ 3] = ( x0 >>> 24) & 0xff;
+            this.block[ 4] = ( x1 >>>  0) & 0xff; this.block[ 5] = ( x1 >>>  8) & 0xff;
+            this.block[ 6] = ( x1 >>> 16) & 0xff; this.block[ 7] = ( x1 >>> 24) & 0xff;
+            this.block[ 8] = ( x2 >>>  0) & 0xff; this.block[ 9] = ( x2 >>>  8) & 0xff;
+            this.block[10] = ( x2 >>> 16) & 0xff; this.block[11] = ( x2 >>> 24) & 0xff;
+            this.block[12] = ( x3 >>>  0) & 0xff; this.block[13] = ( x3 >>>  8) & 0xff;
+            this.block[14] = ( x3 >>> 16) & 0xff; this.block[15] = ( x3 >>> 24) & 0xff;
+            this.block[16] = ( x4 >>>  0) & 0xff; this.block[17] = ( x4 >>>  8) & 0xff;
+            this.block[18] = ( x4 >>> 16) & 0xff; this.block[19] = ( x4 >>> 24) & 0xff;
+            this.block[20] = ( x5 >>>  0) & 0xff; this.block[21] = ( x5 >>>  8) & 0xff;
+            this.block[22] = ( x5 >>> 16) & 0xff; this.block[23] = ( x5 >>> 24) & 0xff;
+            this.block[24] = ( x6 >>>  0) & 0xff; this.block[25] = ( x6 >>>  8) & 0xff;
+            this.block[26] = ( x6 >>> 16) & 0xff; this.block[27] = ( x6 >>> 24) & 0xff;
+            this.block[28] = ( x7 >>>  0) & 0xff; this.block[29] = ( x7 >>>  8) & 0xff;
+            this.block[30] = ( x7 >>> 16) & 0xff; this.block[31] = ( x7 >>> 24) & 0xff;
+            this.block[32] = ( x8 >>>  0) & 0xff; this.block[33] = ( x8 >>>  8) & 0xff;
+            this.block[34] = ( x8 >>> 16) & 0xff; this.block[35] = ( x8 >>> 24) & 0xff;
+            this.block[36] = ( x9 >>>  0) & 0xff; this.block[37] = ( x9 >>>  8) & 0xff;
+            this.block[38] = ( x9 >>> 16) & 0xff; this.block[39] = ( x9 >>> 24) & 0xff;
+            this.block[40] = (x10 >>>  0) & 0xff; this.block[41] = (x10 >>>  8) & 0xff;
+            this.block[42] = (x10 >>> 16) & 0xff; this.block[43] = (x10 >>> 24) & 0xff;
+            this.block[44] = (x11 >>>  0) & 0xff; this.block[45] = (x11 >>>  8) & 0xff;
+            this.block[46] = (x11 >>> 16) & 0xff; this.block[47] = (x11 >>> 24) & 0xff;
+            this.block[48] = (x12 >>>  0) & 0xff; this.block[49] = (x12 >>>  8) & 0xff;
+            this.block[50] = (x12 >>> 16) & 0xff; this.block[51] = (x12 >>> 24) & 0xff;
+            this.block[52] = (x13 >>>  0) & 0xff; this.block[53] = (x13 >>>  8) & 0xff;
+            this.block[54] = (x13 >>> 16) & 0xff; this.block[55] = (x13 >>> 24) & 0xff;
+            this.block[56] = (x14 >>>  0) & 0xff; this.block[57] = (x14 >>>  8) & 0xff;
+            this.block[58] = (x14 >>> 16) & 0xff; this.block[59] = (x14 >>> 24) & 0xff;
+            this.block[60] = (x15 >>>  0) & 0xff; this.block[61] = (x15 >>>  8) & 0xff;
+            this.block[62] = (x15 >>> 16) & 0xff; this.block[63] = (x15 >>> 24) & 0xff;
+    };
+
+  return Salsa20
+
+}))
\ No newline at end of file
diff --git a/lib/strophe.caps.js b/lib/strophe.caps.js
new file mode 100644
index 0000000..de2ffdc
--- /dev/null
+++ b/lib/strophe.caps.js
@@ -0,0 +1,264 @@
+/**
+ * Entity Capabilities (XEP-0115)
+ * 
+ * Depends on disco plugin.
+ * 
+ * See: http://xmpp.org/extensions/xep-0115.html
+ * 
+ * Authors: - Michael Weibel <michael.weibel at gmail.com> - Klaus Herberth <klaus at jsxc.org>
+ * Copyright: - Michael Weibel <michael.weibel at gmail.com>
+ * 
+ * @license MIT
+ */
+
+(function($) {
+   Strophe.addConnectionPlugin('caps', {
+      /**
+       * Constant: HASH Hash used
+       * 
+       * Currently only sha-1 is supported.
+       */
+      HASH: 'sha-1',
+      /**
+       * Variable: node Client which is being used.
+       * 
+       * Can be overwritten as soon as Strophe has been initialized.
+       */
+      node: 'http://strophe.im/strophejs/',
+      /**
+       * PrivateVariable: _ver Own generated version string
+       */
+      _ver: '',
+      /**
+       * PrivateVariable: _connection Strophe connection
+       */
+      _connection: null,
+      /**
+       * PrivateVariable: _knownCapabilities A hashtable containing
+       * version-strings and their capabilities, serialized as string.
+       * 
+       * TODO: Maybe those caps shouldn't be serialized.
+       */
+      _knownCapabilities: JSON.parse(localStorage.getItem('strophe.caps._knownCapabilities')) || {},
+
+      /**
+       * PrivateVariable: _jidVerIndex A hashtable containing jids and their
+       * versions for better lookup of capabilities.
+       */
+      _jidVerIndex: JSON.parse(localStorage.getItem('strophe.caps._jidVerIndex')) || {},
+
+      /**
+       * Function: init Initialize plugin: - Add caps namespace - Add caps
+       * feature to disco plugin - Add handler for caps stanzas
+       * 
+       * Parameters: (Strophe.Connection) conn - Strophe connection
+       */
+      init: function(conn) {
+         this._connection = conn;
+
+         Strophe.addNamespace('CAPS', 'http://jabber.org/protocol/caps');
+
+         if (!this._connection.disco) {
+            throw "Caps plugin requires the disco plugin to be installed.";
+         }
+
+         this._connection.disco.addFeature(Strophe.NS.CAPS);
+         this._connection.addHandler(this._delegateCapabilities.bind(this), Strophe.NS.CAPS);
+      },
+
+      /**
+       * Function: generateCapsAttrs Returns the attributes for generating the
+       * "c"-stanza containing the own version
+       * 
+       * Returns: (Object) - attributes
+       */
+      generateCapsAttrs: function() {
+         return {
+            'xmlns': Strophe.NS.CAPS,
+            'hash': this.HASH,
+            'node': this.node,
+            'ver': this.generateVer()
+         };
+      },
+
+      /**
+       * Function: generateVer Returns the base64 encoded version string
+       * (encoded itself with sha1)
+       * 
+       * Returns: (String) - version
+       */
+      generateVer: function() {
+         if (this._ver !== "") {
+            return this._ver;
+         }
+
+         var ver = "", identities = this._connection.disco._identities.sort(this._sortIdentities), identitiesLen = identities.length, features = this._connection.disco._features.sort(), featuresLen = features.length;
+         for (var i = 0; i < identitiesLen; i++) {
+            var curIdent = identities[i];
+            ver += curIdent.category + "/" + curIdent.type + "/" + curIdent.lang + "/" + curIdent.name + "<";
+         }
+         for (var i = 0; i < featuresLen; i++) {
+            ver += features[i] + '<';
+         }
+
+         this._ver = b64_sha1(ver);
+         return this._ver;
+      },
+
+      /**
+       * Function: getCapabilitiesByJid Returns serialized capabilities of a jid
+       * (if available). Otherwise null.
+       * 
+       * Parameters: (String) jid - Jabber id
+       * 
+       * Returns: (String|null) - capabilities, serialized; or null when not
+       * available.
+       */
+      getCapabilitiesByJid: function(jid) {
+         if (this._jidVerIndex[jid]) {
+            return this._knownCapabilities[this._jidVerIndex[jid]];
+         }
+         return null;
+      },
+      hasFeatureByJid: function(jid, feature) {
+         if (this._jidVerIndex[jid] && feature !== null && typeof feature !== 'undefined') {
+            if(!$.isArray(feature)){
+               feature = $.makeArray(feature);
+            };
+            
+            var i;
+            for (i = 0; i < feature.length; i++) {
+               if (this._knownCapabilities[this._jidVerIndex[jid]]['features'].indexOf(feature[i]) < 0)
+                  return false;
+            }
+            return true;
+         }
+         return false;
+      },
+
+      /**
+       * PrivateFunction: _delegateCapabilities Checks if the version has
+       * already been saved. If yes: do nothing. If no: Request capabilities
+       * 
+       * Parameters: (Strophe.Builder) stanza - Stanza
+       * 
+       * Returns: (Boolean)
+       */
+      _delegateCapabilities: function(stanza) {
+         var from = stanza.getAttribute('from'), c = stanza.querySelector('c'), ver = c.getAttribute('ver'), node = c.getAttribute('node');
+         if (!this._knownCapabilities[ver]) {
+            return this._requestCapabilities(from, node, ver);
+         } else {
+            this._jidVerIndex[from] = ver;
+         }
+         if (!this._jidVerIndex[from] || !this._jidVerIndex[from] !== ver) {
+            this._jidVerIndex[from] = ver;
+         }
+
+         localStorage.setItem('strophe.caps._jidVerIndex', JSON.stringify(this._jidVerIndex));
+         $(document).trigger('caps.strophe', [ from ]);
+
+         return true;
+      },
+
+      /**
+       * PrivateFunction: _requestCapabilities Requests capabilities from the
+       * one which sent the caps-info stanza. This is done using disco info.
+       * 
+       * Additionally, it registers a handler for handling the reply.
+       * 
+       * Parameters: (String) to - Destination jid (String) node - Node
+       * attribute of the caps-stanza (String) ver - Version of the caps-stanza
+       * 
+       * Returns: (Boolean) - true
+       */
+      _requestCapabilities: function(to, node, ver) {
+         if (to !== this._connection.jid) {
+            var id = this._connection.disco.info(to, node + '#' + ver);
+            this._connection.addHandler(this._handleDiscoInfoReply.bind(this), Strophe.NS.DISCO_INFO, 'iq', 'result', id, to);
+         }
+         return true;
+      },
+
+      /**
+       * PrivateFunction: _handleDiscoInfoReply Parses the disco info reply and
+       * adds the version & it's capabilities to the _knownCapabilities
+       * variable. Additionally, it adds the jid & the version to the
+       * _jidVerIndex variable for a better lookup.
+       * 
+       * Parameters: (Strophe.Builder) stanza - Disco info stanza
+       * 
+       * Returns: (Boolean) - false, to automatically remove the handler.
+       */
+      _handleDiscoInfoReply: function(stanza) {
+         var query = stanza.querySelector('query'), node = query.getAttribute('node').split('#'), ver = node[1], from = stanza.getAttribute('from');
+         if (!this._knownCapabilities[ver]) {
+            var childNodes = query.childNodes, childNodesLen = childNodes.length;
+            this._knownCapabilities[ver] = {
+               features: [],
+               identities: []
+            };
+            for (var i = 0; i < childNodesLen; i++) {
+               var node = childNodes[i];
+               if (node.nodeName == 'feature') {
+                  this._knownCapabilities[ver]['features'].push(node.getAttribute('var'));
+               } else if (node.nodeName == 'identity') {
+                  this._knownCapabilities[ver]['identities'].push(this._attributesToJsObject(node.attributes));
+               } else {
+                  if (_knownCapabilities[ver][node.nodeName])
+                     _knownCapabilities[ver][node.nodeName] = [];
+                  this._knownCapabilities[ver][node.nodeName].push(this._attributesToJsObject(node.attributes));
+               }
+            }
+            this._jidVerIndex[from] = ver;
+         } else if (!this._jidVerIndex[from] || !this._jidVerIndex[from] !== ver) {
+            this._jidVerIndex[from] = ver;
+         }
+
+         localStorage.setItem('strophe.caps._jidVerIndex', JSON.stringify(this._jidVerIndex));
+         localStorage.setItem('strophe.caps._knownCapabilities', JSON.stringify(this._knownCapabilities));
+         $(document).trigger('caps.strophe', [ from ]);
+
+         return false;
+      },
+
+      _attributesToJsObject: function(attr) {
+         var obj = {};
+
+         for (i = 0; i < attr.length; i++)
+            obj[attr[i].name] = attr[i].value;
+
+         return obj;
+      },
+
+      /**
+       * PrivateFunction: _sortIdentities Sorts two identities according the
+       * sorting requirements in XEP-0115.
+       * 
+       * Parameters: (Object) a - Identity a (Object) b - Identity b
+       * 
+       * Returns: (Integer) - 1, 0 or -1; according to which one's greater.
+       */
+      _sortIdentities: function(a, b) {
+         if (a.category > b.category) {
+            return 1;
+         }
+         if (a.category < b.category) {
+            return -1;
+         }
+         if (a.type > b.type) {
+            return 1;
+         }
+         if (a.type < b.type) {
+            return -1;
+         }
+         if (a.lang > b.lang) {
+            return 1;
+         }
+         if (a.lang < b.lang) {
+            return -1;
+         }
+         return 0;
+      }
+   });
+}(jQuery));
\ No newline at end of file
diff --git a/lib/strophe.disco.js b/lib/strophe.disco.js
new file mode 100644
index 0000000..eb3f913
--- /dev/null
+++ b/lib/strophe.disco.js
@@ -0,0 +1,232 @@
+/*
+  Copyright 2010, François de Metz <francois at 2metz.fr>
+*/
+
+/**
+ * Disco Strophe Plugin
+ * Implement http://xmpp.org/extensions/xep-0030.html
+ * TODO: manage node hierarchies, and node on info request
+ */
+Strophe.addConnectionPlugin('disco',
+{
+    _connection: null,
+    _identities : [],
+    _features : [],
+    _items : [],
+    /** Function: init
+     * Plugin init
+     *
+     * Parameters:
+     *   (Strophe.Connection) conn - Strophe connection
+     */
+    init: function(conn)
+    {
+    this._connection = conn;
+        this._identities = [];
+        this._features   = [];
+        this._items      = [];
+        // disco info
+        conn.addHandler(this._onDiscoInfo.bind(this), Strophe.NS.DISCO_INFO, 'iq', 'get', null, null);
+        // disco items
+        conn.addHandler(this._onDiscoItems.bind(this), Strophe.NS.DISCO_ITEMS, 'iq', 'get', null, null);
+    },
+    /** Function: addIdentity
+     * See http://xmpp.org/registrar/disco-categories.html
+     * Parameters:
+     *   (String) category - category of identity (like client, automation, etc ...)
+     *   (String) type - type of identity (like pc, web, bot , etc ...)
+     *   (String) name - name of identity in natural language
+     *   (String) lang - lang of name parameter
+     *
+     * Returns:
+     *   Boolean
+     */
+    addIdentity: function(category, type, name, lang)
+    {
+        for (var i=0; i<this._identities.length; i++)
+        {
+            if (this._identities[i].category == category &&
+                this._identities[i].type == type &&
+                this._identities[i].name == name &&
+                this._identities[i].lang == lang)
+            {
+                return false;
+            }
+        }
+        this._identities.push({category: category, type: type, name: name, lang: lang});
+        return true;
+    },
+    /** Function: addFeature
+     *
+     * Parameters:
+     *   (String) var_name - feature name (like jabber:iq:version)
+     *
+     * Returns:
+     *   boolean
+     */
+    addFeature: function(var_name)
+    {
+        for (var i=0; i<this._features.length; i++)
+        {
+             if (this._features[i] == var_name)
+                 return false;
+        }
+        this._features.push(var_name);
+        return true;
+    },
+    /** Function: removeFeature
+     *
+     * Parameters:
+     *   (String) var_name - feature name (like jabber:iq:version)
+     *
+     * Returns:
+     *   boolean
+     */
+    removeFeature: function(var_name)
+    {
+        for (var i=0; i<this._features.length; i++)
+        {
+             if (this._features[i] === var_name){
+                 this._features.splice(i,1)
+                 return true;
+             }
+        }
+        return false;
+    },
+    /** Function: addItem
+     *
+     * Parameters:
+     *   (String) jid
+     *   (String) name
+     *   (String) node
+     *   (Function) call_back
+     *
+     * Returns:
+     *   boolean
+     */
+    addItem: function(jid, name, node, call_back)
+    {
+        if (node && !call_back)
+            return false;
+        this._items.push({jid: jid, name: name, node: node, call_back: call_back});
+        return true;
+    },
+    /** Function: info
+     * Info query
+     *
+     * Parameters:
+     *   (Function) call_back
+     *   (String) jid
+     *   (String) node
+     */
+    info: function(jid, node, success, error, timeout)
+    {
+        var attrs = {xmlns: Strophe.NS.DISCO_INFO};
+        if (node)
+            attrs.node = node;
+
+        var info = $iq({from:this._connection.jid,
+                         to:jid, type:'get'}).c('query', attrs);
+        this._connection.sendIQ(info, success, error, timeout);
+    },
+    /** Function: items
+     * Items query
+     *
+     * Parameters:
+     *   (Function) call_back
+     *   (String) jid
+     *   (String) node
+     */
+    items: function(jid, node, success, error, timeout)
+    {
+        var attrs = {xmlns: Strophe.NS.DISCO_ITEMS};
+        if (node)
+            attrs.node = node;
+
+        var items = $iq({from:this._connection.jid,
+                         to:jid, type:'get'}).c('query', attrs);
+        this._connection.sendIQ(items, success, error, timeout);
+    },
+
+    /** PrivateFunction: _buildIQResult
+     */
+    _buildIQResult: function(stanza, query_attrs)
+    {
+        var id   =  stanza.getAttribute('id');
+        var from = stanza.getAttribute('from');
+        var iqresult = $iq({type: 'result', id: id});
+
+        if (from !== null) {
+            iqresult.attrs({to: from});
+        }
+
+        return iqresult.c('query', query_attrs);
+    },
+
+    /** PrivateFunction: _onDiscoInfo
+     * Called when receive info request
+     */
+    _onDiscoInfo: function(stanza)
+    {
+        var node = stanza.getElementsByTagName('query')[0].getAttribute('node');
+        var attrs = {xmlns: Strophe.NS.DISCO_INFO};
+        if (node)
+        {
+            attrs.node = node;
+        }
+        var iqresult = this._buildIQResult(stanza, attrs);
+        for (var i=0; i<this._identities.length; i++)
+        {
+            var attrs = {category: this._identities[i].category,
+                         type    : this._identities[i].type};
+            if (this._identities[i].name)
+                attrs.name = this._identities[i].name;
+            if (this._identities[i].lang)
+                attrs['xml:lang'] = this._identities[i].lang;
+            iqresult.c('identity', attrs).up();
+        }
+        for (var i=0; i<this._features.length; i++)
+        {
+            iqresult.c('feature', {'var':this._features[i]}).up();
+        }
+        this._connection.send(iqresult.tree());
+        return true;
+    },
+    /** PrivateFunction: _onDiscoItems
+     * Called when receive items request
+     */
+    _onDiscoItems: function(stanza)
+    {
+        var query_attrs = {xmlns: Strophe.NS.DISCO_ITEMS};
+        var node = stanza.getElementsByTagName('query')[0].getAttribute('node');
+        if (node)
+        {
+            query_attrs.node = node;
+            var items = [];
+            for (var i = 0; i < this._items.length; i++)
+            {
+                if (this._items[i].node == node)
+                {
+                    items = this._items[i].call_back(stanza);
+                    break;
+                }
+            }
+        }
+        else
+        {
+            var items = this._items;
+        }
+        var iqresult = this._buildIQResult(stanza, query_attrs);
+        for (var i = 0; i < items.length; i++)
+        {
+            var attrs = {jid:  items[i].jid};
+            if (items[i].name)
+                attrs.name = items[i].name;
+            if (items[i].node)
+                attrs.node = items[i].node;
+            iqresult.c('item', attrs).up();
+        }
+        this._connection.send(iqresult.tree());
+        return true;
+    }
+});
diff --git a/lib/strophe.jingle/strophe.jingle.adapter.js b/lib/strophe.jingle/strophe.jingle.adapter.js
new file mode 100644
index 0000000..6c2ce24
--- /dev/null
+++ b/lib/strophe.jingle/strophe.jingle.adapter.js
@@ -0,0 +1,386 @@
+/* jshint -W117 */
+var setupRTC, getUserMediaWithConstraints, TraceablePeerConnection;
+
+(function($){
+TraceablePeerConnection = function(ice_config, constraints) {
+    var self = this;
+    var RTCPeerconnection = navigator.mozGetUserMedia ? mozRTCPeerConnection : webkitRTCPeerConnection;
+    this.peerconnection = new RTCPeerconnection(ice_config, constraints);
+    this.updateLog = [];
+    this.stats = {};
+    this.statsinterval = null;
+    this.maxstats = 300; // limit to 300 values, i.e. 5 minutes; set to 0 to disable
+
+    // override as desired
+    this.trace = function(what, info) {
+        //console.warn('WTRACE', what, info);
+        self.updateLog.push({
+            time: new Date(),
+            type: what,
+            value: info || ""
+        });
+    };
+    this.onicecandidate = null;
+    this.peerconnection.onicecandidate = function (event) {
+        self.trace('onicecandidate', JSON.stringify(event.candidate, null, ' '));
+        if (self.onicecandidate !== null) {
+            self.onicecandidate(event);
+        }
+    };
+    this.onaddstream = null;
+    this.peerconnection.onaddstream = function (event) {
+        self.trace('onaddstream', event.stream.id);
+        if (self.onaddstream !== null) {
+            self.onaddstream(event);
+        }
+    };
+    this.onremovestream = null;
+    this.peerconnection.onremovestream = function (event) {
+        self.trace('onremovestream', event.stream.id);
+        if (self.onremovestream !== null) {
+            self.onremovestream(event);
+        }
+    };
+    this.onsignalingstatechange = null;
+    this.peerconnection.onsignalingstatechange = function (event) {
+        self.trace('onsignalingstatechange', self.signalingState);
+        if (self.onsignalingstatechange !== null) {
+            self.onsignalingstatechange(event);
+        }
+    };
+    this.oniceconnectionstatechange = null;
+    this.peerconnection.oniceconnectionstatechange = function (event) {
+        self.trace('oniceconnectionstatechange', self.iceConnectionState);
+        if (self.oniceconnectionstatechange !== null) {
+            self.oniceconnectionstatechange(event);
+        }
+    };
+    this.onnegotiationneeded = null;
+    this.peerconnection.onnegotiationneeded = function (event) {
+        self.trace('onnegotiationneeded');
+        if (self.onnegotiationneeded !== null) {
+            self.onnegotiationneeded(event);
+        }
+    };
+    self.ondatachannel = null;
+    this.peerconnection.ondatachannel = function (event) {
+        self.trace('ondatachannel', event);
+        if (self.ondatachannel !== null) {
+            self.ondatachannel(event);
+        }
+    }
+    if (!navigator.mozGetUserMedia) {
+        this.statsinterval = window.setInterval(function() {
+            self.peerconnection.getStats(function(stats) {
+                var results = stats.result();
+                for (var i = 0; i < results.length; ++i) {
+                    //console.log(results[i].type, results[i].id, results[i].names())
+                    var now = new Date();
+                    results[i].names().forEach(function (name) {
+                        var id = results[i].id + '-' + name;
+                        if (!self.stats[id]) {
+                            self.stats[id] = {
+                                startTime: now,
+                                endTime: now,
+                                values: [],
+                                times: []
+                            };
+                        }
+                        self.stats[id].values.push(results[i].stat(name));
+                        self.stats[id].times.push(now.getTime());
+                        if (self.stats[id].values.length > self.maxstats) {
+                            self.stats[id].values.shift();
+                            self.stats[id].times.shift();
+                        }
+                        self.stats[id].endTime = now;
+                    });
+                }
+            });
+
+        }, 1000);
+    }
+};
+
+dumpSDP = function(description) {
+    return 'type: ' + description.type + '\r\n' + description.sdp;
+}
+
+if (TraceablePeerConnection.prototype.__defineGetter__ !== undefined) {
+    TraceablePeerConnection.prototype.__defineGetter__('signalingState', function() { return this.peerconnection.signalingState; });
+    TraceablePeerConnection.prototype.__defineGetter__('iceConnectionState', function() { return this.peerconnection.iceConnectionState; });
+    TraceablePeerConnection.prototype.__defineGetter__('localDescription', function() { return this.peerconnection.localDescription; });
+    TraceablePeerConnection.prototype.__defineGetter__('remoteDescription', function() { return this.peerconnection.remoteDescription; });
+}
+
+TraceablePeerConnection.prototype.addStream = function (stream) {
+    this.trace('addStream', stream.id);
+    this.peerconnection.addStream(stream);
+};
+
+TraceablePeerConnection.prototype.removeStream = function (stream) {
+    this.trace('removeStream', stream.id);
+    this.peerconnection.removeStream(stream);
+};
+
+TraceablePeerConnection.prototype.createDataChannel = function (label, opts) {
+    this.trace('createDataChannel', label, opts);
+    this.peerconnection.createDataChannel(label, opts);
+}
+
+TraceablePeerConnection.prototype.setLocalDescription = function (description, successCallback, failureCallback) {
+    var self = this;
+    this.trace('setLocalDescription', dumpSDP(description));
+    this.peerconnection.setLocalDescription(description, 
+        function () {
+            self.trace('setLocalDescriptionOnSuccess');
+            successCallback();
+        },
+        function (err) {
+            self.trace('setLocalDescriptionOnFailure', err);
+            failureCallback(err);
+        }
+    );
+    /*
+    if (this.statsinterval === null && this.maxstats > 0) {
+        // start gathering stats
+    }
+    */
+};
+
+TraceablePeerConnection.prototype.setRemoteDescription = function (description, successCallback, failureCallback) {
+    var self = this;
+    this.trace('setRemoteDescription', dumpSDP(description));
+    this.peerconnection.setRemoteDescription(description, 
+        function () {
+            self.trace('setRemoteDescriptionOnSuccess');
+            successCallback();
+        },
+        function (err) {
+            self.trace('setRemoteDescriptionOnFailure', err);
+            failureCallback(err);
+        }
+    );
+    /*
+    if (this.statsinterval === null && this.maxstats > 0) {
+        // start gathering stats
+    }
+    */
+};
+
+TraceablePeerConnection.prototype.close = function () {
+    this.trace('stop');
+    if (this.statsinterval !== null) {
+        window.clearInterval(this.statsinterval);
+        this.statsinterval = null;
+    }
+    this.peerconnection.close();
+};
+
+TraceablePeerConnection.prototype.createOffer = function (successCallback, failureCallback, constraints) {
+    var self = this;
+    this.trace('createOffer', JSON.stringify(constraints, null, ' '));
+    this.peerconnection.createOffer(
+        function (offer) {
+            self.trace('createOfferOnSuccess', dumpSDP(offer));
+            successCallback(offer);
+        },
+        function(err) {
+            self.trace('createOfferOnFailure', err);
+            failureCallback(err);
+        },
+        constraints
+    );
+};
+
+TraceablePeerConnection.prototype.createAnswer = function (successCallback, failureCallback, constraints) {
+    var self = this;
+    this.trace('createAnswer', JSON.stringify(constraints, null, ' '));
+    this.peerconnection.createAnswer(
+        function (answer) {
+            self.trace('createAnswerOnSuccess', dumpSDP(answer));
+            successCallback(answer);
+        },
+        function(err) {
+            self.trace('createAnswerOnFailure', err);
+            failureCallback(err);
+        },
+        constraints
+    );
+};
+
+TraceablePeerConnection.prototype.addIceCandidate = function (candidate, successCallback, failureCallback) {
+    var self = this;
+    this.trace('addIceCandidate', JSON.stringify(candidate, null, ' '));
+    this.peerconnection.addIceCandidate(candidate);
+    /* maybe later
+    this.peerconnection.addIceCandidate(candidate, 
+        function () {                                
+            self.trace('addIceCandidateOnSuccess');
+            successCallback();
+        },
+        function (err) {
+            self.trace('addIceCandidateOnFailure', err);
+            failureCallback(err);
+        }
+    );
+    */
+};
+
+TraceablePeerConnection.prototype.getStats = function(callback, errback) {
+    if (navigator.mozGetUserMedia) {
+        // ignore for now...
+    } else {
+        this.peerconnection.getStats(callback);
+    }
+};
+
+// mozilla chrome compat layer -- very similar to adapter.js
+setupRTC = function (){
+    var RTC = null;
+    if (navigator.mozGetUserMedia) {
+        console.log('This appears to be Firefox');
+        var version = parseInt(navigator.userAgent.match(/Firefox\/([0-9]+)\./)[1], 10);
+        if (version >= 22) {
+            RTC = {
+                peerconnection: mozRTCPeerConnection,
+                browser: 'firefox',
+                getUserMedia: navigator.mozGetUserMedia.bind(navigator),
+                attachMediaStream: function (element, stream) {
+                    element[0].mozSrcObject = stream;
+                    element[0].play();
+                },
+                pc_constraints: {}
+            };
+            if (!MediaStream.prototype.getVideoTracks)
+                MediaStream.prototype.getVideoTracks = function () { return []; };
+            if (!MediaStream.prototype.getAudioTracks)
+                MediaStream.prototype.getAudioTracks = function () { return []; };
+            RTCSessionDescription = mozRTCSessionDescription;
+            RTCIceCandidate = mozRTCIceCandidate;
+        }
+    } else if (navigator.webkitGetUserMedia) {
+        console.log('This appears to be Chrome');
+        RTC = {
+            peerconnection: webkitRTCPeerConnection,
+            browser: 'chrome',
+            getUserMedia: navigator.webkitGetUserMedia.bind(navigator),
+            attachMediaStream: function (element, stream) {
+                element.attr('src', webkitURL.createObjectURL(stream));
+            },
+            // DTLS should now be enabled by default but..
+            pc_constraints: {'optional': [{'DtlsSrtpKeyAgreement': 'true'}]} 
+        };
+        if (navigator.userAgent.indexOf('Android') != -1) {
+            RTC.pc_constraints = {}; // disable DTLS on Android
+        }
+        if (!webkitMediaStream.prototype.getVideoTracks) {
+            webkitMediaStream.prototype.getVideoTracks = function () {
+                return this.videoTracks;
+            };
+        }
+        if (!webkitMediaStream.prototype.getAudioTracks) {
+            webkitMediaStream.prototype.getAudioTracks = function () {
+                return this.audioTracks;
+            };
+        }
+    }
+    if (RTC === null) {
+        try { console.log('Browser does not appear to be WebRTC-capable'); } catch (e) { }
+    }
+    return RTC;
+};
+
+getUserMediaWithConstraints = function(um, resolution, bandwidth, fps) {
+    var constraints = {audio: false, video: false};
+
+    if (um.indexOf('video') >= 0) {
+        constraints.video = {mandatory: {}};// same behaviour as true
+    }
+    if (um.indexOf('audio') >= 0) {
+        constraints.audio = {};// same behaviour as true
+    }
+    if (um.indexOf('screen') >= 0) {
+        constraints.video = {
+            "mandatory": {
+                "chromeMediaSource": "screen"
+            }
+        };
+    }
+
+    if (resolution && !constraints.video) {
+        constraints.video = {mandatory: {}};// same behaviour as true
+    }
+    // see https://code.google.com/p/chromium/issues/detail?id=143631#c9 for list of supported resolutions
+    switch (resolution) {
+    // 16:9 first
+    case '1080':
+    case 'fullhd':
+        constraints.video.mandatory.minWidth = 1920;
+        constraints.video.mandatory.minHeight = 1080;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+    case '720':
+    case 'hd':
+        constraints.video.mandatory.minWidth = 1280;
+        constraints.video.mandatory.minHeight = 720;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+    case '360':
+        constraints.video.mandatory.minWidth = 640;
+        constraints.video.mandatory.minHeight = 360;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+    case '180':
+        constraints.video.mandatory.minWidth = 320;
+        constraints.video.mandatory.minHeight = 180;
+        constraints.video.mandatory.minAspectRatio = 1.77;
+        break;
+        // 4:3
+    case '960':
+        constraints.video.mandatory.minWidth = 960;
+        constraints.video.mandatory.minHeight = 720;
+        break;
+    case '640':
+    case 'vga':
+        constraints.video.mandatory.minWidth = 640;
+        constraints.video.mandatory.minHeight = 480;
+        break;
+    case '320':
+        constraints.video.mandatory.minWidth = 320;
+        constraints.video.mandatory.minHeight = 240;
+        break;
+    default:
+        if (navigator.userAgent.indexOf('Android') != -1) {
+            constraints.video.mandatory.minWidth = 320;
+            constraints.video.mandatory.minHeight = 240;
+            constraints.video.mandatory.maxFrameRate = 15;
+        }
+        break;
+    }
+
+    if (bandwidth) { // doesn't work currently, see webrtc issue 1846
+        if (!constraints.video) constraints.video = {mandatory: {}};//same behaviour as true
+        constraints.video.optional = [{bandwidth: bandwidth}];
+    }
+    if (fps) { // for some cameras it might be necessary to request 30fps
+        // so they choose 30fps mjpg over 10fps yuy2
+        if (!constraints.video) constraints.video = {mandatory: {}};// same behaviour as tru;
+        constraints.video.mandatory.minFrameRate = fps;
+    }
+ 
+    try {
+        RTC.getUserMedia(constraints,
+                function (stream) {
+                    console.log('onUserMediaSuccess');
+                    $(document).trigger('mediaready.jingle', [stream]);
+                },
+                function (error) {
+                    console.warn('Failed to get access to local media. Error ', error);
+                    $(document).trigger('mediafailure.jingle', [error]);
+                });
+    } catch (e) {
+        console.error('GUM failed: ', e);
+        $(document).trigger('mediafailure.jingle');
+    }
+}
+}(jQuery));
diff --git a/lib/strophe.jingle/strophe.jingle.js b/lib/strophe.jingle/strophe.jingle.js
new file mode 100644
index 0000000..6b65258
--- /dev/null
+++ b/lib/strophe.jingle/strophe.jingle.js
@@ -0,0 +1,266 @@
+/* jshint -W117 */
+(function($){
+Strophe.addConnectionPlugin('jingle', {
+    connection: null,
+    sessions: {},
+    jid2session: {},
+    ice_config: {iceServers: []},
+    pc_constraints: {},
+    media_constraints: {
+        mandatory: {
+            'OfferToReceiveAudio': true,
+            'OfferToReceiveVideo': true
+        }
+        // MozDontOfferDataChannel: true when this is firefox
+    },
+    localStream: null,
+
+    init: function (conn) {
+        this.connection = conn;
+        if (this.connection.disco) {
+            // http://xmpp.org/extensions/xep-0167.html#support
+            // http://xmpp.org/extensions/xep-0176.html#support
+            this.connection.disco.addFeature('urn:xmpp:jingle:1');
+            this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:1');
+            this.connection.disco.addFeature('urn:xmpp:jingle:transports:ice-udp:1');
+            this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:audio');
+            this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:video');
+
+
+            // this is dealt with by SDP O/A so we don't need to annouce this
+            //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtcp-fb:0'); // XEP-0293
+            //this.connection.disco.addFeature('urn:xmpp:jingle:apps:rtp:rtp-hdrext:0'); // XEP-0294
+            this.connection.disco.addFeature('urn:ietf:rfc:5761'); // rtcp-mux
+            //this.connection.disco.addFeature('urn:ietf:rfc:5888'); // a=group, e.g. bundle
+            //this.connection.disco.addFeature('urn:ietf:rfc:5576'); // a=ssrc
+        }
+        this.connection.addHandler(this.onJingle.bind(this), 'urn:xmpp:jingle:1', 'iq', 'set', null, null);
+    },
+    onJingle: function (iq) {
+        var sid = $(iq).find('jingle').attr('sid');
+        var action = $(iq).find('jingle').attr('action');
+        // send ack first
+        var ack = $iq({type: 'result',
+              to: iq.getAttribute('from'),
+              id: iq.getAttribute('id')
+        });
+        console.log('on jingle ' + action);
+        var sess = this.sessions[sid];
+        if ('session-initiate' != action) {
+            if (sess === null) {
+                ack.type = 'error';
+                ack.c('error', {type: 'cancel'})
+                   .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
+                   .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+                this.connection.send(ack);
+                return true;
+            }
+            // compare from to sess.peerjid (bare jid comparison for later compat with message-mode)
+            // local jid is not checked
+            if (Strophe.getBareJidFromJid(iq.getAttribute('from')) != Strophe.getBareJidFromJid(sess.peerjid)) {
+                console.warn('jid mismatch for session id', sid, iq.getAttribute('from'), sess.peerjid);
+                ack.type = 'error';
+                ack.c('error', {type: 'cancel'})
+                   .c('item-not-found', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up()
+                   .c('unknown-session', {xmlns: 'urn:xmpp:jingle:errors:1'});
+                this.connection.send(ack);
+                return true;
+            }
+        } else if (sess !== undefined) {
+            // existing session with same session id
+            // this might be out-of-order if the sess.peerjid is the same as from
+            ack.type = 'error';
+            ack.c('error', {type: 'cancel'})
+               .c('service-unavailable', {xmlns: 'urn:ietf:params:xml:ns:xmpp-stanzas'}).up();
+            console.warn('duplicate session id', sid);
+            this.connection.send(ack);
+            return true;
+        }
+        // FIXME: check for a defined action
+        this.connection.send(ack);
+        // see http://xmpp.org/extensions/xep-0166.html#concepts-session
+        switch (action) {
+        case 'session-initiate':
+            sess = new JingleSession($(iq).attr('to'), $(iq).find('jingle').attr('sid'), this.connection);
+            // configure session
+            if (this.localStream) {
+                sess.localStreams.push(this.localStream);
+            }
+            sess.media_constraints = this.media_constraints;
+            sess.pc_constraints = this.pc_constraints;
+            sess.ice_config = this.ice_config;
+
+            sess.initiate($(iq).attr('from'), false);
+            sess.setRemoteDescription($(iq).find('>jingle'), 'offer');
+
+            if ($(iq).find('>jingle>muted[xmlns="http://jitsi.org/protocol/meet#startmuted"]').length) {
+                console.log('got a request to start muted');
+                sess.startmuted = true;
+            }
+
+            this.sessions[sess.sid] = sess;
+            this.jid2session[sess.peerjid] = sess;
+
+            // the callback should either 
+            // .sendAnswer and .accept
+            // or .sendTerminate -- not necessarily synchronus
+            $(document).trigger('callincoming.jingle', [sess.sid]);
+            break;
+        case 'session-accept':
+            sess.setRemoteDescription($(iq).find('>jingle'), 'answer');
+            sess.accept();
+            $(document).trigger('callaccepted.jingle', [sess.sid]);
+            break;
+        case 'session-terminate':
+            console.log('terminating...');
+            sess.terminate();
+            this.terminate(sess.sid);
+            if ($(iq).find('>jingle>reason').length) {
+                $(document).trigger('callterminated.jingle', [
+                    sess.sid,
+                    $(iq).find('>jingle>reason>:first')[0].tagName,
+                    $(iq).find('>jingle>reason>text').text()
+                ]);
+            } else {
+                $(document).trigger('callterminated.jingle', [sess.sid]);
+            }
+            break;
+        case 'transport-info':
+            sess.addIceCandidate($(iq).find('>jingle>content'));
+            break;
+        case 'session-info':
+            var affected;
+            if ($(iq).find('>jingle>ringing[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+                $(document).trigger('ringing.jingle', [sess.sid]);
+            } else if ($(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+                affected = $(iq).find('>jingle>mute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+                $(document).trigger('mute.jingle', [sess.sid, affected]);
+            } else if ($(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').length) {
+                affected = $(iq).find('>jingle>unmute[xmlns="urn:xmpp:jingle:apps:rtp:info:1"]').attr('name');
+                $(document).trigger('unmute.jingle', [sess.sid, affected]);
+            }
+            break;
+        case 'addsource': // FIXME: proprietary
+            sess.addSource($(iq).find('>jingle>content'));
+            break;
+        case 'removesource': // FIXME: proprietary
+            sess.removeSource($(iq).find('>jingle>content'));
+            break;
+        default:
+            console.warn('jingle action not implemented', action);
+            break;
+        }
+        return true;
+    },
+    initiate: function (peerjid, myjid) { // initiate a new jinglesession to peerjid
+        var sess = new JingleSession(myjid || this.connection.jid,
+                                     Math.random().toString(36).substr(2, 12), // random string
+                                     this.connection);
+        // configure session
+        if (this.localStream) {
+            sess.localStreams.push(this.localStream);
+        }
+        sess.media_constraints = this.media_constraints;
+        sess.pc_constraints = this.pc_constraints;
+        sess.ice_config = this.ice_config;
+
+        sess.initiate(peerjid, true);
+        this.sessions[sess.sid] = sess;
+        this.jid2session[sess.peerjid] = sess;
+        sess.sendOffer();
+        return sess;
+    },
+    terminate: function (sid, reason, text) { // terminate by sessionid (or all sessions)
+        if (sid === null || sid === undefined) {
+            for (sid in this.sessions) {
+                if (this.sessions[sid].state != 'ended') {
+                    this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+                    this.sessions[sid].terminate();
+                }
+                delete this.jid2session[this.sessions[sid].peerjid];
+                delete this.sessions[sid];
+            }
+        } else if (this.sessions.hasOwnProperty(sid)) {
+            if (this.sessions[sid].state != 'ended') {
+                this.sessions[sid].sendTerminate(reason || (!this.sessions[sid].active()) ? 'cancel' : null, text);
+                this.sessions[sid].terminate();
+            }
+            delete this.jid2session[this.sessions[sid].peerjid];
+            delete this.sessions[sid];
+        }
+    },
+    terminateByJid: function (jid) {
+        if (this.jid2session.hasOwnProperty(jid)) {
+            var sess = this.jid2session[jid];
+            if (sess) {
+                sess.terminate();
+                console.log('peer went away silently', jid);
+                delete this.sessions[sess.sid];
+                delete this.jid2session[jid];
+                $(document).trigger('callterminated.jingle', [sess.sid, 'gone']);
+            }
+        }
+    },
+    getStunAndTurnCredentials: function () {
+        // get stun and turn configuration from server via xep-0215
+        // uses time-limited credentials as described in
+        // http://tools.ietf.org/html/draft-uberti-behave-turn-rest-00
+        //
+        // see https://code.google.com/p/prosody-modules/source/browse/mod_turncredentials/mod_turncredentials.lua
+        // for a prosody module which implements this
+        //
+        // currently, this doesn't work with updateIce and therefore credentials with a long
+        // validity have to be fetched before creating the peerconnection
+        // TODO: implement refresh via updateIce as described in
+        //      https://code.google.com/p/webrtc/issues/detail?id=1650
+        var self = this;
+        this.connection.sendIQ(
+            $iq({type: 'get', to: this.connection.domain})
+                .c('services', {xmlns: 'urn:xmpp:extdisco:1'}).c('service', {host: 'turn.' + this.connection.domain}),
+            function (res) {
+                var iceservers = [];
+                $(res).find('>services>service').each(function (idx, el) {
+                    el = $(el);
+                    var dict = {};
+                    switch (el.attr('type')) {
+                    case 'stun':
+                        dict.url = 'stun:' + el.attr('host');
+                        if (el.attr('port')) {
+                            dict.url += ':' + el.attr('port');
+                        }
+                        iceservers.push(dict);
+                        break;
+                    case 'turn':
+                        dict.url = 'turn:';
+                        if (el.attr('username')) { // https://code.google.com/p/webrtc/issues/detail?id=1508
+                            if (navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./) && parseInt(navigator.userAgent.match(/Chrom(e|ium)\/([0-9]+)\./)[2], 10) < 28) {
+                                dict.url += el.attr('username') + '@';
+                            } else {
+                                dict.username = el.attr('username'); // only works in M28
+                            }
+                        }
+                        dict.url += el.attr('host');
+                        if (el.attr('port') && el.attr('port') != '3478') {
+                            dict.url += ':' + el.attr('port');
+                        }
+                        if (el.attr('transport') && el.attr('transport') != 'udp') {
+                            dict.url += '?transport=' + el.attr('transport');
+                        }
+                        if (el.attr('password')) {
+                            dict.credential = el.attr('password');
+                        }
+                        iceservers.push(dict);
+                        break;
+                    }
+                });
+                self.ice_config.iceServers = iceservers;
+            },
+            function (err) {
+                console.warn('getting turn credentials failed', err);
+                console.warn('is mod_turncredentials or similar installed?');
+            }
+        );
+        // implement push?
+    }
+});
+}(jQuery));
diff --git a/lib/strophe.jingle/strophe.jingle.sdp.js b/lib/strophe.jingle/strophe.jingle.sdp.js
new file mode 100644
index 0000000..140a52a
--- /dev/null
+++ b/lib/strophe.jingle/strophe.jingle.sdp.js
@@ -0,0 +1,822 @@
+/* jshint -W117 */
+var SDP;
+
+(function($){
+// SDP STUFF
+SDP = function(sdp) {
+    this.media = sdp.split('\r\nm=');
+    for (var i = 1; i < this.media.length; i++) {
+        this.media[i] = 'm=' + this.media[i];
+        if (i != this.media.length - 1) {
+            this.media[i] += '\r\n';
+        }
+    }
+    this.session = this.media.shift() + '\r\n';
+    this.raw = this.session + this.media.join('');
+}
+
+// remove iSAC and CN from SDP
+SDP.prototype.mangle = function () {
+    var i, j, mline, lines, rtpmap, newdesc;
+    for (i = 0; i < this.media.length; i++) {
+        lines = this.media[i].split('\r\n');
+        lines.pop(); // remove empty last element
+        mline = SDPUtil.parse_mline(lines.shift());
+        if (mline.media != 'audio')
+            continue;
+        newdesc = '';
+        mline.fmt.length = 0;
+        for (j = 0; j < lines.length; j++) {
+            if (lines[j].substr(0, 9) == 'a=rtpmap:') {
+                rtpmap = SDPUtil.parse_rtpmap(lines[j]);
+                if (rtpmap.name == 'CN' || rtpmap.name == 'ISAC')
+                    continue;
+                mline.fmt.push(rtpmap.id);
+                newdesc += lines[j] + '\r\n';
+            } else {
+                newdesc += lines[j] + '\r\n';
+            }
+        }
+        this.media[i] = SDPUtil.build_mline(mline) + '\r\n';
+        this.media[i] += newdesc;
+    }
+    this.raw = this.session + this.media.join('');
+};
+
+// remove lines matching prefix from session section
+SDP.prototype.removeSessionLines = function(prefix) {
+    var self = this;
+    var lines = SDPUtil.find_lines(this.session, prefix);
+    lines.forEach(function(line) {
+        self.session = self.session.replace(line + '\r\n', '');
+    });
+    this.raw = this.session + this.media.join('');
+    return lines;
+}
+// remove lines matching prefix from a media section specified by mediaindex
+// TODO: non-numeric mediaindex could match mid
+SDP.prototype.removeMediaLines = function(mediaindex, prefix) {
+    var self = this;
+    var lines = SDPUtil.find_lines(this.media[mediaindex], prefix);
+    lines.forEach(function(line) {
+        self.media[mediaindex] = self.media[mediaindex].replace(line + '\r\n', '');
+    });
+    this.raw = this.session + this.media.join('');
+    return lines;
+}
+
+// add content's to a jingle element
+SDP.prototype.toJingle = function (elem, thecreator) {
+    var i, j, k, mline, ssrc, rtpmap, tmp, line, lines;
+    var self = this;
+    // new bundle plan
+    if (SDPUtil.find_line(this.session, 'a=group:')) {
+        lines = SDPUtil.find_lines(this.session, 'a=group:');
+        for (i = 0; i < lines.length; i++) {
+            tmp = lines[i].split(' ');
+            var semantics = tmp.shift().substr(8);
+            elem.c('group', {xmlns: 'urn:xmpp:jingle:apps:grouping:0', semantics:semantics});
+            for (j = 0; j < tmp.length; j++) {
+                elem.c('content', {name: tmp[j]}).up();
+            }
+            elem.up();
+        }
+    }
+    // old bundle plan, to be removed
+    var bundle = [];
+    if (SDPUtil.find_line(this.session, 'a=group:BUNDLE')) {
+        bundle = SDPUtil.find_line(this.session, 'a=group:BUNDLE ').split(' ');
+        bundle.shift();
+    }
+    for (i = 0; i < this.media.length; i++) {
+        mline = SDPUtil.parse_mline(this.media[i].split('\r\n')[0]);
+        if (!(mline.media == 'audio' || mline.media == 'video')) {
+            continue;
+        }
+        if (SDPUtil.find_line(this.media[i], 'a=ssrc:')) {
+            ssrc = SDPUtil.find_line(this.media[i], 'a=ssrc:').substring(7).split(' ')[0]; // take the first
+        } else {
+            ssrc = false;
+        }
+
+        elem.c('content', {creator: thecreator, name: mline.media});
+        if (SDPUtil.find_line(this.media[i], 'a=mid:')) {
+            // prefer identifier from a=mid if present
+            var mid = SDPUtil.parse_mid(SDPUtil.find_line(this.media[i], 'a=mid:'));
+            elem.attrs({ name: mid });
+
+            // old BUNDLE plan, to be removed
+            if (bundle.indexOf(mid) != -1) {
+                elem.c('bundle', {xmlns: 'http://estos.de/ns/bundle'}).up();
+                bundle.splice(bundle.indexOf(mid), 1);
+            }
+        }
+        if (SDPUtil.find_line(this.media[i], 'a=rtpmap:').length) {
+            elem.c('description',
+                 {xmlns: 'urn:xmpp:jingle:apps:rtp:1',
+                  media: mline.media });
+            if (ssrc) {
+                elem.attrs({ssrc: ssrc});
+            }
+            for (j = 0; j < mline.fmt.length; j++) {
+                rtpmap = SDPUtil.find_line(this.media[i], 'a=rtpmap:' + mline.fmt[j]);
+                elem.c('payload-type', SDPUtil.parse_rtpmap(rtpmap));
+                // put any 'a=fmtp:' + mline.fmt[j] lines into <param name=foo value=bar/>
+                if (SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j])) {
+                    tmp = SDPUtil.parse_fmtp(SDPUtil.find_line(this.media[i], 'a=fmtp:' + mline.fmt[j]));
+                    for (k = 0; k < tmp.length; k++) {
+                        elem.c('parameter', tmp[k]).up();
+                    }
+                }
+                this.RtcpFbToJingle(i, elem, mline.fmt[j]); // XEP-0293 -- map a=rtcp-fb
+
+                elem.up();
+            }
+            if (SDPUtil.find_line(this.media[i], 'a=crypto:', this.session)) {
+                elem.c('encryption', {required: 1});
+                var crypto = SDPUtil.find_lines(this.media[i], 'a=crypto:', this.session);
+                crypto.forEach(function(line) {
+                    elem.c('crypto', SDPUtil.parse_crypto(line)).up();
+                });
+                elem.up(); // end of encryption
+            }
+
+            if (ssrc) {
+                // new style mapping
+                elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+                // FIXME: group by ssrc and support multiple different ssrcs
+                var ssrclines = SDPUtil.find_lines(this.media[i], 'a=ssrc:');
+                ssrclines.forEach(function(line) {
+                    idx = line.indexOf(' ');
+                    var linessrc = line.substr(0, idx).substr(7);
+                    if (linessrc != ssrc) {
+                        elem.up();
+                        ssrc = linessrc;
+                        elem.c('source', { ssrc: ssrc, xmlns: 'urn:xmpp:jingle:apps:rtp:ssma:0' });
+                    }
+                    var kv = line.substr(idx + 1);
+                    elem.c('parameter');
+                    if (kv.indexOf(':') == -1) {
+                        elem.attrs({ name: kv });
+                    } else {
+                        elem.attrs({ name: kv.split(':', 2)[0] });
+                        elem.attrs({ value: kv.split(':', 2)[1] });
+                    }
+                    elem.up();
+                });
+                elem.up();
+
+                // old proprietary mapping, to be removed at some point
+                tmp = SDPUtil.parse_ssrc(this.media[i]);
+                tmp.xmlns = 'http://estos.de/ns/ssrc';
+                tmp.ssrc = ssrc;
+                elem.c('ssrc', tmp).up(); // ssrc is part of description
+            }
+
+            if (SDPUtil.find_line(this.media[i], 'a=rtcp-mux')) {
+                elem.c('rtcp-mux').up();
+            }
+
+            // XEP-0293 -- map a=rtcp-fb:*
+            this.RtcpFbToJingle(i, elem, '*');
+
+            // XEP-0294
+            if (SDPUtil.find_line(this.media[i], 'a=extmap:')) {
+                lines = SDPUtil.find_lines(this.media[i], 'a=extmap:');
+                for (j = 0; j < lines.length; j++) {
+                    tmp = SDPUtil.parse_extmap(lines[j]);
+                    elem.c('rtp-hdrext', { xmlns: 'urn:xmpp:jingle:apps:rtp:rtp-hdrext:0',
+                                    uri: tmp.uri,
+                                    id: tmp.value });
+                    if (tmp.hasOwnProperty('direction')) {
+                        switch (tmp.direction) {
+                        case 'sendonly':
+                            elem.attrs({senders: 'responder'});
+                            break;
+                        case 'recvonly':
+                            elem.attrs({senders: 'initiator'});
+                            break;
+                        case 'sendrecv':
+                            elem.attrs({senders: 'both'});
+                            break;
+                        case 'inactive':
+                            elem.attrs({senders: 'none'});
+                            break;
+                        }
+                    }
+                    // TODO: handle params
+                    elem.up();
+                }
+            }
+            elem.up(); // end of description
+        }
+
+        // map ice-ufrag/pwd, dtls fingerprint, candidates
+        this.TransportToJingle(i, elem);
+
+        if (SDPUtil.find_line(this.media[i], 'a=sendrecv', this.session)) {
+            elem.attrs({senders: 'both'});
+        } else if (SDPUtil.find_line(this.media[i], 'a=sendonly', this.session)) {
+            elem.attrs({senders: 'initiator'});
+        } else if (SDPUtil.find_line(this.media[i], 'a=recvonly', this.session)) {
+            elem.attrs({senders: 'responder'});
+        } else if (SDPUtil.find_line(this.media[i], 'a=inactive', this.session)) {
+            elem.attrs({senders: 'none'});
+        }
+        if (mline.port == '0') {
+            // estos hack to reject an m-line
+            elem.attrs({senders: 'rejected'});
+        }
+        elem.up(); // end of content
+    }
+    elem.up();
+    return elem;
+};
+
+SDP.prototype.TransportToJingle = function (mediaindex, elem) {
+    var i = mediaindex;
+    var tmp;
+    var self = this;
+    elem.c('transport');
+
+    // XEP-0320
+    var fingerprints = SDPUtil.find_lines(this.media[mediaindex], 'a=fingerprint:', this.session);
+    fingerprints.forEach(function(line) {
+        tmp = SDPUtil.parse_fingerprint(line);
+        tmp.xmlns = 'urn:xmpp:tmp:jingle:apps:dtls:0';
+        // tmp.xmlns = 'urn:xmpp:jingle:apps:dtls:0'; -- FIXME: update receivers first
+        elem.c('fingerprint').t(tmp.fingerprint);
+        delete tmp.fingerprint;
+        line = SDPUtil.find_line(self.media[mediaindex], 'a=setup:', self.session);
+        if (line) {
+            tmp.setup = line.substr(8);
+        }
+        elem.attrs(tmp);
+        elem.up(); // end of fingerprint
+    });
+    tmp = SDPUtil.iceparams(this.media[mediaindex], this.session);
+    if (tmp) {
+        tmp.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+        elem.attrs(tmp);
+        // XEP-0176
+        if (SDPUtil.find_line(this.media[mediaindex], 'a=candidate:', this.session)) { // add any a=candidate lines
+            var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=candidate:', this.session);
+            lines.forEach(function (line) {
+                elem.c('candidate', SDPUtil.candidateToJingle(line)).up();
+            });
+        }
+    }
+    elem.up(); // end of transport
+}
+
+SDP.prototype.RtcpFbToJingle = function (mediaindex, elem, payloadtype) { // XEP-0293
+    var lines = SDPUtil.find_lines(this.media[mediaindex], 'a=rtcp-fb:' + payloadtype);
+    lines.forEach(function (line) {
+        var tmp = SDPUtil.parse_rtcpfb(line);
+        if (tmp.type == 'trr-int') {
+            elem.c('rtcp-fb-trr-int', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', value: tmp.params[0]});
+            elem.up();
+        } else {
+            elem.c('rtcp-fb', {xmlns: 'urn:xmpp:jingle:apps:rtp:rtcp-fb:0', type: tmp.type});
+            if (tmp.params.length > 0) {
+                elem.attrs({'subtype': tmp.params[0]});
+            }
+            elem.up();
+        }
+    });
+};
+
+SDP.prototype.RtcpFbFromJingle = function (elem, payloadtype) { // XEP-0293
+    var media = '';
+    var tmp = elem.find('>rtcp-fb-trr-int[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
+    if (tmp.length) {
+        media += 'a=rtcp-fb:' + '*' + ' ' + 'trr-int' + ' ';
+        if (tmp.attr('value')) {
+            media += tmp.attr('value');
+        } else {
+            media += '0';
+        }
+        media += '\r\n';
+    }
+    tmp = elem.find('>rtcp-fb[xmlns="urn:xmpp:jingle:apps:rtp:rtcp-fb:0"]');
+    tmp.each(function () {
+        media += 'a=rtcp-fb:' + payloadtype + ' ' + $(this).attr('type');
+        if ($(this).attr('subtype')) {
+            media += ' ' + $(this).attr('subtype');
+        }
+        media += '\r\n';
+    });
+    return media;
+};
+
+// construct an SDP from a jingle stanza
+SDP.prototype.fromJingle = function (jingle) {
+    var self = this;
+    this.raw = 'v=0\r\n' +
+        'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
+        's=-\r\n' +
+        't=0 0\r\n';
+    // http://tools.ietf.org/html/draft-ietf-mmusic-sdp-bundle-negotiation-04#section-8
+    if ($(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').length) {
+        $(jingle).find('>group[xmlns="urn:xmpp:jingle:apps:grouping:0"]').each(function (idx, group) {
+            var contents = $(group).find('>content').map(function (idx, content) {
+                return content.getAttribute('name');
+            }).get();
+            if (contents.length > 0) {
+                self.raw += 'a=group:' + (group.getAttribute('semantics') || group.getAttribute('type')) + ' ' + contents.join(' ') + '\r\n';
+            }
+        });
+    } else if ($(jingle).find('>group[xmlns="urn:ietf:rfc:5888"]').length) {
+        // temporary namespace, not to be used. to be removed soon.
+        $(jingle).find('>group[xmlns="urn:ietf:rfc:5888"]').each(function (idx, group) {
+            var contents = $(group).find('>content').map(function (idx, content) {
+                return content.getAttribute('name');
+            }).get();
+            if (group.getAttribute('type') !== null && contents.length > 0) {
+                self.raw += 'a=group:' + group.getAttribute('type') + ' ' + contents.join(' ') + '\r\n';
+            }
+        });
+    } else {
+        // for backward compability, to be removed soon
+        // assume all contents are in the same bundle group, can be improved upon later
+        var bundle = $(jingle).find('>content').filter(function (idx, content) {
+            //elem.c('bundle', {xmlns:'http://estos.de/ns/bundle'});
+            return $(content).find('>bundle').length > 0;
+        }).map(function (idx, content) {
+            return content.getAttribute('name');
+        }).get();
+        if (bundle.length) {
+            this.raw += 'a=group:BUNDLE ' + bundle.join(' ') + '\r\n';
+        }
+    }
+
+    this.session = this.raw;
+    jingle.find('>content').each(function () {
+        var m = self.jingle2media($(this));
+        self.media.push(m);
+    });
+
+    // reconstruct msid-semantic -- apparently not necessary
+    /*
+    var msid = SDPUtil.parse_ssrc(this.raw);
+    if (msid.hasOwnProperty('mslabel')) {
+        this.session += "a=msid-semantic: WMS " + msid.mslabel + "\r\n";
+    }
+    */
+
+    this.raw = this.session + this.media.join('');
+};
+
+// translate a jingle content element into an an SDP media part
+SDP.prototype.jingle2media = function (content) {
+    var media = '',
+        desc = content.find('description'),
+        ssrc = desc.attr('ssrc'),
+        self = this,
+        tmp;
+
+    tmp = { media: desc.attr('media') };
+    tmp.port = '1';
+    if (content.attr('senders') == 'rejected') {
+        // estos hack to reject an m-line.
+        tmp.port = '0';
+    }
+    if (content.find('>transport>fingerprint').length || desc.find('encryption').length) {
+        tmp.proto = 'RTP/SAVPF';
+    } else {
+        tmp.proto = 'RTP/AVPF';
+    }
+    tmp.fmt = desc.find('payload-type').map(function () { return this.getAttribute('id'); }).get();
+    media += SDPUtil.build_mline(tmp) + '\r\n';
+    media += 'c=IN IP4 0.0.0.0\r\n';
+    media += 'a=rtcp:1 IN IP4 0.0.0.0\r\n';
+    tmp = content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]');
+    if (tmp.length) {
+        if (tmp.attr('ufrag')) {
+            media += SDPUtil.build_iceufrag(tmp.attr('ufrag')) + '\r\n';
+        }
+        if (tmp.attr('pwd')) {
+            media += SDPUtil.build_icepwd(tmp.attr('pwd')) + '\r\n';
+        }
+        tmp.find('>fingerprint').each(function () {
+            // FIXME: check namespace at some point
+            media += 'a=fingerprint:' + this.getAttribute('hash');
+            media += ' ' + $(this).text();
+            media += '\r\n';
+            if (this.getAttribute('setup')) {
+                media += 'a=setup:' + this.getAttribute('setup') + '\r\n';
+            }
+        });
+    }
+    switch (content.attr('senders')) {
+    case 'initiator':
+        media += 'a=sendonly\r\n';
+        break;
+    case 'responder':
+        media += 'a=recvonly\r\n';
+        break;
+    case 'none':
+        media += 'a=inactive\r\n';
+        break;
+    case 'both':
+        media += 'a=sendrecv\r\n';
+        break;
+    }
+    media += 'a=mid:' + content.attr('name') + '\r\n';
+
+    // <description><rtcp-mux/></description>
+    // see http://code.google.com/p/libjingle/issues/detail?id=309 -- no spec though
+    // and http://mail.jabber.org/pipermail/jingle/2011-December/001761.html
+    if (desc.find('rtcp-mux').length) {
+        media += 'a=rtcp-mux\r\n';
+    }
+
+    if (desc.find('encryption').length) {
+        desc.find('encryption>crypto').each(function () {
+            media += 'a=crypto:' + this.getAttribute('tag');
+            media += ' ' + this.getAttribute('crypto-suite');
+            media += ' ' + this.getAttribute('key-params');
+            if (this.getAttribute('session-params')) {
+                media += ' ' + this.getAttribute('session-params');
+            }
+            media += '\r\n';
+        });
+    }
+    desc.find('payload-type').each(function () {
+        media += SDPUtil.build_rtpmap(this) + '\r\n';
+        if ($(this).find('>parameter').length) {
+            media += 'a=fmtp:' + this.getAttribute('id') + ' ';
+            media += $(this).find('parameter').map(function () { return (this.getAttribute('name') ? (this.getAttribute('name') + '=') : '') + this.getAttribute('value'); }).get().join(';');
+            media += '\r\n';
+        }
+        // xep-0293
+        media += self.RtcpFbFromJingle($(this), this.getAttribute('id'));
+    });
+
+    // xep-0293
+    media += self.RtcpFbFromJingle(desc, '*');
+
+    // xep-0294
+    tmp = desc.find('>rtp-hdrext[xmlns="urn:xmpp:jingle:apps:rtp:rtp-hdrext:0"]');
+    tmp.each(function () {
+        media += 'a=extmap:' + this.getAttribute('id') + ' ' + this.getAttribute('uri') + '\r\n';
+    });
+
+    content.find('>transport[xmlns="urn:xmpp:jingle:transports:ice-udp:1"]>candidate').each(function () {
+        media += SDPUtil.candidateFromJingle(this);
+    });
+
+    tmp = content.find('description>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+    tmp.each(function () {
+        var ssrc = this.getAttribute('ssrc');
+        $(this).find('>parameter').each(function () {
+            media += 'a=ssrc:' + ssrc + ' ' + this.getAttribute('name');
+            if (this.getAttribute('value') && this.getAttribute('value').length)
+                media += ':' + this.getAttribute('value');
+            media += '\r\n';
+        });
+    });
+
+    if (tmp.length === 0) {
+        // fallback to proprietary mapping of a=ssrc lines
+        tmp = content.find('description>ssrc[xmlns="http://estos.de/ns/ssrc"]');
+        if (tmp.length) {
+            media += 'a=ssrc:' + ssrc + ' cname:' + tmp.attr('cname') + '\r\n';
+            media += 'a=ssrc:' + ssrc + ' msid:' + tmp.attr('msid') + '\r\n';
+            media += 'a=ssrc:' + ssrc + ' mslabel:' + tmp.attr('mslabel') + '\r\n';
+            media += 'a=ssrc:' + ssrc + ' label:' + tmp.attr('label') + '\r\n';
+        }
+    }
+    return media;
+};
+
+SDPUtil = {
+    iceparams: function (mediadesc, sessiondesc) {
+        var data = null;
+        if (SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc) &&
+            SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc)) {
+            data = {
+                ufrag: SDPUtil.parse_iceufrag(SDPUtil.find_line(mediadesc, 'a=ice-ufrag:', sessiondesc)),
+                pwd: SDPUtil.parse_icepwd(SDPUtil.find_line(mediadesc, 'a=ice-pwd:', sessiondesc))
+            };
+        }
+        return data;
+    },
+    parse_iceufrag: function (line) {
+        return line.substring(12);
+    },
+    build_iceufrag: function (frag) {
+        return 'a=ice-ufrag:' + frag;
+    },
+    parse_icepwd: function (line) {
+        return line.substring(10);
+    },
+    build_icepwd: function (pwd) {
+        return 'a=ice-pwd:' + pwd;
+    },
+    parse_mid: function (line) {
+        return line.substring(6);
+    },
+    parse_mline: function (line) {
+        var parts = line.substring(2).split(' '),
+        data = {};
+        data.media = parts.shift();
+        data.port = parts.shift();
+        data.proto = parts.shift();
+        if (parts[parts.length - 1] === '') { // trailing whitespace
+            parts.pop();
+        }
+        data.fmt = parts;
+        return data;
+    },
+    build_mline: function (mline) {
+        return 'm=' + mline.media + ' ' + mline.port + ' ' + mline.proto + ' ' + mline.fmt.join(' ');
+    },
+    parse_rtpmap: function (line) {
+        var parts = line.substring(9).split(' '),
+            data = {};
+        data.id = parts.shift();
+        parts = parts[0].split('/');
+        data.name = parts.shift();
+        data.clockrate = parts.shift();
+        data.channels = parts.length ? parts.shift() : '1';
+        return data;
+    },
+    build_rtpmap: function (el) {
+        var line = 'a=rtpmap:' + el.getAttribute('id') + ' ' + el.getAttribute('name') + '/' + el.getAttribute('clockrate');
+        if (el.getAttribute('channels') && el.getAttribute('channels') != '1') {
+            line += '/' + el.getAttribute('channels');
+        }
+        return line;
+    },
+    parse_crypto: function (line) {
+        var parts = line.substring(9).split(' '),
+        data = {};
+        data.tag = parts.shift();
+        data['crypto-suite'] = parts.shift();
+        data['key-params'] = parts.shift();
+        if (parts.length) {
+            data['session-params'] = parts.join(' ');
+        }
+        return data;
+    },
+    parse_fingerprint: function (line) { // RFC 4572
+        var parts = line.substring(14).split(' '),
+        data = {};
+        data.hash = parts.shift();
+        data.fingerprint = parts.shift();
+        // TODO assert that fingerprint satisfies 2UHEX *(":" 2UHEX) ?
+        return data;
+    },
+    parse_fmtp: function (line) {
+        var parts = line.split(' '),
+            i, key, value,
+            data = [];
+        parts.shift();
+        parts = parts.join(' ').split(';');
+        for (i = 0; i < parts.length; i++) {
+            key = parts[i].split('=')[0];
+            while (key.length && key[0] == ' ') {
+                key = key.substring(1);
+            }
+            value = parts[i].split('=')[1];
+            if (key && value) {
+                data.push({name: key, value: value});
+            } else if (key) {
+                // rfc 4733 (DTMF) style stuff
+                data.push({name: '', value: key});
+            }
+        }
+        return data;
+    },
+    parse_icecandidate: function (line) {
+        var candidate = {},
+            elems = line.split(' ');
+        candidate.foundation = elems[0].substring(12);
+        candidate.component = elems[1];
+        candidate.protocol = elems[2].toLowerCase();
+        candidate.priority = elems[3];
+        candidate.ip = elems[4];
+        candidate.port = elems[5];
+        // elems[6] => "typ"
+        candidate.type = elems[7];
+        candidate.generation = 0; // default value, may be overwritten below
+        for (var i = 8; i < elems.length; i += 2) {
+            switch (elems[i]) {
+            case 'raddr':
+                candidate['rel-addr'] = elems[i + 1];
+                break;
+            case 'rport':
+                candidate['rel-port'] = elems[i + 1];
+                break;
+            case 'generation':
+                candidate.generation = elems[i + 1];
+                break;
+            case 'tcptype':
+                candidate.tcptype = elems[i + 1];
+                break;
+            default: // TODO
+                console.log('parse_icecandidate not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+            }
+        }
+        candidate.network = '1';
+        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+        return candidate;
+    },
+    build_icecandidate: function (cand) {
+        var line = ['a=candidate:' + cand.foundation, cand.component, cand.protocol, cand.priority, cand.ip, cand.port, 'typ', cand.type].join(' ');
+        line += ' ';
+        switch (cand.type) {
+        case 'srflx':
+        case 'prflx':
+        case 'relay':
+            if (cand.hasOwnAttribute('rel-addr') && cand.hasOwnAttribute('rel-port')) {
+                line += 'raddr';
+                line += ' ';
+                line += cand['rel-addr'];
+                line += ' ';
+                line += 'rport';
+                line += ' ';
+                line += cand['rel-port'];
+                line += ' ';
+            }
+            break;
+        }
+        if (cand.hasOwnAttribute('tcptype')) {
+            line += 'tcptype';
+            line += ' ';
+            line += cand.tcptype;
+            line += ' ';
+        }
+        line += 'generation';
+        line += ' ';
+        line += cand.hasOwnAttribute('generation') ? cand.generation : '0';
+        return line;
+    },
+    parse_ssrc: function (desc) {
+        // proprietary mapping of a=ssrc lines
+        // TODO: see "Jingle RTP Source Description" by Juberti and P. Thatcher on google docs
+        // and parse according to that
+        var lines = desc.split('\r\n'),
+            data = {};
+        for (var i = 0; i < lines.length; i++) {
+            if (lines[i].substring(0, 7) == 'a=ssrc:') {
+                var idx = lines[i].indexOf(' ');
+                data[lines[i].substr(idx + 1).split(':', 2)[0]] = lines[i].substr(idx + 1).split(':', 2)[1];
+            }
+        }
+        return data;
+    },
+    parse_rtcpfb: function (line) {
+        var parts = line.substr(10).split(' ');
+        var data = {};
+        data.pt = parts.shift();
+        data.type = parts.shift();
+        data.params = parts;
+        return data;
+    },
+    parse_extmap: function (line) {
+        var parts = line.substr(9).split(' ');
+        var data = {};
+        data.value = parts.shift();
+        if (data.value.indexOf('/') != -1) {
+            data.direction = data.value.substr(data.value.indexOf('/') + 1);
+            data.value = data.value.substr(0, data.value.indexOf('/'));
+        } else {
+            data.direction = 'both';
+        }
+        data.uri = parts.shift();
+        data.params = parts;
+        return data;
+    },
+    find_line: function (haystack, needle, sessionpart) {
+        var lines = haystack.split('\r\n');
+        for (var i = 0; i < lines.length; i++) {
+            if (lines[i].substring(0, needle.length) == needle) {
+                return lines[i];
+            }
+        }
+        if (!sessionpart) {
+            return false;
+        }
+        // search session part
+        lines = sessionpart.split('\r\n');
+        for (var j = 0; j < lines.length; j++) {
+            if (lines[j].substring(0, needle.length) == needle) {
+                return lines[j];
+            }
+        }
+        return false;
+    },
+    find_lines: function (haystack, needle, sessionpart) {
+        var lines = haystack.split('\r\n'),
+            needles = [];
+        for (var i = 0; i < lines.length; i++) {
+            if (lines[i].substring(0, needle.length) == needle)
+                needles.push(lines[i]);
+        }
+        if (needles.length || !sessionpart) {
+            return needles;
+        }
+        // search session part
+        lines = sessionpart.split('\r\n');
+        for (var j = 0; j < lines.length; j++) {
+            if (lines[j].substring(0, needle.length) == needle) {
+                needles.push(lines[j]);
+            }
+        }
+        return needles;
+    },
+    candidateToJingle: function (line) {
+        // a=candidate:2979166662 1 udp 2113937151 192.168.2.100 57698 typ host generation 0
+        //      <candidate component=... foundation=... generation=... id=... ip=... network=... port=... priority=... protocol=... type=.../>
+        if (line.indexOf('candidate:') === 0) {
+            line = 'a=' + line;
+        } else if (line.substring(0, 12) != 'a=candidate:') {            
+			console.log('parseCandidate called with a line that is not a candidate line');
+            console.log(line);
+            return null;
+        }
+        if (line.substring(line.length - 2) == '\r\n') // chomp it
+            line = line.substring(0, line.length - 2);
+        var candidate = {},
+            elems = line.split(' '),
+            i;
+        if (elems[6] != 'typ') {
+            console.log('did not find typ in the right place');
+            console.log(line);
+            return null;
+        }
+        candidate.foundation = elems[0].substring(12);
+        candidate.component = elems[1];
+        candidate.protocol = elems[2].toLowerCase();
+        candidate.priority = elems[3];
+        candidate.ip = elems[4];
+        candidate.port = elems[5];
+        // elems[6] => "typ"
+        candidate.type = elems[7];
+
+        candidate.generation = '0'; // fippo from jitsi-meet: default, may be overwritten below
+
+        for (i = 8; i < elems.length; i += 2) {
+            switch (elems[i]) {
+            case 'raddr':
+                candidate['rel-addr'] = elems[i + 1];
+                break;
+            case 'rport':
+                candidate['rel-port'] = elems[i + 1];
+                break;
+            case 'generation':
+                candidate.generation = elems[i + 1];
+                break;
+            case 'tcptype':
+                candidate.tcptype = elems[i + 1];
+                break;
+            default: // TODO
+                console.log('not translating "' + elems[i] + '" = "' + elems[i + 1] + '"');
+            }
+        }
+        candidate.network = '1';
+        candidate.id = Math.random().toString(36).substr(2, 10); // not applicable to SDP -- FIXME: should be unique, not just random
+        return candidate;
+    },
+    candidateFromJingle: function (cand) {
+        var line = 'a=candidate:';
+        line += cand.getAttribute('foundation');
+        line += ' ';
+        line += cand.getAttribute('component');
+        line += ' ';
+        line += cand.getAttribute('protocol'); //.toUpperCase(); // chrome M23 doesn't like this
+        line += ' ';
+        line += cand.getAttribute('priority');
+        line += ' ';
+        line += cand.getAttribute('ip');
+        line += ' ';
+        line += cand.getAttribute('port');
+        line += ' ';
+        line += 'typ';
+        line += ' ' + cand.getAttribute('type');
+        line += ' ';
+        switch (cand.getAttribute('type')) {
+        case 'srflx':
+        case 'prflx':
+        case 'relay':
+            if (cand.getAttribute('rel-addr') && cand.getAttribute('rel-port')) {
+                line += 'raddr';
+                line += ' ';
+                line += cand.getAttribute('rel-addr');
+                line += ' ';
+                line += 'rport';
+                line += ' ';
+                line += cand.getAttribute('rel-port');
+                line += ' ';
+            }
+            break;
+        }
+        line += 'generation';
+        line += ' ';
+        line += cand.getAttribute('generation') || '0';
+        return line + '\r\n';
+    }
+};
+}(jQuery));
diff --git a/lib/strophe.jingle/strophe.jingle.session.js b/lib/strophe.jingle/strophe.jingle.session.js
new file mode 100644
index 0000000..e5ee6e0
--- /dev/null
+++ b/lib/strophe.jingle/strophe.jingle.session.js
@@ -0,0 +1,914 @@
+/* jshint -W117 */
+// Jingle stuff
+var JingleSession;
+
+(function($){
+JingleSession = function(me, sid, connection) {
+    this.me = me;
+    this.sid = sid;
+    this.connection = connection;
+    this.initiator = null;
+    this.responder = null;
+    this.isInitiator = null;
+    this.peerjid = null;
+    this.state = null;
+    this.peerconnection = null;
+    this.remoteStream = null;
+    this.localSDP = null;
+    this.remoteSDP = null;
+    this.localStreams = [];
+    this.relayedStreams = [];
+    this.remoteStreams = [];
+    this.startTime = null;
+    this.stopTime = null;
+    this.media_constraints = null;
+    this.pc_constraints = null;
+    this.ice_config = {};
+    this.drip_container = [];
+
+    this.usetrickle = true;
+    this.usepranswer = false; // early transport warmup -- mind you, this might fail. depends on webrtc issue 1718
+    this.usedrip = false; // dripping is sending trickle candidates not one-by-one
+
+    this.hadstuncandidate = false;
+    this.hadturncandidate = false;
+    this.lasticecandidate = false;
+
+    this.statsinterval = null;
+
+    this.reason = null;
+
+    this.addssrc = [];
+    this.removessrc = [];
+    this.pendingop = null;
+
+    this.wait = true;
+
+    // XEP-0172 support, non-standard
+    this.nickname = null;
+
+    // non-standard "please start muted" support for colibri/meet
+    this.startmuted = false;
+
+    // Filter for testcases with ICE Candidates
+    this.filter_candidates = null;
+}
+
+JingleSession.prototype.initiate = function (peerjid, isInitiator) {
+    var self = this;
+    if (this.state !== null) {
+        console.error('attempt to initiate on session ' + this.sid +
+                  'in state ' + this.state);
+        return;
+    }
+    this.isInitiator = isInitiator;
+    this.state = 'pending';
+    this.initiator = isInitiator ? this.me : peerjid;
+    this.responder = !isInitiator ? this.me : peerjid;
+    this.peerjid = peerjid;
+    //console.log('create PeerConnection ' + JSON.stringify(this.ice_config));
+    try {
+        this.peerconnection = new RTCPeerconnection(this.ice_config,
+                                                     this.pc_constraints);
+    } catch (e) {
+        console.error('Failed to create PeerConnection, exception: ',
+                      e.message);
+        console.error(e);
+        return;
+    }
+    this.hadstuncandidate = false;
+    this.hadturncandidate = false;
+    this.lasticecandidate = false;
+    this.peerconnection.onicecandidate = function (event) {
+        self.sendIceCandidate(event.candidate);
+    };
+    this.peerconnection.onaddstream = function (event) {
+        self.remoteStream = event.stream;
+        self.remoteStreams.push(event.stream);
+        $(document).trigger('remotestreamadded.jingle', [event, self.sid]);
+    };
+    this.peerconnection.onremovestream = function (event) {
+        self.remoteStream = null;
+        // FIXME: remove from this.remoteStreams
+        $(document).trigger('remotestreamremoved.jingle', [event, self.sid]);
+    };
+    this.peerconnection.onsignalingstatechange = function (event) {
+        if (!(self && self.peerconnection)) return;
+    };
+    this.peerconnection.oniceconnectionstatechange = function (event) {
+        if (!(self && self.peerconnection)) return;
+        switch (self.peerconnection.iceConnectionState) {
+        case 'connected':
+            this.startTime = new Date();
+            break;
+        case 'disconnected':
+            this.stopTime = new Date();
+            break;
+        }
+        $(document).trigger('iceconnectionstatechange.jingle', [self.sid, self]);
+    };
+    // add any local and relayed stream
+    this.localStreams.forEach(function(stream) {
+        self.peerconnection.addStream(stream);
+    });
+    this.relayedStreams.forEach(function(stream) {
+        self.peerconnection.addStream(stream);
+    });
+};
+
+JingleSession.prototype.accept = function () {
+    var self = this;
+    this.state = 'active';
+
+    var pranswer = this.peerconnection.localDescription;
+    if (!pranswer || pranswer.type != 'pranswer') {
+        return;
+    }
+    console.log('going from pranswer to answer');
+    if (this.usetrickle) {
+        // remove candidates already sent from session-accept
+        var lines = SDPUtil.find_lines(pranswer.sdp, 'a=candidate:');
+        for (var i = 0; i < lines.length; i++) {
+            pranswer.sdp = pranswer.sdp.replace(lines[i] + '\r\n', '');
+        }
+    }
+    while (SDPUtil.find_line(pranswer.sdp, 'a=inactive')) {
+        // FIXME: change any inactive to sendrecv or whatever they were originally
+        pranswer.sdp = pranswer.sdp.replace('a=inactive', 'a=sendrecv');
+    }
+    var prsdp = new SDP(pranswer.sdp);
+    var accept = $iq({to: this.peerjid,
+             type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-accept',
+           initiator: this.initiator,
+           responder: this.responder,
+           sid: this.sid });
+    prsdp.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder');
+    this.connection.sendIQ(accept,
+        function () {
+            var ack = {};
+            ack.source = 'answer';
+            $(document).trigger('ack.jingle', [self.sid, ack]);
+        },
+        function (stanza) {
+            var error = ($(stanza).find('error').length) ? {
+                code: $(stanza).find('error').attr('code'),
+                reason: $(stanza).find('error :first')[0].tagName,
+            }:{};
+            error.source = 'answer';
+            $(document).trigger('error.jingle', [self.sid, error]);
+        },
+    10000);
+
+    var sdp = this.peerconnection.localDescription.sdp;
+    while (SDPUtil.find_line(sdp, 'a=inactive')) {
+        // FIXME: change any inactive to sendrecv or whatever they were originally
+        sdp = sdp.replace('a=inactive', 'a=sendrecv');
+    }
+    this.peerconnection.setLocalDescription(new RTCSessionDescription({type: 'answer', sdp: sdp}),
+        function () {
+            //console.log('setLocalDescription success');
+            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+        },
+        function (e) {
+            console.error('setLocalDescription failed', e);
+        }
+    );
+};
+
+JingleSession.prototype.terminate = function (reason) {
+    this.state = 'ended';
+    this.reason = reason;
+    this.peerconnection.close();
+    if (this.statsinterval !== null) {
+        window.clearInterval(this.statsinterval);
+        this.statsinterval = null;
+    }
+};
+
+JingleSession.prototype.active = function () {
+    return this.state == 'active';
+};
+
+JingleSession.prototype.sendIceCandidate = function (candidate) {
+    var self = this;
+    if (candidate && !this.lasticecandidate) {
+        var ice = SDPUtil.iceparams(this.localSDP.media[candidate.sdpMLineIndex], this.localSDP.session);
+        var jcand = SDPUtil.candidateToJingle(candidate.candidate);
+        if (!(ice && jcand)) {
+            console.error('failed to get ice && jcand');
+            return;
+        }
+        ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+
+        if (jcand.type === 'srflx') {
+            this.hadstuncandidate = true;
+        } else if (jcand.type === 'relay') {
+            this.hadturncandidate = true;
+        }
+
+        if(this.filter_candidates === null || jcand.type === this.filter_candidates) {
+            if (this.usetrickle) {
+                console.log('sendIceCandidate using trickle');
+                if (this.usedrip) {
+                    if (this.drip_container.length === 0) {
+                        // start 20ms callout
+                        window.setTimeout(function () {
+                            console.log('sending drip container');
+                            if (self.drip_container.length === 0) return;
+                            self.sendIceCandidates(self.drip_container);
+                            self.drip_container = [];
+                        }, 20);
+
+                    }
+                    this.drip_container.push(event.candidate);
+                    return;
+                } else {
+                    console.log('sending single candidate');
+                    self.sendIceCandidates([event.candidate]);
+                }
+            }
+        }
+    } else {
+        console.log('sendIceCandidate: last candidate...');
+        if (!this.usetrickle) {
+            console.log('should send full offer now...');
+            var init = $iq({to: this.peerjid,
+                       type: 'set'})
+                .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+                   action: this.peerconnection.localDescription.type == 'offer' ? 'session-initiate' : 'session-accept',
+                   initiator: this.initiator,
+                   sid: this.sid});
+            if (this.nickname != null) {
+                init.c('nick', {xmlns:'http://jabber.org/protocol/nick'}).t(this.nickname).up();
+            }
+            if (this.startmuted) {
+                init.c('muted', {xmlns:'http://jitsi.org/protocol/meet#startmuted'}).up();
+            }
+            this.localSDP = new SDP(this.peerconnection.localDescription.sdp);
+            this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder');
+            console.log('try to send ack(offer)...');
+            this.connection.sendIQ(init,
+                function () {
+                    console.log('Sent session initiate (ACK, offer)...');
+                    var ack = {};
+                    ack.source = 'offer';
+                    $(document).trigger('ack.jingle', [self.sid, ack]);
+                },
+                function (stanza) {
+                    self.state = 'error';
+                    self.peerconnection.close();
+                    var error = ($(stanza).find('error').length) ? {
+                        code: $(stanza).find('error').attr('code'),
+                        reason: $(stanza).find('error :first')[0].tagName,
+                    }:{};
+                    error.source = 'offer';
+                    $(document).trigger('error.jingle', [self.sid, error]);
+                },
+            10000);
+        }
+        this.lasticecandidate = true;
+        console.log('Have we encountered any srflx candidates? ' + this.hadstuncandidate);
+        console.log('Have we encountered any relay candidates? ' + this.hadturncandidate);
+
+        if (!(this.hadstuncandidate || this.hadturncandidate) && this.peerconnection.signalingState != 'closed') {
+            console.log('no candidates found!');
+            $(document).trigger('nostuncandidates.jingle', [this.sid]);
+        }
+    }
+};
+
+JingleSession.prototype.sendIceCandidates = function (candidates) {
+    console.log('sendIceCandidates', candidates);
+    var cand = $iq({to: this.peerjid, type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'transport-info',
+           initiator: this.initiator,
+           sid: this.sid});
+    for (var mid = 0; mid < this.localSDP.media.length; mid++) {
+        var cands = candidates.filter(function (el) { return el.sdpMLineIndex == mid; });
+        if (cands.length > 0) {
+            var ice = SDPUtil.iceparams(this.localSDP.media[mid], this.localSDP.session);
+            ice.xmlns = 'urn:xmpp:jingle:transports:ice-udp:1';
+            cand.c('content', {creator: this.initiator == this.me ? 'initiator' : 'responder',
+                   name: cands[0].sdpMid
+            }).c('transport', ice);
+            for (var i = 0; i < cands.length; i++) {
+                cand.c('candidate', SDPUtil.candidateToJingle(cands[i].candidate)).up();
+            }
+            // add fingerprint
+            if (SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session)) {
+                var tmp = SDPUtil.parse_fingerprint(SDPUtil.find_line(this.localSDP.media[mid], 'a=fingerprint:', this.localSDP.session));
+                tmp.required = true;
+                cand.c('fingerprint').t(tmp.fingerprint);
+                delete tmp.fingerprint;
+                cand.attrs(tmp);
+                cand.up();
+            }
+            cand.up(); // transport
+            cand.up(); // content
+        }
+    }
+    // might merge last-candidate notification into this, but it is called alot later. See webrtc issue #2340
+    //console.log('was this the last candidate', this.lasticecandidate);
+    console.log('try to send ack(transportinfo)...');
+    this.connection.sendIQ(cand,
+        function () {
+            var ack = {};
+            ack.source = 'transportinfo';
+            console.log('Sent session initiate (ACK, transportinfo)...');
+            $(document).trigger('ack.jingle', [this.sid, ack]);
+        },
+        function (stanza) {
+            var error = ($(stanza).find('error').length) ? {
+                code: $(stanza).find('error').attr('code'),
+                reason: $(stanza).find('error :first')[0].tagName,
+            }:{};
+            error.source = 'transportinfo';
+            $(document).trigger('error.jingle', [this.sid, error]);
+        },
+    10000);
+};
+
+
+JingleSession.prototype.sendOffer = function () {
+    //console.log('sendOffer...');
+    var self = this;
+    this.peerconnection.createOffer(function (sdp) {
+            self.createdOffer(sdp);
+        },
+        function (e) {
+            console.error('createOffer failed', e);
+        },
+        this.media_constraints
+    );
+};
+
+JingleSession.prototype.createdOffer = function (sdp) {
+    //console.log('createdOffer', sdp);
+    var self = this;
+    this.localSDP = new SDP(sdp.sdp);
+    //this.localSDP.mangle();
+    if (this.usetrickle) {
+        var init = $iq({to: this.peerjid,
+                   type: 'set'})
+            .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+               action: 'session-initiate',
+               initiator: this.initiator,
+               sid: this.sid});
+        if (this.nickname != null) {
+            init.c('nick', {xmlns:'http://jabber.org/protocol/nick'}).t(this.nickname).up();
+        }
+        if (this.startmuted) {
+            init.c('muted', {xmlns:'http://jitsi.org/protocol/meet#startmuted'}).up();
+        }
+        this.localSDP.toJingle(init, this.initiator == this.me ? 'initiator' : 'responder');
+        this.connection.sendIQ(init,
+            function () {
+                var ack = {};
+                ack.source = 'offer';
+                $(document).trigger('ack.jingle', [self.sid, ack]);
+            },
+            function (stanza) {
+                self.state = 'error';
+                self.peerconnection.close();
+                var error = ($(stanza).find('error').length) ? {
+                    code: $(stanza).find('error').attr('code'),
+                    reason: $(stanza).find('error :first')[0].tagName,
+                }:{};
+                error.source = 'offer';
+                $(document).trigger('error.jingle', [self.sid, error]);
+            },
+        10000);
+    }
+    sdp.sdp = this.localSDP.raw;
+    this.peerconnection.setLocalDescription(sdp, 
+        function () {
+            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+            //console.log('setLocalDescription success');
+        },
+        function (e) {
+            console.error('setLocalDescription failed', e);
+        }
+    );
+    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
+    for (var i = 0; i < cands.length; i++) {
+        var cand = SDPUtil.parse_icecandidate(cands[i]);
+        if (cand.type == 'srflx') {
+            this.hadstuncandidate = true;
+        } else if (cand.type == 'relay') {
+            this.hadturncandidate = true;
+        }
+    }
+};
+
+JingleSession.prototype.setRemoteDescription = function (elem, desctype) {
+    //console.log('setting remote description... ', desctype);
+    this.remoteSDP = new SDP('');
+    this.remoteSDP.fromJingle(elem);
+    if (this.peerconnection.remoteDescription !== null) {
+        console.log('setRemoteDescription when remote description is not null, should be pranswer', this.peerconnection.remoteDescription);
+        if (this.peerconnection.remoteDescription.type == 'pranswer') {
+            var pranswer = new SDP(this.peerconnection.remoteDescription.sdp);
+            for (var i = 0; i < pranswer.media.length; i++) {
+                // make sure we have ice ufrag and pwd
+                if (!SDPUtil.find_line(this.remoteSDP.media[i], 'a=ice-ufrag:', this.remoteSDP.session)) {
+                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session)) {
+                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-ufrag:', pranswer.session) + '\r\n';
+                    } else {
+                        console.warn('no ice ufrag?');
+                    }
+                    if (SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session)) {
+                        this.remoteSDP.media[i] += SDPUtil.find_line(pranswer.media[i], 'a=ice-pwd:', pranswer.session) + '\r\n';
+                    } else {
+                        console.warn('no ice pwd?');
+                    }
+                }
+                // copy over candidates
+                var lines = SDPUtil.find_lines(pranswer.media[i], 'a=candidate:');
+                for (var j = 0; j < lines.length; j++) {
+                    this.remoteSDP.media[i] += lines[j] + '\r\n';
+                }
+            }
+            this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
+        }
+    }
+    var remotedesc = new RTCSessionDescription({type: desctype, sdp: this.remoteSDP.raw});
+    
+    this.peerconnection.setRemoteDescription(remotedesc,
+        function () {
+            //console.log('setRemoteDescription success');
+        },
+        function (e) {
+            console.error('setRemoteDescription error', e);
+        }
+    );
+};
+
+JingleSession.prototype.addIceCandidate = function (elem) {
+    var self = this;
+    if (this.peerconnection.signalingState == 'closed') {
+        return;
+    }
+    if (!this.peerconnection.remoteDescription && this.peerconnection.signalingState == 'have-local-offer') {
+        console.log('trickle ice candidate arriving before session accept...');
+        // create a PRANSWER for setRemoteDescription
+        if (!this.remoteSDP) {
+            var cobbled = 'v=0\r\n' +
+                'o=- ' + '1923518516' + ' 2 IN IP4 0.0.0.0\r\n' +// FIXME
+                's=-\r\n' +
+                't=0 0\r\n';
+            // first, take some things from the local description
+            for (var i = 0; i < this.localSDP.media.length; i++) {
+                cobbled += SDPUtil.find_line(this.localSDP.media[i], 'm=') + '\r\n';
+                cobbled += SDPUtil.find_lines(this.localSDP.media[i], 'a=rtpmap:').join('\r\n') + '\r\n';
+                if (SDPUtil.find_line(this.localSDP.media[i], 'a=mid:')) {
+                    cobbled += SDPUtil.find_line(this.localSDP.media[i], 'a=mid:') + '\r\n';
+                }
+                cobbled += 'a=inactive\r\n';
+            }
+            this.remoteSDP = new SDP(cobbled);
+        }
+        // then add things like ice and dtls from remote candidate
+        elem.each(function () {
+            for (var i = 0; i < self.remoteSDP.media.length; i++) {
+                if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+                        self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+                    if (!SDPUtil.find_line(self.remoteSDP.media[i], 'a=ice-ufrag:')) {
+                        var tmp = $(this).find('transport');
+                        self.remoteSDP.media[i] += 'a=ice-ufrag:' + tmp.attr('ufrag') + '\r\n';
+                        self.remoteSDP.media[i] += 'a=ice-pwd:' + tmp.attr('pwd') + '\r\n';
+                        tmp = $(this).find('transport>fingerprint');
+                        if (tmp.length) {
+                            self.remoteSDP.media[i] += 'a=fingerprint:' + tmp.attr('hash') + ' ' + tmp.text() + '\r\n';
+                        } else {
+                            console.log('no dtls fingerprint (webrtc issue #1718?)');
+                            self.remoteSDP.media[i] += 'a=crypto:1 AES_CM_128_HMAC_SHA1_80 inline:BAADBAADBAADBAADBAADBAADBAADBAADBAADBAAD\r\n';
+                        }
+                        break;
+                    }
+                }
+            }
+        });
+        this.remoteSDP.raw = this.remoteSDP.session + this.remoteSDP.media.join('');
+
+        // we need a complete SDP with ice-ufrag/ice-pwd in all parts
+        // this makes the assumption that the PRANSWER is constructed such that the ice-ufrag is in all mediaparts
+        // but it could be in the session part as well. since the code above constructs this sdp this can't happen however
+        var iscomplete = this.remoteSDP.media.filter(function (mediapart) {
+            return SDPUtil.find_line(mediapart, 'a=ice-ufrag:');
+        }).length == this.remoteSDP.media.length;
+
+        if (iscomplete) {
+            console.log('setting pranswer');
+            try {
+                this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'pranswer', sdp: this.remoteSDP.raw }),
+                    function() {
+                    },
+                    function(e) {
+                        console.log('setRemoteDescription pranswer failed', e.toString());
+                    });
+            } catch (e) {
+                console.error('setting pranswer failed', e);
+            }
+        } else {
+            //console.log('not yet setting pranswer');
+        }
+    }
+    // operate on each content element
+    elem.each(function () {
+        // would love to deactivate this, but firefox still requires it
+        var idx = -1;
+        var i;
+        for (i = 0; i < self.remoteSDP.media.length; i++) {
+            if (SDPUtil.find_line(self.remoteSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+                self.remoteSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+                idx = i;
+                break;
+            }
+        }
+        if (idx == -1) { // fall back to localdescription
+            for (i = 0; i < self.localSDP.media.length; i++) {
+                if (SDPUtil.find_line(self.localSDP.media[i], 'a=mid:' + $(this).attr('name')) ||
+                    self.localSDP.media[i].indexOf('m=' + $(this).attr('name')) === 0) {
+                    idx = i;
+                    break;
+                }
+            }
+        }
+        var name = $(this).attr('name');
+        // TODO: check ice-pwd and ice-ufrag?
+        $(this).find('transport>candidate').each(function () {
+            var line, candidate;
+            line = SDPUtil.candidateFromJingle(this);
+            candidate = new RTCIceCandidate({sdpMLineIndex: idx,
+                                            sdpMid: name,
+                                            candidate: line});
+            try {
+                self.peerconnection.addIceCandidate(candidate);
+            } catch (e) {
+                console.error('addIceCandidate failed', e.toString(), line);
+            }
+        });
+    });
+};
+
+JingleSession.prototype.sendAnswer = function (provisional) {
+    //console.log('createAnswer', provisional);
+    var self = this;
+    this.peerconnection.createAnswer(
+        function (sdp) {
+            self.createdAnswer(sdp, provisional);
+        },
+        function (e) {
+            console.error('createAnswer failed', e);
+        },
+        this.media_constraints
+    );
+};
+
+JingleSession.prototype.createdAnswer = function (sdp, provisional) {
+    //console.log('createAnswer callback');
+    var self = this;
+    this.localSDP = new SDP(sdp.sdp);
+    //this.localSDP.mangle();
+    this.usepranswer = provisional === true;
+
+    if (this.startmuted) {
+        console.log('we got a request to start muted...');
+        this.connection.jingle.localStream.getAudioTracks().forEach(function (track) {
+            track.enabled = false;
+        });
+        // doing this freezes local video, too (which probably means it should be replaced
+        // by a symbol
+        this.connection.jingle.localStream.getVideoTracks().forEach(function (track) {
+            track.enabled = false;
+        });
+
+        // set video to recvonly
+        this.localSDP.media[1] = this.localSDP.media[1].replace('a=sendrecv', 'a=recvonly');
+        // and remove a=ssrc lines. Weird things happen otherwise
+        SDPUtil.find_lines(this.localSDP.media[1], 'a=ssrc:').forEach(function (line) {
+            self.localSDP.media[1] = self.localSDP.media[1].replace(line + '\r\n', '');
+        });
+        this.localSDP.raw = this.localSDP.session + this.localSDP.media.join('');
+    }
+
+    if (this.usetrickle) {
+        if (!this.usepranswer) {
+            var accept = $iq({to: this.peerjid,
+                     type: 'set'})
+                .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+                   action: 'session-accept',
+                   initiator: this.initiator,
+                   responder: this.responder,
+                   sid: this.sid });
+            this.localSDP.toJingle(accept, this.initiator == this.me ? 'initiator' : 'responder');
+            this.connection.sendIQ(accept,
+                function () {
+                    var ack = {};
+                    ack.source = 'answer';
+                    $(document).trigger('ack.jingle', [self.sid, ack]);
+                },
+                function (stanza) {
+                    var error = ($(stanza).find('error').length) ? {
+                        code: $(stanza).find('error').attr('code'),
+                        reason: $(stanza).find('error :first')[0].tagName,
+                    }:{};
+                    error.source = 'answer';
+                    $(document).trigger('error.jingle', [self.sid, error]);
+                },
+            10000);
+        } else {
+            sdp.type = 'pranswer';
+            for (var i = 0; i < this.localSDP.media.length; i++) {
+                this.localSDP.media[i] = this.localSDP.media[i].replace('a=sendrecv\r\n', 'a=inactive\r\n');
+            }
+            this.localSDP.raw = this.localSDP.session + this.localSDP.media.join('');
+        }
+    }
+    sdp.sdp = this.localSDP.raw;
+    this.peerconnection.setLocalDescription(sdp,
+        function () {
+            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+            //console.log('setLocalDescription success');
+        },
+        function (e) {
+            console.error('setLocalDescription failed', e);
+        }
+    );
+    var cands = SDPUtil.find_lines(this.localSDP.raw, 'a=candidate:');
+    for (var j = 0; j < cands.length; j++) {
+        var cand = SDPUtil.parse_icecandidate(cands[j]);
+        if (cand.type == 'srflx') {
+            this.hadstuncandidate = true;
+        } else if (cand.type == 'relay') {
+            this.hadturncandidate = true;
+        }
+    }
+};
+
+JingleSession.prototype.sendTerminate = function (reason, text) {
+    var self = this,
+        term = $iq({to: this.peerjid,
+               type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-terminate',
+           initiator: this.initiator,
+           sid: this.sid})
+        .c('reason')
+        .c(reason || 'success');
+        
+    if (text) {
+        term.up().c('text').t(text);
+    }
+    
+    this.connection.sendIQ(term,
+        function () {
+            self.peerconnection.close();
+            self.peerconnection = null;
+            self.terminate();
+            var ack = {};
+            ack.source = 'terminate';
+            $(document).trigger('ack.jingle', [self.sid, ack]);
+        },
+        function (stanza) {
+            var error = ($(stanza).find('error').length) ? {
+                code: $(stanza).find('error').attr('code'),
+                reason: $(stanza).find('error :first')[0].tagName,
+            }:{};
+            $(document).trigger('ack.jingle', [self.sid, error]);
+        },
+    10000);
+    if (this.statsinterval !== null) {
+        window.clearInterval(this.statsinterval);
+        this.statsinterval = null;
+    }
+};
+
+
+JingleSession.prototype.addSource = function (elem) {
+    console.log('addssrc', new Date().getTime());
+    console.log('ice', this.peerconnection.iceConnectionState);
+    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+    var self = this;
+    $(elem).each(function (idx, content) {
+        var name = $(content).attr('name');
+        var lines = '';
+        tmp = $(content).find('>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+        tmp.each(function () {
+            var ssrc = $(this).attr('ssrc');
+            $(this).find('>parameter').each(function () {
+                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
+                if ($(this).attr('value') && $(this).attr('value').length)
+                    lines += ':' + $(this).attr('value');
+                lines += '\r\n';
+            });
+        });
+        sdp.media.forEach(function(media, idx) {
+            if (!SDPUtil.find_line(media, 'a=mid:' + name))
+                return;
+            sdp.media[idx] += lines;
+            if (!self.addssrc[idx]) self.addssrc[idx] = '';
+            self.addssrc[idx] += lines;
+        });
+        sdp.raw = sdp.session + sdp.media.join('');
+    });
+    this.modifySources();
+};
+
+JingleSession.prototype.removeSource = function (elem) {
+    console.log('removessrc', new Date().getTime());
+    console.log('ice', this.peerconnection.iceConnectionState);
+    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+    var self = this;
+    $(elem).each(function (idx, content) {
+        var name = $(content).attr('name');
+        var lines = '';
+        tmp = $(content).find('>source[xmlns="urn:xmpp:jingle:apps:rtp:ssma:0"]');
+        tmp.each(function () {
+            var ssrc = $(this).attr('ssrc');
+            $(this).find('>parameter').each(function () {
+                lines += 'a=ssrc:' + ssrc + ' ' + $(this).attr('name');
+                if ($(this).attr('value') && $(this).attr('value').length)
+                    lines += ':' + $(this).attr('value');
+                lines += '\r\n';
+            });
+        });
+        sdp.media.forEach(function(media, idx) {
+            if (!SDPUtil.find_line(media, 'a=mid:' + name))
+                return;
+            sdp.media[idx] += lines;
+            if (!self.addssrc[idx]) self.removessrc[idx] = '';
+            self.removessrc[idx] += lines;
+        });
+        sdp.raw = sdp.session + sdp.media.join('');
+    });
+    this.modifySources();
+};
+
+JingleSession.prototype.modifySources = function() {
+    var self = this;
+    if (this.peerconnection.signalingState == 'closed') return;
+    if (!(this.addssrc.length || this.removessrc.length || this.pendingop !== null)) return;
+    if (!(this.peerconnection.signalingState == 'stable' && this.peerconnection.iceConnectionState == 'connected')) {
+        console.warn('modifySources not yet', this.peerconnection.signalingState, this.peerconnection.iceConnectionState);
+        this.wait = true;
+        window.setTimeout(function() { self.modifySources(); }, 250);
+        return;
+    }
+    if (this.wait) {
+        window.setTimeout(function() { self.modifySources(); }, 2500);
+        this.wait = false;
+        return;
+    }
+
+    var sdp = new SDP(this.peerconnection.remoteDescription.sdp);
+
+    // add sources
+    this.addssrc.forEach(function(lines, idx) {
+        sdp.media[idx] += lines;
+    });
+    this.addssrc = [];
+
+    // remove sources
+    this.removessrc.forEach(function(lines, idx) {
+        lines = lines.split('\r\n');
+        lines.pop(); // remove empty last element;
+        lines.forEach(function(line) {
+            sdp.media[idx] = sdp.media[idx].replace(line + '\r\n', '');
+        });
+    });
+    this.removessrc = [];
+
+    sdp.raw = sdp.session + sdp.media.join('');
+    this.peerconnection.setRemoteDescription(new RTCSessionDescription({type: 'offer', sdp: sdp.raw}),
+        function() {
+            self.peerconnection.createAnswer(
+                function(modifiedAnswer) {
+                    // change video direction, see https://github.com/jitsi/jitmeet/issues/41
+                    if (self.pendingop !== null) {
+                        var sdp = new SDP(modifiedAnswer.sdp);
+                        if (sdp.media.length > 1) {
+                            switch(self.pendingop) {
+                            case 'mute':
+                                sdp.media[1] = sdp.media[1].replace('a=sendrecv', 'a=recvonly');
+                                break;
+                            case 'unmute':
+                                sdp.media[1] = sdp.media[1].replace('a=recvonly', 'a=sendrecv');
+                                break;
+                            }
+                            sdp.raw = sdp.session + sdp.media.join('');
+                            modifiedAnswer.sdp = sdp.raw;
+                        }
+                        self.pendingop = null;
+                    }
+
+                    self.peerconnection.setLocalDescription(modifiedAnswer,
+                        function() {
+                            //console.log('modified setLocalDescription ok');
+                            $(document).trigger('setLocalDescription.jingle', [self.sid]);
+                        },
+                        function(error) {
+                            console.log('modified setLocalDescription failed');
+                        }
+                    );
+                },
+                function(error) {
+                    console.log('modified answer failed');
+                }
+            );
+        },
+        function(error) {
+            console.log('modify failed');
+        }
+    );
+};
+
+// SDP-based mute by going recvonly/sendrecv
+// FIXME: should probably black out the screen as well
+JingleSession.prototype.hardMuteVideo = function (muted) {
+    this.pendingop = muted ? 'mute' : 'unmute';
+    this.modifySources();
+
+    this.connection.jingle.localStream.getVideoTracks().forEach(function (track) {
+        track.enabled = !muted;
+    });
+};
+
+JingleSession.prototype.sendMute = function (muted, content) {
+    var info = $iq({to: this.peerjid,
+             type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-info',
+           initiator: this.initiator,
+           sid: this.sid });
+    info.c(muted ? 'mute' : 'unmute', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
+    info.attrs({'creator': this.me == this.initiator ? 'creator' : 'responder'});
+    if (content) {
+        info.attrs({'name': content});
+    }
+    this.connection.send(info);
+};
+
+JingleSession.prototype.sendRinging = function () {
+    var info = $iq({to: this.peerjid,
+             type: 'set'})
+        .c('jingle', {xmlns: 'urn:xmpp:jingle:1',
+           action: 'session-info',
+           initiator: this.initiator,
+           sid: this.sid });
+    info.c('ringing', {xmlns: 'urn:xmpp:jingle:apps:rtp:info:1'});
+    this.connection.send(info);
+};
+
+JingleSession.prototype.getStats = function (interval) {
+    var self = this;
+    var recv = {audio: 0, video: 0};
+    var lost = {audio: 0, video: 0};
+    var lastrecv = {audio: 0, video: 0};
+    var lastlost = {audio: 0, video: 0};
+    var loss = {audio: 0, video: 0};
+    var delta = {audio: 0, video: 0};
+    this.statsinterval = window.setInterval(function () {
+        if (self && self.peerconnection && self.peerconnection.getStats) {
+            self.peerconnection.getStats(function (stats) {
+                var results = stats.result();
+                // TODO: there are so much statistics you can get from this..
+                for (var i = 0; i < results.length; ++i) {
+                    if (results[i].type == 'ssrc') {
+                        var packetsrecv = results[i].stat('packetsReceived');
+                        var packetslost = results[i].stat('packetsLost');
+                        if (packetsrecv && packetslost) {
+                            packetsrecv = parseInt(packetsrecv, 10);
+                            packetslost = parseInt(packetslost, 10);
+                            
+                            if (results[i].stat('googFrameRateReceived')) {
+                                lastlost.video = lost.video;
+                                lastrecv.video = recv.video;
+                                recv.video = packetsrecv;
+                                lost.video = packetslost;
+                            } else {
+                                lastlost.audio = lost.audio;
+                                lastrecv.audio = recv.audio;
+                                recv.audio = packetsrecv;
+                                lost.audio = packetslost;
+                            }
+                        }
+                    }
+                }
+                delta.audio = recv.audio - lastrecv.audio;
+                delta.video = recv.video - lastrecv.video;
+                loss.audio = (delta.audio > 0) ? Math.ceil(100 * (lost.audio - lastlost.audio) / delta.audio) : 0;
+                loss.video = (delta.video > 0) ? Math.ceil(100 * (lost.video - lastlost.video) / delta.video) : 0;
+                $(document).trigger('packetloss.jingle', [self.sid, loss]);
+            });
+        }
+    }, interval || 3000);
+    return this.statsinterval;
+};
+
+}(jQuery));
diff --git a/lib/strophe.js b/lib/strophe.js
new file mode 100644
index 0000000..24273aa
--- /dev/null
+++ b/lib/strophe.js
@@ -0,0 +1,5153 @@
+/**
+ * Modified by
+ * Klaus Herberth, 2014
+ */
+
+/*! This code was written by Tyler Akins and has been placed in the
+   public domain.  It would be nice if you left this header intact.
+   Base64 code from Tyler Akins -- http://rumkin.com
+*/
+
+var Base64 = (function () {
+    var keyStr = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=";
+
+    var obj = {
+        /**
+         * Encodes a string in base64
+         * @param {String} input The string to encode in base64.
+         */
+        encode: function (input) {
+            var output = "";
+            var chr1, chr2, chr3;
+            var enc1, enc2, enc3, enc4;
+            var i = 0;
+
+            do {
+                chr1 = input.charCodeAt(i++);
+                chr2 = input.charCodeAt(i++);
+                chr3 = input.charCodeAt(i++);
+
+                enc1 = chr1 >> 2;
+                enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);
+                enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);
+                enc4 = chr3 & 63;
+
+                if (isNaN(chr2)) {
+                    enc3 = enc4 = 64;
+                } else if (isNaN(chr3)) {
+                    enc4 = 64;
+                }
+
+                output = output + keyStr.charAt(enc1) + keyStr.charAt(enc2) +
+                    keyStr.charAt(enc3) + keyStr.charAt(enc4);
+            } while (i < input.length);
+
+            return output;
+        },
+
+        /**
+         * Decodes a base64 string.
+         * @param {String} input The string to decode.
+         */
+        decode: function (input) {
+            var output = "";
+            var chr1, chr2, chr3;
+            var enc1, enc2, enc3, enc4;
+            var i = 0;
+
+            // remove all characters that are not A-Z, a-z, 0-9, +, /, or =
+            input = input.replace(/[^A-Za-z0-9\+\/\=]/g, "");
+
+            do {
+                enc1 = keyStr.indexOf(input.charAt(i++));
+                enc2 = keyStr.indexOf(input.charAt(i++));
+                enc3 = keyStr.indexOf(input.charAt(i++));
+                enc4 = keyStr.indexOf(input.charAt(i++));
+
+                chr1 = (enc1 << 2) | (enc2 >> 4);
+                chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);
+                chr3 = ((enc3 & 3) << 6) | enc4;
+
+                output = output + String.fromCharCode(chr1);
+
+                if (enc3 != 64) {
+                    output = output + String.fromCharCode(chr2);
+                }
+                if (enc4 != 64) {
+                    output = output + String.fromCharCode(chr3);
+                }
+            } while (i < input.length);
+
+            return output;
+        }
+    };
+
+    return obj;
+})();
+
+/*!
+ * A JavaScript implementation of the Secure Hash Algorithm, SHA-1, as defined
+ * in FIPS PUB 180-1
+ * Version 2.1a Copyright Paul Johnston 2000 - 2002.
+ * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+ * Distributed under the BSD License
+ * See http://pajhome.org.uk/crypt/md5 for details.
+ */
+
+/* Some functions and variables have been stripped for use with Strophe */
+
+/*
+ * These are the functions you'll usually want to call
+ * They take string arguments and return either hex or base-64 encoded strings
+ */
+function b64_sha1(s){return binb2b64(core_sha1(str2binb(s),s.length * 8));}
+function str_sha1(s){return binb2str(core_sha1(str2binb(s),s.length * 8));}
+function b64_hmac_sha1(key, data){ return binb2b64(core_hmac_sha1(key, data));}
+function str_hmac_sha1(key, data){ return binb2str(core_hmac_sha1(key, data));}
+
+/*
+ * Calculate the SHA-1 of an array of big-endian words, and a bit length
+ */
+function core_sha1(x, len)
+{
+  /* append padding */
+  x[len >> 5] |= 0x80 << (24 - len % 32);
+  x[((len + 64 >> 9) << 4) + 15] = len;
+
+  var w = new Array(80);
+  var a =  1732584193;
+  var b = -271733879;
+  var c = -1732584194;
+  var d =  271733878;
+  var e = -1009589776;
+
+  var i, j, t, olda, oldb, oldc, oldd, olde;
+  for (i = 0; i < x.length; i += 16)
+  {
+    olda = a;
+    oldb = b;
+    oldc = c;
+    oldd = d;
+    olde = e;
+
+    for (j = 0; j < 80; j++)
+    {
+      if (j < 16) { w[j] = x[i + j]; }
+      else { w[j] = rol(w[j-3] ^ w[j-8] ^ w[j-14] ^ w[j-16], 1); }
+      t = safe_add(safe_add(rol(a, 5), sha1_ft(j, b, c, d)),
+                       safe_add(safe_add(e, w[j]), sha1_kt(j)));
+      e = d;
+      d = c;
+      c = rol(b, 30);
+      b = a;
+      a = t;
+    }
+
+    a = safe_add(a, olda);
+    b = safe_add(b, oldb);
+    c = safe_add(c, oldc);
+    d = safe_add(d, oldd);
+    e = safe_add(e, olde);
+  }
+  return [a, b, c, d, e];
+}
+
+/*
+ * Perform the appropriate triplet combination function for the current
+ * iteration
+ */
+function sha1_ft(t, b, c, d)
+{
+  if (t < 20) { return (b & c) | ((~b) & d); }
+  if (t < 40) { return b ^ c ^ d; }
+  if (t < 60) { return (b & c) | (b & d) | (c & d); }
+  return b ^ c ^ d;
+}
+
+/*
+ * Determine the appropriate additive constant for the current iteration
+ */
+function sha1_kt(t)
+{
+  return (t < 20) ?  1518500249 : (t < 40) ?  1859775393 :
+         (t < 60) ? -1894007588 : -899497514;
+}
+
+/*
+ * Calculate the HMAC-SHA1 of a key and some data
+ */
+function core_hmac_sha1(key, data)
+{
+  var bkey = str2binb(key);
+  if (bkey.length > 16) { bkey = core_sha1(bkey, key.length * 8); }
+
+  var ipad = new Array(16), opad = new Array(16);
+  for (var i = 0; i < 16; i++)
+  {
+    ipad[i] = bkey[i] ^ 0x36363636;
+    opad[i] = bkey[i] ^ 0x5C5C5C5C;
+  }
+
+  var hash = core_sha1(ipad.concat(str2binb(data)), 512 + data.length * 8);
+  return core_sha1(opad.concat(hash), 512 + 160);
+}
+
+/*
+ * Add integers, wrapping at 2^32. This uses 16-bit operations internally
+ * to work around bugs in some JS interpreters.
+ */
+function safe_add(x, y)
+{
+  var lsw = (x & 0xFFFF) + (y & 0xFFFF);
+  var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
+  return (msw << 16) | (lsw & 0xFFFF);
+}
+
+/*
+ * Bitwise rotate a 32-bit number to the left.
+ */
+function rol(num, cnt)
+{
+  return (num << cnt) | (num >>> (32 - cnt));
+}
+
+/*
+ * Convert an 8-bit or 16-bit string to an array of big-endian words
+ * In 8-bit function, characters >255 have their hi-byte silently ignored.
+ */
+function str2binb(str)
+{
+  var bin = [];
+  var mask = 255;
+  for (var i = 0; i < str.length * 8; i += 8)
+  {
+    bin[i>>5] |= (str.charCodeAt(i / 8) & mask) << (24 - i%32);
+  }
+  return bin;
+}
+
+/*
+ * Convert an array of big-endian words to a string
+ */
+function binb2str(bin)
+{
+  var str = "";
+  var mask = 255;
+  for (var i = 0; i < bin.length * 32; i += 8)
+  {
+    str += String.fromCharCode((bin[i>>5] >>> (24 - i%32)) & mask);
+  }
+  return str;
+}
+
+/*
+ * Convert an array of big-endian words to a base-64 string
+ */
+function binb2b64(binarray)
+{
+  var tab = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
+  var str = "";
+  var triplet, j;
+  for (var i = 0; i < binarray.length * 4; i += 3)
+  {
+    triplet = (((binarray[i   >> 2] >> 8 * (3 -  i   %4)) & 0xFF) << 16) |
+              (((binarray[i+1 >> 2] >> 8 * (3 - (i+1)%4)) & 0xFF) << 8 ) |
+               ((binarray[i+2 >> 2] >> 8 * (3 - (i+2)%4)) & 0xFF);
+    for (j = 0; j < 4; j++)
+    {
+      if (i * 8 + j * 6 > binarray.length * 32) { str += "="; }
+      else { str += tab.charAt((triplet >> 6*(3-j)) & 0x3F); }
+    }
+  }
+  return str;
+}
+
+/*!
+ * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
+ * Digest Algorithm, as defined in RFC 1321.
+ * Version 2.1 Copyright (C) Paul Johnston 1999 - 2002.
+ * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
+ * Distributed under the BSD License
+ * See http://pajhome.org.uk/crypt/md5 for more info.
+ */
+
+/*
+ * Everything that isn't used by Strophe has been stripped here!
+ */
+
+var MD5 = (function () {
+    /*
+     * Add integers, wrapping at 2^32. This uses 16-bit operations internally
+     * to work around bugs in some JS interpreters.
+     */
+    var safe_add = function (x, y) {
+        var lsw = (x & 0xFFFF) + (y & 0xFFFF);
+        var msw = (x >> 16) + (y >> 16) + (lsw >> 16);
+        return (msw << 16) | (lsw & 0xFFFF);
+    };
+
+    /*
+     * Bitwise rotate a 32-bit number to the left.
+     */
+    var bit_rol = function (num, cnt) {
+        return (num << cnt) | (num >>> (32 - cnt));
+    };
+
+    /*
+     * Convert a string to an array of little-endian words
+     */
+    var str2binl = function (str) {
+        var bin = [];
+        for(var i = 0; i < str.length * 8; i += 8)
+        {
+            bin[i>>5] |= (str.charCodeAt(i / 8) & 255) << (i%32);
+        }
+        return bin;
+    };
+
+    /*
+     * Convert an array of little-endian words to a string
+     */
+    var binl2str = function (bin) {
+        var str = "";
+        for(var i = 0; i < bin.length * 32; i += 8)
+        {
+            str += String.fromCharCode((bin[i>>5] >>> (i % 32)) & 255);
+        }
+        return str;
+    };
+
+    /*
+     * Convert an array of little-endian words to a hex string.
+     */
+    var binl2hex = function (binarray) {
+        var hex_tab = "0123456789abcdef";
+        var str = "";
+        for(var i = 0; i < binarray.length * 4; i++)
+        {
+            str += hex_tab.charAt((binarray[i>>2] >> ((i%4)*8+4)) & 0xF) +
+                hex_tab.charAt((binarray[i>>2] >> ((i%4)*8  )) & 0xF);
+        }
+        return str;
+    };
+
+    /*
+     * These functions implement the four basic operations the algorithm uses.
+     */
+    var md5_cmn = function (q, a, b, x, s, t) {
+        return safe_add(bit_rol(safe_add(safe_add(a, q),safe_add(x, t)), s),b);
+    };
+
+    var md5_ff = function (a, b, c, d, x, s, t) {
+        return md5_cmn((b & c) | ((~b) & d), a, b, x, s, t);
+    };
+
+    var md5_gg = function (a, b, c, d, x, s, t) {
+        return md5_cmn((b & d) | (c & (~d)), a, b, x, s, t);
+    };
+
+    var md5_hh = function (a, b, c, d, x, s, t) {
+        return md5_cmn(b ^ c ^ d, a, b, x, s, t);
+    };
+
+    var md5_ii = function (a, b, c, d, x, s, t) {
+        return md5_cmn(c ^ (b | (~d)), a, b, x, s, t);
+    };
+
+    /*
+     * Calculate the MD5 of an array of little-endian words, and a bit length
+     */
+    var core_md5 = function (x, len) {
+        /* append padding */
+        x[len >> 5] |= 0x80 << ((len) % 32);
+        x[(((len + 64) >>> 9) << 4) + 14] = len;
+
+        var a =  1732584193;
+        var b = -271733879;
+        var c = -1732584194;
+        var d =  271733878;
+
+        var olda, oldb, oldc, oldd;
+        for (var i = 0; i < x.length; i += 16)
+        {
+            olda = a;
+            oldb = b;
+            oldc = c;
+            oldd = d;
+
+            a = md5_ff(a, b, c, d, x[i+ 0], 7 , -680876936);
+            d = md5_ff(d, a, b, c, x[i+ 1], 12, -389564586);
+            c = md5_ff(c, d, a, b, x[i+ 2], 17,  606105819);
+            b = md5_ff(b, c, d, a, x[i+ 3], 22, -1044525330);
+            a = md5_ff(a, b, c, d, x[i+ 4], 7 , -176418897);
+            d = md5_ff(d, a, b, c, x[i+ 5], 12,  1200080426);
+            c = md5_ff(c, d, a, b, x[i+ 6], 17, -1473231341);
+            b = md5_ff(b, c, d, a, x[i+ 7], 22, -45705983);
+            a = md5_ff(a, b, c, d, x[i+ 8], 7 ,  1770035416);
+            d = md5_ff(d, a, b, c, x[i+ 9], 12, -1958414417);
+            c = md5_ff(c, d, a, b, x[i+10], 17, -42063);
+            b = md5_ff(b, c, d, a, x[i+11], 22, -1990404162);
+            a = md5_ff(a, b, c, d, x[i+12], 7 ,  1804603682);
+            d = md5_ff(d, a, b, c, x[i+13], 12, -40341101);
+            c = md5_ff(c, d, a, b, x[i+14], 17, -1502002290);
+            b = md5_ff(b, c, d, a, x[i+15], 22,  1236535329);
+
+            a = md5_gg(a, b, c, d, x[i+ 1], 5 , -165796510);
+            d = md5_gg(d, a, b, c, x[i+ 6], 9 , -1069501632);
+            c = md5_gg(c, d, a, b, x[i+11], 14,  643717713);
+            b = md5_gg(b, c, d, a, x[i+ 0], 20, -373897302);
+            a = md5_gg(a, b, c, d, x[i+ 5], 5 , -701558691);
+            d = md5_gg(d, a, b, c, x[i+10], 9 ,  38016083);
+            c = md5_gg(c, d, a, b, x[i+15], 14, -660478335);
+            b = md5_gg(b, c, d, a, x[i+ 4], 20, -405537848);
+            a = md5_gg(a, b, c, d, x[i+ 9], 5 ,  568446438);
+            d = md5_gg(d, a, b, c, x[i+14], 9 , -1019803690);
+            c = md5_gg(c, d, a, b, x[i+ 3], 14, -187363961);
+            b = md5_gg(b, c, d, a, x[i+ 8], 20,  1163531501);
+            a = md5_gg(a, b, c, d, x[i+13], 5 , -1444681467);
+            d = md5_gg(d, a, b, c, x[i+ 2], 9 , -51403784);
+            c = md5_gg(c, d, a, b, x[i+ 7], 14,  1735328473);
+            b = md5_gg(b, c, d, a, x[i+12], 20, -1926607734);
+
+            a = md5_hh(a, b, c, d, x[i+ 5], 4 , -378558);
+            d = md5_hh(d, a, b, c, x[i+ 8], 11, -2022574463);
+            c = md5_hh(c, d, a, b, x[i+11], 16,  1839030562);
+            b = md5_hh(b, c, d, a, x[i+14], 23, -35309556);
+            a = md5_hh(a, b, c, d, x[i+ 1], 4 , -1530992060);
+            d = md5_hh(d, a, b, c, x[i+ 4], 11,  1272893353);
+            c = md5_hh(c, d, a, b, x[i+ 7], 16, -155497632);
+            b = md5_hh(b, c, d, a, x[i+10], 23, -1094730640);
+            a = md5_hh(a, b, c, d, x[i+13], 4 ,  681279174);
+            d = md5_hh(d, a, b, c, x[i+ 0], 11, -358537222);
+            c = md5_hh(c, d, a, b, x[i+ 3], 16, -722521979);
+            b = md5_hh(b, c, d, a, x[i+ 6], 23,  76029189);
+            a = md5_hh(a, b, c, d, x[i+ 9], 4 , -640364487);
+            d = md5_hh(d, a, b, c, x[i+12], 11, -421815835);
+            c = md5_hh(c, d, a, b, x[i+15], 16,  530742520);
+            b = md5_hh(b, c, d, a, x[i+ 2], 23, -995338651);
+
+            a = md5_ii(a, b, c, d, x[i+ 0], 6 , -198630844);
+            d = md5_ii(d, a, b, c, x[i+ 7], 10,  1126891415);
+            c = md5_ii(c, d, a, b, x[i+14], 15, -1416354905);
+            b = md5_ii(b, c, d, a, x[i+ 5], 21, -57434055);
+            a = md5_ii(a, b, c, d, x[i+12], 6 ,  1700485571);
+            d = md5_ii(d, a, b, c, x[i+ 3], 10, -1894986606);
+            c = md5_ii(c, d, a, b, x[i+10], 15, -1051523);
+            b = md5_ii(b, c, d, a, x[i+ 1], 21, -2054922799);
+            a = md5_ii(a, b, c, d, x[i+ 8], 6 ,  1873313359);
+            d = md5_ii(d, a, b, c, x[i+15], 10, -30611744);
+            c = md5_ii(c, d, a, b, x[i+ 6], 15, -1560198380);
+            b = md5_ii(b, c, d, a, x[i+13], 21,  1309151649);
+            a = md5_ii(a, b, c, d, x[i+ 4], 6 , -145523070);
+            d = md5_ii(d, a, b, c, x[i+11], 10, -1120210379);
+            c = md5_ii(c, d, a, b, x[i+ 2], 15,  718787259);
+            b = md5_ii(b, c, d, a, x[i+ 9], 21, -343485551);
+
+            a = safe_add(a, olda);
+            b = safe_add(b, oldb);
+            c = safe_add(c, oldc);
+            d = safe_add(d, oldd);
+        }
+        return [a, b, c, d];
+    };
+
+
+    var obj = {
+        /*
+         * These are the functions you'll usually want to call.
+         * They take string arguments and return either hex or base-64 encoded
+         * strings.
+         */
+        hexdigest: function (s) {
+            return binl2hex(core_md5(str2binl(s), s.length * 8));
+        },
+
+        hash: function (s) {
+            return binl2str(core_md5(str2binl(s), s.length * 8));
+        }
+    };
+
+    return obj;
+})();
+
+/*!
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+
+/* jshint undef: true, unused: true:, noarg: true, latedef: true */
+/*global document, window, setTimeout, clearTimeout, console,
+    ActiveXObject, Base64, MD5, DOMParser */
+// from sha1.js
+/*global core_hmac_sha1, binb2str, str_hmac_sha1, str_sha1, b64_hmac_sha1*/
+
+/** File: strophe.js
+ *  A JavaScript library for XMPP BOSH/XMPP over Websocket.
+ *
+ *  This is the JavaScript version of the Strophe library.  Since JavaScript
+ *  had no facilities for persistent TCP connections, this library uses
+ *  Bidirectional-streams Over Synchronous HTTP (BOSH) to emulate
+ *  a persistent, stateful, two-way connection to an XMPP server.  More
+ *  information on BOSH can be found in XEP 124.
+ *
+ *  This version of Strophe also works with WebSockets.
+ *  For more information on XMPP-over WebSocket see this RFC draft:
+ *  http://tools.ietf.org/html/draft-ietf-xmpp-websocket-00
+ */
+
+/** PrivateFunction: Function.prototype.bind
+ *  Bind a function to an instance.
+ *
+ *  This Function object extension method creates a bound method similar
+ *  to those in Python.  This means that the 'this' object will point
+ *  to the instance you want.  See
+ *  <a href='https://developer.mozilla.org/en/JavaScript/Reference/Global_Objects/Function/bind'>MDC's bind() documentation</a> and
+ *  <a href='http://benjamin.smedbergs.us/blog/2007-01-03/bound-functions-and-function-imports-in-javascript/'>Bound Functions and Function Imports in JavaScript</a>
+ *  for a complete explanation.
+ *
+ *  This extension already exists in some browsers (namely, Firefox 3), but
+ *  we provide it to support those that don't.
+ *
+ *  Parameters:
+ *    (Object) obj - The object that will become 'this' in the bound function.
+ *    (Object) argN - An option argument that will be prepended to the
+ *      arguments given for the function call
+ *
+ *  Returns:
+ *    The bound function.
+ */
+if (!Function.prototype.bind) {
+    Function.prototype.bind = function (obj /*, arg1, arg2, ... */)
+    {
+        var func = this;
+        var _slice = Array.prototype.slice;
+        var _concat = Array.prototype.concat;
+        var _args = _slice.call(arguments, 1);
+
+        return function () {
+            return func.apply(obj ? obj : this,
+                              _concat.call(_args,
+                                           _slice.call(arguments, 0)));
+        };
+    };
+}
+
+/** PrivateFunction: Array.prototype.indexOf
+ *  Return the index of an object in an array.
+ *
+ *  This function is not supplied by some JavaScript implementations, so
+ *  we provide it if it is missing.  This code is from:
+ *  http://developer.mozilla.org/En/Core_JavaScript_1.5_Reference:Objects:Array:indexOf
+ *
+ *  Parameters:
+ *    (Object) elt - The object to look for.
+ *    (Integer) from - The index from which to start looking. (optional).
+ *
+ *  Returns:
+ *    The index of elt in the array or -1 if not found.
+ */
+if (!Array.prototype.indexOf)
+{
+    Array.prototype.indexOf = function(elt /*, from*/)
+    {
+        var len = this.length;
+
+        var from = Number(arguments[1]) || 0;
+        from = (from < 0) ? Math.ceil(from) : Math.floor(from);
+        if (from < 0) {
+            from += len;
+        }
+
+        for (; from < len; from++) {
+            if (from in this && this[from] === elt) {
+                return from;
+            }
+        }
+
+        return -1;
+    };
+}
+
+/* All of the Strophe globals are defined in this special function below so
+ * that references to the globals become closures.  This will ensure that
+ * on page reload, these references will still be available to callbacks
+ * that are still executing.
+ */
+
+(function (callback) {
+var Strophe;
+
+/** Function: $build
+ *  Create a Strophe.Builder.
+ *  This is an alias for 'new Strophe.Builder(name, attrs)'.
+ *
+ *  Parameters:
+ *    (String) name - The root element name.
+ *    (Object) attrs - The attributes for the root element in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $build(name, attrs) { return new Strophe.Builder(name, attrs); }
+/** Function: $msg
+ *  Create a Strophe.Builder with a <message/> element as the root.
+ *
+ *  Parmaeters:
+ *    (Object) attrs - The <message/> element attributes in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $msg(attrs) { return new Strophe.Builder("message", attrs); }
+/** Function: $iq
+ *  Create a Strophe.Builder with an <iq/> element as the root.
+ *
+ *  Parameters:
+ *    (Object) attrs - The <iq/> element attributes in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $iq(attrs) { return new Strophe.Builder("iq", attrs); }
+/** Function: $pres
+ *  Create a Strophe.Builder with a <presence/> element as the root.
+ *
+ *  Parameters:
+ *    (Object) attrs - The <presence/> element attributes in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder object.
+ */
+function $pres(attrs) { return new Strophe.Builder("presence", attrs); }
+
+/** Class: Strophe
+ *  An object container for all Strophe library functions.
+ *
+ *  This class is just a container for all the objects and constants
+ *  used in the library.  It is not meant to be instantiated, but to
+ *  provide a namespace for library objects, constants, and functions.
+ */
+Strophe = {
+    /** Constant: VERSION
+     *  The version of the Strophe library. Unreleased builds will have
+     *  a version of head-HASH where HASH is a partial revision.
+     */
+    VERSION: "1.1.3",
+
+    /** Constants: XMPP Namespace Constants
+     *  Common namespace constants from the XMPP RFCs and XEPs.
+     *
+     *  NS.HTTPBIND - HTTP BIND namespace from XEP 124.
+     *  NS.BOSH - BOSH namespace from XEP 206.
+     *  NS.CLIENT - Main XMPP client namespace.
+     *  NS.AUTH - Legacy authentication namespace.
+     *  NS.ROSTER - Roster operations namespace.
+     *  NS.PROFILE - Profile namespace.
+     *  NS.DISCO_INFO - Service discovery info namespace from XEP 30.
+     *  NS.DISCO_ITEMS - Service discovery items namespace from XEP 30.
+     *  NS.MUC - Multi-User Chat namespace from XEP 45.
+     *  NS.SASL - XMPP SASL namespace from RFC 3920.
+     *  NS.STREAM - XMPP Streams namespace from RFC 3920.
+     *  NS.BIND - XMPP Binding namespace from RFC 3920.
+     *  NS.SESSION - XMPP Session namespace from RFC 3920.
+     *  NS.XHTML_IM - XHTML-IM namespace from XEP 71.
+     *  NS.XHTML - XHTML body namespace from XEP 71.
+     */
+    NS: {
+        HTTPBIND: "http://jabber.org/protocol/httpbind",
+        BOSH: "urn:xmpp:xbosh",
+        CLIENT: "jabber:client",
+        AUTH: "jabber:iq:auth",
+        ROSTER: "jabber:iq:roster",
+        PROFILE: "jabber:iq:profile",
+        DISCO_INFO: "http://jabber.org/protocol/disco#info",
+        DISCO_ITEMS: "http://jabber.org/protocol/disco#items",
+        MUC: "http://jabber.org/protocol/muc",
+        SASL: "urn:ietf:params:xml:ns:xmpp-sasl",
+        STREAM: "http://etherx.jabber.org/streams",
+        BIND: "urn:ietf:params:xml:ns:xmpp-bind",
+        SESSION: "urn:ietf:params:xml:ns:xmpp-session",
+        VERSION: "jabber:iq:version",
+        STANZAS: "urn:ietf:params:xml:ns:xmpp-stanzas",
+        XHTML_IM: "http://jabber.org/protocol/xhtml-im",
+        XHTML: "http://www.w3.org/1999/xhtml"
+    },
+
+
+    /** Constants: XHTML_IM Namespace
+     *  contains allowed tags, tag attributes, and css properties.
+     *  Used in the createHtml function to filter incoming html into the allowed XHTML-IM subset.
+     *  See http://xmpp.org/extensions/xep-0071.html#profile-summary for the list of recommended
+     *  allowed tags and their attributes.
+     */
+    XHTML: {
+                tags: ['a','blockquote','br','cite','em','img','li','ol','p','span','strong','ul','body'],
+                attributes: {
+                        'a':          ['href'],
+                        'blockquote': ['style'],
+                        'br':         [],
+                        'cite':       ['style'],
+                        'em':         [],
+                        'img':        ['src', 'alt', 'style', 'height', 'width'],
+                        'li':         ['style'],
+                        'ol':         ['style'],
+                        'p':          ['style'],
+                        'span':       ['style'],
+                        'strong':     [],
+                        'ul':         ['style'],
+                        'body':       []
+                },
+                css: ['background-color','color','font-family','font-size','font-style','font-weight','margin-left','margin-right','text-align','text-decoration'],
+                validTag: function(tag)
+                {
+                        for(var i = 0; i < Strophe.XHTML.tags.length; i++) {
+                                if(tag == Strophe.XHTML.tags[i]) {
+                                        return true;
+                                }
+                        }
+                        return false;
+                },
+                validAttribute: function(tag, attribute)
+                {
+                        if(typeof Strophe.XHTML.attributes[tag] !== 'undefined' && Strophe.XHTML.attributes[tag].length > 0) {
+                                for(var i = 0; i < Strophe.XHTML.attributes[tag].length; i++) {
+                                        if(attribute == Strophe.XHTML.attributes[tag][i]) {
+                                                return true;
+                                        }
+                                }
+                        }
+                        return false;
+                },
+                validCSS: function(style)
+                {
+                        for(var i = 0; i < Strophe.XHTML.css.length; i++) {
+                                if(style == Strophe.XHTML.css[i]) {
+                                        return true;
+                                }
+                        }
+                        return false;
+                }
+    },
+
+    /** Constants: Connection Status Constants
+     *  Connection status constants for use by the connection handler
+     *  callback.
+     *
+     *  Status.ERROR - An error has occurred
+     *  Status.CONNECTING - The connection is currently being made
+     *  Status.CONNFAIL - The connection attempt failed
+     *  Status.AUTHENTICATING - The connection is authenticating
+     *  Status.AUTHFAIL - The authentication attempt failed
+     *  Status.CONNECTED - The connection has succeeded
+     *  Status.DISCONNECTED - The connection has been terminated
+     *  Status.DISCONNECTING - The connection is currently being terminated
+     *  Status.ATTACHED - The connection has been attached
+     */
+    Status: {
+        ERROR: 0,
+        CONNECTING: 1,
+        CONNFAIL: 2,
+        AUTHENTICATING: 3,
+        AUTHFAIL: 4,
+        CONNECTED: 5,
+        DISCONNECTED: 6,
+        DISCONNECTING: 7,
+        ATTACHED: 8
+    },
+
+    /** Constants: Log Level Constants
+     *  Logging level indicators.
+     *
+     *  LogLevel.DEBUG - Debug output
+     *  LogLevel.INFO - Informational output
+     *  LogLevel.WARN - Warnings
+     *  LogLevel.ERROR - Errors
+     *  LogLevel.FATAL - Fatal errors
+     */
+    LogLevel: {
+        DEBUG: 0,
+        INFO: 1,
+        WARN: 2,
+        ERROR: 3,
+        FATAL: 4
+    },
+
+    /** PrivateConstants: DOM Element Type Constants
+     *  DOM element types.
+     *
+     *  ElementType.NORMAL - Normal element.
+     *  ElementType.TEXT - Text data element.
+     *  ElementType.FRAGMENT - XHTML fragment element.
+     */
+    ElementType: {
+        NORMAL: 1,
+        TEXT: 3,
+        CDATA: 4,
+        FRAGMENT: 11
+    },
+
+    /** PrivateConstants: Timeout Values
+     *  Timeout values for error states.  These values are in seconds.
+     *  These should not be changed unless you know exactly what you are
+     *  doing.
+     *
+     *  TIMEOUT - Timeout multiplier. A waiting request will be considered
+     *      failed after Math.floor(TIMEOUT * wait) seconds have elapsed.
+     *      This defaults to 1.1, and with default wait, 66 seconds.
+     *  SECONDARY_TIMEOUT - Secondary timeout multiplier. In cases where
+     *      Strophe can detect early failure, it will consider the request
+     *      failed if it doesn't return after
+     *      Math.floor(SECONDARY_TIMEOUT * wait) seconds have elapsed.
+     *      This defaults to 0.1, and with default wait, 6 seconds.
+     */
+    TIMEOUT: 1.1,
+    SECONDARY_TIMEOUT: 0.1,
+
+    /** Function: addNamespace
+     *  This function is used to extend the current namespaces in
+     *  Strophe.NS.  It takes a key and a value with the key being the
+     *  name of the new namespace, with its actual value.
+     *  For example:
+     *  Strophe.addNamespace('PUBSUB', "http://jabber.org/protocol/pubsub");
+     *
+     *  Parameters:
+     *    (String) name - The name under which the namespace will be
+     *      referenced under Strophe.NS
+     *    (String) value - The actual namespace.
+     */
+    addNamespace: function (name, value)
+    {
+      Strophe.NS[name] = value;
+    },
+
+    /** Function: forEachChild
+     *  Map a function over some or all child elements of a given element.
+     *
+     *  This is a small convenience function for mapping a function over
+     *  some or all of the children of an element.  If elemName is null, all
+     *  children will be passed to the function, otherwise only children
+     *  whose tag names match elemName will be passed.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The element to operate on.
+     *    (String) elemName - The child element tag name filter.
+     *    (Function) func - The function to apply to each child.  This
+     *      function should take a single argument, a DOM element.
+     */
+    forEachChild: function (elem, elemName, func)
+    {
+        var i, childNode;
+
+        for (i = 0; i < elem.childNodes.length; i++) {
+            childNode = elem.childNodes[i];
+            if (childNode.nodeType == Strophe.ElementType.NORMAL &&
+                (!elemName || this.isTagEqual(childNode, elemName))) {
+                func(childNode);
+            }
+        }
+    },
+
+    /** Function: isTagEqual
+     *  Compare an element's tag name with a string.
+     *
+     *  This function is case insensitive.
+     *
+     *  Parameters:
+     *    (XMLElement) el - A DOM element.
+     *    (String) name - The element name.
+     *
+     *  Returns:
+     *    true if the element's tag name matches _el_, and false
+     *    otherwise.
+     */
+    isTagEqual: function (el, name)
+    {
+        return el.tagName.toLowerCase() == name.toLowerCase();
+    },
+
+    /** PrivateVariable: _xmlGenerator
+     *  _Private_ variable that caches a DOM document to
+     *  generate elements.
+     */
+    _xmlGenerator: null,
+
+    /** PrivateFunction: _makeGenerator
+     *  _Private_ function that creates a dummy XML DOM document to serve as
+     *  an element and text node generator.
+     */
+    _makeGenerator: function () {
+        var doc;
+
+        // IE9 does implement createDocument(); however, using it will cause the browser to leak memory on page unload.
+        // Here, we test for presence of createDocument() plus IE's proprietary documentMode attribute, which would be
+                // less than 10 in the case of IE9 and below.
+        if (document.implementation.createDocument === undefined ||
+                        document.implementation.createDocument && document.documentMode && document.documentMode < 10) {
+            doc = this._getIEXmlDom();
+            doc.appendChild(doc.createElement('strophe'));
+        } else {
+            doc = document.implementation
+                .createDocument('jabber:client', 'strophe', null);
+        }
+
+        return doc;
+    },
+
+    /** Function: xmlGenerator
+     *  Get the DOM document to generate elements.
+     *
+     *  Returns:
+     *    The currently used DOM document.
+     */
+    xmlGenerator: function () {
+        if (!Strophe._xmlGenerator) {
+            Strophe._xmlGenerator = Strophe._makeGenerator();
+        }
+        return Strophe._xmlGenerator;
+    },
+
+    /** PrivateFunction: _getIEXmlDom
+     *  Gets IE xml doc object
+     *
+     *  Returns:
+     *    A Microsoft XML DOM Object
+     *  See Also:
+     *    http://msdn.microsoft.com/en-us/library/ms757837%28VS.85%29.aspx
+     */
+    _getIEXmlDom : function() {
+        var doc = null;
+        var docStrings = [
+            "Msxml2.DOMDocument.6.0",
+            "Msxml2.DOMDocument.5.0",
+            "Msxml2.DOMDocument.4.0",
+            "MSXML2.DOMDocument.3.0",
+            "MSXML2.DOMDocument",
+            "MSXML.DOMDocument",
+            "Microsoft.XMLDOM"
+        ];
+
+        for (var d = 0; d < docStrings.length; d++) {
+            if (doc === null) {
+                try {
+                    doc = new ActiveXObject(docStrings[d]);
+                } catch (e) {
+                    doc = null;
+                }
+            } else {
+                break;
+            }
+        }
+
+        return doc;
+    },
+
+    /** Function: xmlElement
+     *  Create an XML DOM element.
+     *
+     *  This function creates an XML DOM element correctly across all
+     *  implementations. Note that these are not HTML DOM elements, which
+     *  aren't appropriate for XMPP stanzas.
+     *
+     *  Parameters:
+     *    (String) name - The name for the element.
+     *    (Array|Object) attrs - An optional array or object containing
+     *      key/value pairs to use as element attributes. The object should
+     *      be in the format {'key': 'value'} or {key: 'value'}. The array
+     *      should have the format [['key1', 'value1'], ['key2', 'value2']].
+     *    (String) text - The text child data for the element.
+     *
+     *  Returns:
+     *    A new XML DOM element.
+     */
+    xmlElement: function (name)
+    {
+        if (!name) { return null; }
+
+        var node = Strophe.xmlGenerator().createElement(name);
+
+        // FIXME: this should throw errors if args are the wrong type or
+        // there are more than two optional args
+        var a, i, k;
+        for (a = 1; a < arguments.length; a++) {
+            if (!arguments[a]) { continue; }
+            if (typeof(arguments[a]) == "string" ||
+                typeof(arguments[a]) == "number") {
+                node.appendChild(Strophe.xmlTextNode(arguments[a]));
+            } else if (typeof(arguments[a]) == "object" &&
+                       typeof(arguments[a].sort) == "function") {
+                for (i = 0; i < arguments[a].length; i++) {
+                    if (typeof(arguments[a][i]) == "object" &&
+                        typeof(arguments[a][i].sort) == "function") {
+                        node.setAttribute(arguments[a][i][0],
+                                          arguments[a][i][1]);
+                    }
+                }
+            } else if (typeof(arguments[a]) == "object") {
+                for (k in arguments[a]) {
+                    if (arguments[a].hasOwnProperty(k)) {
+                        node.setAttribute(k, arguments[a][k]);
+                    }
+                }
+            }
+        }
+
+        return node;
+    },
+
+    /*  Function: xmlescape
+     *  Excapes invalid xml characters.
+     *
+     *  Parameters:
+     *     (String) text - text to escape.
+     *
+     *  Returns:
+     *      Escaped text.
+     */
+    xmlescape: function(text)
+    {
+        text = text.replace(/\&/g, "&");
+        text = text.replace(/</g,  "<");
+        text = text.replace(/>/g,  ">");
+        text = text.replace(/'/g,  "'");
+        text = text.replace(/"/g,  """);
+        return text;
+    },
+
+    /** Function: xmlTextNode
+     *  Creates an XML DOM text node.
+     *
+     *  Provides a cross implementation version of document.createTextNode.
+     *
+     *  Parameters:
+     *    (String) text - The content of the text node.
+     *
+     *  Returns:
+     *    A new XML DOM text node.
+     */
+    xmlTextNode: function (text)
+    {
+        return Strophe.xmlGenerator().createTextNode(text);
+    },
+
+    /** Function: xmlHtmlNode
+     *  Creates an XML DOM html node.
+     *
+     *  Parameters:
+     *    (String) html - The content of the html node.
+     *
+     *  Returns:
+     *    A new XML DOM text node.
+     */
+    xmlHtmlNode: function (html)
+    {
+        var node;
+        //ensure text is escaped
+        if (window.DOMParser) {
+            var parser = new DOMParser();
+            node = parser.parseFromString(html, "text/xml");
+        } else {
+            node = new ActiveXObject("Microsoft.XMLDOM");
+            node.async="false";
+            node.loadXML(html);
+        }
+        return node;
+    },
+
+    /** Function: getText
+     *  Get the concatenation of all text children of an element.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    A String with the concatenated text of all text element children.
+     */
+    getText: function (elem)
+    {
+        if (!elem) { return null; }
+
+        var str = "";
+        if (elem.childNodes.length === 0 && elem.nodeType ==
+            Strophe.ElementType.TEXT) {
+            str += elem.nodeValue;
+        }
+
+        for (var i = 0; i < elem.childNodes.length; i++) {
+            if (elem.childNodes[i].nodeType == Strophe.ElementType.TEXT) {
+                str += elem.childNodes[i].nodeValue;
+            }
+        }
+
+        return Strophe.xmlescape(str);
+    },
+
+    /** Function: copyElement
+     *  Copy an XML DOM element.
+     *
+     *  This function copies a DOM element and all its descendants and returns
+     *  the new copy.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    A new, copied DOM element tree.
+     */
+    copyElement: function (elem)
+    {
+        var i, el;
+        if (elem.nodeType == Strophe.ElementType.NORMAL) {
+            el = Strophe.xmlElement(elem.tagName);
+
+            for (i = 0; i < elem.attributes.length; i++) {
+                el.setAttribute(elem.attributes[i].nodeName.toLowerCase(),
+                                elem.attributes[i].value);
+            }
+
+            for (i = 0; i < elem.childNodes.length; i++) {
+                el.appendChild(Strophe.copyElement(elem.childNodes[i]));
+            }
+        } else if (elem.nodeType == Strophe.ElementType.TEXT) {
+            el = Strophe.xmlGenerator().createTextNode(elem.nodeValue);
+        }
+
+        return el;
+    },
+
+
+    /** Function: createHtml
+     *  Copy an HTML DOM element into an XML DOM.
+     *
+     *  This function copies a DOM element and all its descendants and returns
+     *  the new copy.
+     *
+     *  Parameters:
+     *    (HTMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    A new, copied DOM element tree.
+     */
+    createHtml: function (elem)
+    {
+        var i, el, j, tag, attribute, value, css, cssAttrs, attr, cssName, cssValue;
+        if (elem.nodeType == Strophe.ElementType.NORMAL) {
+            tag = elem.nodeName.toLowerCase();
+            if(Strophe.XHTML.validTag(tag)) {
+                try {
+                    el = Strophe.xmlElement(tag);
+                    for(i = 0; i < Strophe.XHTML.attributes[tag].length; i++) {
+                        attribute = Strophe.XHTML.attributes[tag][i];
+                        value = elem.getAttribute(attribute);
+                        if(typeof value == 'undefined' || value === null || value === '' || value === false || value === 0) {
+                            continue;
+                        }
+                        if(attribute == 'style' && typeof value == 'object') {
+                            if(typeof value.cssText != 'undefined') {
+                                value = value.cssText; // we're dealing with IE, need to get CSS out
+                            }
+                        }
+                        // filter out invalid css styles
+                        if(attribute == 'style') {
+                            css = [];
+                            cssAttrs = value.split(';');
+                            for(j = 0; j < cssAttrs.length; j++) {
+                                attr = cssAttrs[j].split(':');
+                                cssName = attr[0].replace(/^\s*/, "").replace(/\s*$/, "").toLowerCase();
+                                if(Strophe.XHTML.validCSS(cssName)) {
+                                    cssValue = attr[1].replace(/^\s*/, "").replace(/\s*$/, "");
+                                    css.push(cssName + ': ' + cssValue);
+                                }
+                            }
+                            if(css.length > 0) {
+                                value = css.join('; ');
+                                el.setAttribute(attribute, value);
+                            }
+                        } else {
+                            el.setAttribute(attribute, value);
+                        }
+                    }
+
+                    for (i = 0; i < elem.childNodes.length; i++) {
+                        el.appendChild(Strophe.createHtml(elem.childNodes[i]));
+                    }
+                } catch(e) { // invalid elements
+                  el = Strophe.xmlTextNode('');
+                }
+            } else {
+                el = Strophe.xmlGenerator().createDocumentFragment();
+                for (i = 0; i < elem.childNodes.length; i++) {
+                    el.appendChild(Strophe.createHtml(elem.childNodes[i]));
+                }
+            }
+        } else if (elem.nodeType == Strophe.ElementType.FRAGMENT) {
+            el = Strophe.xmlGenerator().createDocumentFragment();
+            for (i = 0; i < elem.childNodes.length; i++) {
+                el.appendChild(Strophe.createHtml(elem.childNodes[i]));
+            }
+        } else if (elem.nodeType == Strophe.ElementType.TEXT) {
+            el = Strophe.xmlTextNode(elem.nodeValue);
+        }
+
+        return el;
+    },
+
+    /** Function: escapeNode
+     *  Escape the node part (also called local part) of a JID.
+     *
+     *  Parameters:
+     *    (String) node - A node (or local part).
+     *
+     *  Returns:
+     *    An escaped node (or local part).
+     */
+    escapeNode: function (node)
+    {
+        return node.replace(/^\s+|\s+$/g, '')
+            .replace(/\\/g,  "\\5c")
+            .replace(/ /g,   "\\20")
+            .replace(/\"/g,  "\\22")
+            .replace(/\&/g,  "\\26")
+            .replace(/\'/g,  "\\27")
+            .replace(/\//g,  "\\2f")
+            .replace(/:/g,   "\\3a")
+            .replace(/</g,   "\\3c")
+            .replace(/>/g,   "\\3e")
+            .replace(/@/g,   "\\40");
+    },
+
+    /** Function: unescapeNode
+     *  Unescape a node part (also called local part) of a JID.
+     *
+     *  Parameters:
+     *    (String) node - A node (or local part).
+     *
+     *  Returns:
+     *    An unescaped node (or local part).
+     */
+    unescapeNode: function (node)
+    {
+        return node.replace(/\\20/g, " ")
+            .replace(/\\22/g, '"')
+            .replace(/\\26/g, "&")
+            .replace(/\\27/g, "'")
+            .replace(/\\2f/g, "/")
+            .replace(/\\3a/g, ":")
+            .replace(/\\3c/g, "<")
+            .replace(/\\3e/g, ">")
+            .replace(/\\40/g, "@")
+            .replace(/\\5c/g, "\\");
+    },
+
+    /** Function: getNodeFromJid
+     *  Get the node portion of a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the node.
+     */
+    getNodeFromJid: function (jid)
+    {
+        if (jid.indexOf("@") < 0) { return null; }
+        return jid.split("@")[0];
+    },
+
+    /** Function: getDomainFromJid
+     *  Get the domain portion of a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the domain.
+     */
+    getDomainFromJid: function (jid)
+    {
+        var bare = Strophe.getBareJidFromJid(jid);
+        if (bare.indexOf("@") < 0) {
+            return bare;
+        } else {
+            var parts = bare.split("@");
+            parts.splice(0, 1);
+            return parts.join('@');
+        }
+    },
+
+    /** Function: getResourceFromJid
+     *  Get the resource portion of a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the resource.
+     */
+    getResourceFromJid: function (jid)
+    {
+        var s = jid.split("/");
+        if (s.length < 2) { return null; }
+        s.splice(0, 1);
+        return s.join('/');
+    },
+
+    /** Function: getBareJidFromJid
+     *  Get the bare JID from a JID String.
+     *
+     *  Parameters:
+     *    (String) jid - A JID.
+     *
+     *  Returns:
+     *    A String containing the bare JID.
+     */
+    getBareJidFromJid: function (jid)
+    {
+        return jid ? jid.split("/")[0] : null;
+    },
+
+    /** Function: log
+     *  User overrideable logging function.
+     *
+     *  This function is called whenever the Strophe library calls any
+     *  of the logging functions.  The default implementation of this
+     *  function does nothing.  If client code wishes to handle the logging
+     *  messages, it should override this with
+     *  > Strophe.log = function (level, msg) {
+     *  >   (user code here)
+     *  > };
+     *
+     *  Please note that data sent and received over the wire is logged
+     *  via Strophe.Connection.rawInput() and Strophe.Connection.rawOutput().
+     *
+     *  The different levels and their meanings are
+     *
+     *    DEBUG - Messages useful for debugging purposes.
+     *    INFO - Informational messages.  This is mostly information like
+     *      'disconnect was called' or 'SASL auth succeeded'.
+     *    WARN - Warnings about potential problems.  This is mostly used
+     *      to report transient connection errors like request timeouts.
+     *    ERROR - Some error occurred.
+     *    FATAL - A non-recoverable fatal error occurred.
+     *
+     *  Parameters:
+     *    (Integer) level - The log level of the log message.  This will
+     *      be one of the values in Strophe.LogLevel.
+     *    (String) msg - The log message.
+     */
+    /* jshint ignore:start */
+    log: function (level, msg)
+    {
+        return;
+    },
+    /* jshint ignore:end */
+
+    /** Function: debug
+     *  Log a message at the Strophe.LogLevel.DEBUG level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    debug: function(msg)
+    {
+        this.log(this.LogLevel.DEBUG, msg);
+    },
+
+    /** Function: info
+     *  Log a message at the Strophe.LogLevel.INFO level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    info: function (msg)
+    {
+        this.log(this.LogLevel.INFO, msg);
+    },
+
+    /** Function: warn
+     *  Log a message at the Strophe.LogLevel.WARN level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    warn: function (msg)
+    {
+        this.log(this.LogLevel.WARN, msg);
+    },
+
+    /** Function: error
+     *  Log a message at the Strophe.LogLevel.ERROR level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    error: function (msg)
+    {
+        this.log(this.LogLevel.ERROR, msg);
+    },
+
+    /** Function: fatal
+     *  Log a message at the Strophe.LogLevel.FATAL level.
+     *
+     *  Parameters:
+     *    (String) msg - The log message.
+     */
+    fatal: function (msg)
+    {
+        this.log(this.LogLevel.FATAL, msg);
+    },
+
+    /** Function: serialize
+     *  Render a DOM element and all descendants to a String.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    The serialized element tree as a String.
+     */
+    serialize: function (elem)
+    {
+        var result;
+
+        if (!elem) { return null; }
+
+        if (typeof(elem.tree) === "function") {
+            elem = elem.tree();
+        }
+
+        var nodeName = elem.nodeName;
+        var i, child;
+
+        if (elem.getAttribute("_realname")) {
+            nodeName = elem.getAttribute("_realname");
+        }
+
+        result = "<" + nodeName;
+        for (i = 0; i < elem.attributes.length; i++) {
+               if(elem.attributes[i].nodeName != "_realname") {
+                 result += " " + elem.attributes[i].nodeName.toLowerCase() +
+                "='" + elem.attributes[i].value
+                    .replace(/&/g, "&")
+                       .replace(/\'/g, "'")
+                       .replace(/>/g, ">")
+                       .replace(/</g, "<") + "'";
+               }
+        }
+
+        if (elem.childNodes.length > 0) {
+            result += ">";
+            for (i = 0; i < elem.childNodes.length; i++) {
+                child = elem.childNodes[i];
+                switch( child.nodeType ){
+                  case Strophe.ElementType.NORMAL:
+                    // normal element, so recurse
+                    result += Strophe.serialize(child);
+                    break;
+                  case Strophe.ElementType.TEXT:
+                    // text element to escape values
+                    result += Strophe.xmlescape(child.nodeValue);
+                    break;
+                  case Strophe.ElementType.CDATA:
+                    // cdata section so don't escape values
+                    result += "<![CDATA["+child.nodeValue+"]]>";
+                }
+            }
+            result += "</" + nodeName + ">";
+        } else {
+            result += "/>";
+        }
+
+        return result;
+    },
+
+    /** PrivateVariable: _requestId
+     *  _Private_ variable that keeps track of the request ids for
+     *  connections.
+     */
+    _requestId: 0,
+
+    /** PrivateVariable: Strophe.connectionPlugins
+     *  _Private_ variable Used to store plugin names that need
+     *  initialization on Strophe.Connection construction.
+     */
+    _connectionPlugins: {},
+
+    /** Function: addConnectionPlugin
+     *  Extends the Strophe.Connection object with the given plugin.
+     *
+     *  Parameters:
+     *    (String) name - The name of the extension.
+     *    (Object) ptype - The plugin's prototype.
+     */
+    addConnectionPlugin: function (name, ptype)
+    {
+        Strophe._connectionPlugins[name] = ptype;
+    }
+};
+
+/** Class: Strophe.Builder
+ *  XML DOM builder.
+ *
+ *  This object provides an interface similar to JQuery but for building
+ *  DOM element easily and rapidly.  All the functions except for toString()
+ *  and tree() return the object, so calls can be chained.  Here's an
+ *  example using the $iq() builder helper.
+ *  > $iq({to: 'you', from: 'me', type: 'get', id: '1'})
+ *  >     .c('query', {xmlns: 'strophe:example'})
+ *  >     .c('example')
+ *  >     .toString()
+ *  The above generates this XML fragment
+ *  > <iq to='you' from='me' type='get' id='1'>
+ *  >   <query xmlns='strophe:example'>
+ *  >     <example/>
+ *  >   </query>
+ *  > </iq>
+ *  The corresponding DOM manipulations to get a similar fragment would be
+ *  a lot more tedious and probably involve several helper variables.
+ *
+ *  Since adding children makes new operations operate on the child, up()
+ *  is provided to traverse up the tree.  To add two children, do
+ *  > builder.c('child1', ...).up().c('child2', ...)
+ *  The next operation on the Builder will be relative to the second child.
+ */
+
+/** Constructor: Strophe.Builder
+ *  Create a Strophe.Builder object.
+ *
+ *  The attributes should be passed in object notation.  For example
+ *  > var b = new Builder('message', {to: 'you', from: 'me'});
+ *  or
+ *  > var b = new Builder('messsage', {'xml:lang': 'en'});
+ *
+ *  Parameters:
+ *    (String) name - The name of the root element.
+ *    (Object) attrs - The attributes for the root element in object notation.
+ *
+ *  Returns:
+ *    A new Strophe.Builder.
+ */
+Strophe.Builder = function (name, attrs)
+{
+    // Set correct namespace for jabber:client elements
+    if (name == "presence" || name == "message" || name == "iq") {
+        if (attrs && !attrs.xmlns) {
+            attrs.xmlns = Strophe.NS.CLIENT;
+        } else if (!attrs) {
+            attrs = {xmlns: Strophe.NS.CLIENT};
+        }
+    }
+
+    // Holds the tree being built.
+    this.nodeTree = Strophe.xmlElement(name, attrs);
+
+    // Points to the current operation node.
+    this.node = this.nodeTree;
+};
+
+Strophe.Builder.prototype = {
+    /** Function: tree
+     *  Return the DOM tree.
+     *
+     *  This function returns the current DOM tree as an element object.  This
+     *  is suitable for passing to functions like Strophe.Connection.send().
+     *
+     *  Returns:
+     *    The DOM tree as a element object.
+     */
+    tree: function ()
+    {
+        return this.nodeTree;
+    },
+
+    /** Function: toString
+     *  Serialize the DOM tree to a String.
+     *
+     *  This function returns a string serialization of the current DOM
+     *  tree.  It is often used internally to pass data to a
+     *  Strophe.Request object.
+     *
+     *  Returns:
+     *    The serialized DOM tree in a String.
+     */
+    toString: function ()
+    {
+        return Strophe.serialize(this.nodeTree);
+    },
+
+    /** Function: up
+     *  Make the current parent element the new current element.
+     *
+     *  This function is often used after c() to traverse back up the tree.
+     *  For example, to add two children to the same element
+     *  > builder.c('child1', {}).up().c('child2', {});
+     *
+     *  Returns:
+     *    The Stophe.Builder object.
+     */
+    up: function ()
+    {
+        this.node = this.node.parentNode;
+        return this;
+    },
+
+    /** Function: attrs
+     *  Add or modify attributes of the current element.
+     *
+     *  The attributes should be passed in object notation.  This function
+     *  does not move the current element pointer.
+     *
+     *  Parameters:
+     *    (Object) moreattrs - The attributes to add/modify in object notation.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    attrs: function (moreattrs)
+    {
+        for (var k in moreattrs) {
+            if (moreattrs.hasOwnProperty(k)) {
+                this.node.setAttribute(k, moreattrs[k]);
+            }
+        }
+        return this;
+    },
+
+    /** Function: c
+     *  Add a child to the current element and make it the new current
+     *  element.
+     *
+     *  This function moves the current element pointer to the child,
+     *  unless text is provided.  If you need to add another child, it
+     *  is necessary to use up() to go back to the parent in the tree.
+     *
+     *  Parameters:
+     *    (String) name - The name of the child.
+     *    (Object) attrs - The attributes of the child in object notation.
+     *    (String) text - The text to add to the child.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    c: function (name, attrs, text)
+    {
+        var child = Strophe.xmlElement(name, attrs, text);
+        this.node.appendChild(child);
+        if (!text) {
+            this.node = child;
+        }
+        return this;
+    },
+
+    /** Function: cnode
+     *  Add a child to the current element and make it the new current
+     *  element.
+     *
+     *  This function is the same as c() except that instead of using a
+     *  name and an attributes object to create the child it uses an
+     *  existing DOM element object.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - A DOM element.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    cnode: function (elem)
+    {
+        var impNode;
+        var xmlGen = Strophe.xmlGenerator();
+        try {
+            impNode = (xmlGen.importNode !== undefined);
+        }
+        catch (e) {
+            impNode = false;
+        }
+        var newElem = impNode ?
+                      xmlGen.importNode(elem, true) :
+                      Strophe.copyElement(elem);
+        this.node.appendChild(newElem);
+        this.node = newElem;
+        return this;
+    },
+
+    /** Function: t
+     *  Add a child text element.
+     *
+     *  This *does not* make the child the new current element since there
+     *  are no children of text elements.
+     *
+     *  Parameters:
+     *    (String) text - The text data to append to the current element.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    t: function (text)
+    {
+        var child = Strophe.xmlTextNode(text);
+        this.node.appendChild(child);
+        return this;
+    },
+
+    /** Function: h
+     *  Replace current element contents with the HTML passed in.
+     *
+     *  This *does not* make the child the new current element
+     *
+     *  Parameters:
+     *    (String) html - The html to insert as contents of current element.
+     *
+     *  Returns:
+     *    The Strophe.Builder object.
+     */
+    h: function (html)
+    {
+        var fragment = document.createElement('body');
+
+        // force the browser to try and fix any invalid HTML tags
+        fragment.innerHTML = html;
+
+        // copy cleaned html into an xml dom
+        var xhtml = Strophe.createHtml(fragment);
+
+        while(xhtml.childNodes.length > 0) {
+            this.node.appendChild(xhtml.childNodes[0]);
+        }
+        return this;
+    }
+};
+
+/** PrivateClass: Strophe.Handler
+ *  _Private_ helper class for managing stanza handlers.
+ *
+ *  A Strophe.Handler encapsulates a user provided callback function to be
+ *  executed when matching stanzas are received by the connection.
+ *  Handlers can be either one-off or persistant depending on their
+ *  return value. Returning true will cause a Handler to remain active, and
+ *  returning false will remove the Handler.
+ *
+ *  Users will not use Strophe.Handler objects directly, but instead they
+ *  will use Strophe.Connection.addHandler() and
+ *  Strophe.Connection.deleteHandler().
+ */
+
+/** PrivateConstructor: Strophe.Handler
+ *  Create and initialize a new Strophe.Handler.
+ *
+ *  Parameters:
+ *    (Function) handler - A function to be executed when the handler is run.
+ *    (String) ns - The namespace to match.
+ *    (String) name - The element name to match.
+ *    (String) type - The element type to match.
+ *    (String) id - The element id attribute to match.
+ *    (String) from - The element from attribute to match.
+ *    (Object) options - Handler options
+ *
+ *  Returns:
+ *    A new Strophe.Handler object.
+ */
+Strophe.Handler = function (handler, ns, name, type, id, from, options)
+{
+    this.handler = handler;
+    this.ns = ns;
+    this.name = name;
+    this.type = type;
+    this.id = id;
+    this.options = options || {matchBare: false};
+
+    // default matchBare to false if undefined
+    if (!this.options.matchBare) {
+        this.options.matchBare = false;
+    }
+
+    if (this.options.matchBare) {
+        this.from = from ? Strophe.getBareJidFromJid(from) : null;
+    } else {
+        this.from = from;
+    }
+
+    // whether the handler is a user handler or a system handler
+    this.user = true;
+};
+
+Strophe.Handler.prototype = {
+    /** PrivateFunction: isMatch
+     *  Tests if a stanza matches the Strophe.Handler.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The XML element to test.
+     *
+     *  Returns:
+     *    true if the stanza matches and false otherwise.
+     */
+    isMatch: function (elem)
+    {
+        var nsMatch;
+        var from = null;
+
+        if (this.options.matchBare) {
+            from = Strophe.getBareJidFromJid(elem.getAttribute('from'));
+        } else {
+            from = elem.getAttribute('from');
+        }
+
+        nsMatch = false;
+        if (!this.ns) {
+            nsMatch = true;
+        } else {
+            var that = this;
+            Strophe.forEachChild(elem, null, function (elem) {
+                if (elem.getAttribute("xmlns") == that.ns) {
+                    nsMatch = true;
+                }
+            });
+
+            nsMatch = nsMatch || elem.getAttribute("xmlns") == this.ns;
+        }
+
+        if (nsMatch &&
+            (!this.name || Strophe.isTagEqual(elem, this.name)) &&
+            (!this.type || elem.getAttribute("type") == this.type) &&
+            (!this.id || elem.getAttribute("id") == this.id) &&
+            (!this.from || from == this.from)) {
+                return true;
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: run
+     *  Run the callback on a matching stanza.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The DOM element that triggered the
+     *      Strophe.Handler.
+     *
+     *  Returns:
+     *    A boolean indicating if the handler should remain active.
+     */
+    run: function (elem)
+    {
+        var result = null;
+        try {
+            result = this.handler(elem);
+        } catch (e) {
+            if (e.sourceURL) {
+                Strophe.fatal("error: " + this.handler +
+                              " " + e.sourceURL + ":" +
+                              e.line + " - " + e.name + ": " + e.message);
+            } else if (e.fileName) {
+                if (typeof(console) != "undefined") {
+                    console.trace();
+                    console.error(this.handler, " - error - ", e, e.message);
+                }
+                Strophe.fatal("error: " + this.handler + " " +
+                              e.fileName + ":" + e.lineNumber + " - " +
+                              e.name + ": " + e.message);
+            } else {
+                Strophe.fatal("error: " + e.message + "\n" + e.stack);
+            }
+
+            throw e;
+        }
+
+        return result;
+    },
+
+    /** PrivateFunction: toString
+     *  Get a String representation of the Strophe.Handler object.
+     *
+     *  Returns:
+     *    A String.
+     */
+    toString: function ()
+    {
+        return "{Handler: " + this.handler + "(" + this.name + "," +
+            this.id + "," + this.ns + ")}";
+    }
+};
+
+/** PrivateClass: Strophe.TimedHandler
+ *  _Private_ helper class for managing timed handlers.
+ *
+ *  A Strophe.TimedHandler encapsulates a user provided callback that
+ *  should be called after a certain period of time or at regular
+ *  intervals.  The return value of the callback determines whether the
+ *  Strophe.TimedHandler will continue to fire.
+ *
+ *  Users will not use Strophe.TimedHandler objects directly, but instead
+ *  they will use Strophe.Connection.addTimedHandler() and
+ *  Strophe.Connection.deleteTimedHandler().
+ */
+
+/** PrivateConstructor: Strophe.TimedHandler
+ *  Create and initialize a new Strophe.TimedHandler object.
+ *
+ *  Parameters:
+ *    (Integer) period - The number of milliseconds to wait before the
+ *      handler is called.
+ *    (Function) handler - The callback to run when the handler fires.  This
+ *      function should take no arguments.
+ *
+ *  Returns:
+ *    A new Strophe.TimedHandler object.
+ */
+Strophe.TimedHandler = function (period, handler)
+{
+    this.period = period;
+    this.handler = handler;
+
+    this.lastCalled = new Date().getTime();
+    this.user = true;
+};
+
+Strophe.TimedHandler.prototype = {
+    /** PrivateFunction: run
+     *  Run the callback for the Strophe.TimedHandler.
+     *
+     *  Returns:
+     *    true if the Strophe.TimedHandler should be called again, and false
+     *      otherwise.
+     */
+    run: function ()
+    {
+        this.lastCalled = new Date().getTime();
+        return this.handler();
+    },
+
+    /** PrivateFunction: reset
+     *  Reset the last called time for the Strophe.TimedHandler.
+     */
+    reset: function ()
+    {
+        this.lastCalled = new Date().getTime();
+    },
+
+    /** PrivateFunction: toString
+     *  Get a string representation of the Strophe.TimedHandler object.
+     *
+     *  Returns:
+     *    The string representation.
+     */
+    toString: function ()
+    {
+        return "{TimedHandler: " + this.handler + "(" + this.period +")}";
+    }
+};
+
+/** Class: Strophe.Connection
+ *  XMPP Connection manager.
+ *
+ *  This class is the main part of Strophe.  It manages a BOSH connection
+ *  to an XMPP server and dispatches events to the user callbacks as
+ *  data arrives.  It supports SASL PLAIN, SASL DIGEST-MD5, SASL SCRAM-SHA1
+ *  and legacy authentication.
+ *
+ *  After creating a Strophe.Connection object, the user will typically
+ *  call connect() with a user supplied callback to handle connection level
+ *  events like authentication failure, disconnection, or connection
+ *  complete.
+ *
+ *  The user will also have several event handlers defined by using
+ *  addHandler() and addTimedHandler().  These will allow the user code to
+ *  respond to interesting stanzas or do something periodically with the
+ *  connection.  These handlers will be active once authentication is
+ *  finished.
+ *
+ *  To send data to the connection, use send().
+ */
+
+/** Constructor: Strophe.Connection
+ *  Create and initialize a Strophe.Connection object.
+ *
+ *  The transport-protocol for this connection will be chosen automatically
+ *  based on the given service parameter. URLs starting with "ws://" or
+ *  "wss://" will use WebSockets, URLs starting with "http://", "https://"
+ *  or without a protocol will use BOSH.
+ *
+ *  To make Strophe connect to the current host you can leave out the protocol
+ *  and host part and just pass the path, e.g.
+ *
+ *  > var conn = new Strophe.Connection("/http-bind/");
+ *
+ *  WebSocket options:
+ *
+ *  If you want to connect to the current host with a WebSocket connection you
+ *  can tell Strophe to use WebSockets through a "protocol" attribute in the
+ *  optional options parameter. Valid values are "ws" for WebSocket and "wss"
+ *  for Secure WebSocket.
+ *  So to connect to "wss://CURRENT_HOSTNAME/xmpp-websocket" you would call
+ *
+ *  > var conn = new Strophe.Connection("/xmpp-websocket/", {protocol: "wss"});
+ *
+ *  Note that relative URLs _NOT_ starting with a "/" will also include the path
+ *  of the current site.
+ *
+ *  Also because downgrading security is not permitted by browsers, when using
+ *  relative URLs both BOSH and WebSocket connections will use their secure
+ *  variants if the current connection to the site is also secure (https).
+ *
+ *  BOSH options:
+ *
+ *  by adding "sync" to the options, you can control if requests will
+ *  be made synchronously or not. The default behaviour is asynchronous.
+ *  If you want to make requests synchronous, make "sync" evaluate to true:
+ *  > var conn = new Strophe.Connection("/http-bind/", {sync: true});
+ *  You can also toggle this on an already established connection:
+ *  > conn.options.sync = true;
+ *
+ *
+ *  Parameters:
+ *    (String) service - The BOSH or WebSocket service URL.
+ *    (Object) options - A hash of configuration options
+ *
+ *  Returns:
+ *    A new Strophe.Connection object.
+ */
+Strophe.Connection = function (service, options)
+{
+    // The service URL
+    this.service = service;
+
+    // Configuration options
+    this.options = options || {};
+    var proto = this.options.protocol || "";
+
+    // Select protocal based on service or options
+    if (service.indexOf("ws:") === 0 || service.indexOf("wss:") === 0 ||
+            proto.indexOf("ws") === 0) {
+        this._proto = new Strophe.Websocket(this);
+    } else {
+        this._proto = new Strophe.Bosh(this);
+    }
+    /* The connected JID. */
+    this.jid = "";
+    /* the JIDs domain */
+    this.domain = null;
+    /* stream:features */
+    this.features = null;
+
+    // SASL
+    this._sasl_data = {};
+    this.do_session = false;
+    this.do_bind = false;
+
+    // handler lists
+    this.timedHandlers = [];
+    this.handlers = [];
+    this.removeTimeds = [];
+    this.removeHandlers = [];
+    this.addTimeds = [];
+    this.addHandlers = [];
+
+    this._authentication = {};
+    this._idleTimeout = null;
+    this._disconnectTimeout = null;
+
+    this.do_authentication = true;
+    this.authenticated = false;
+    this.disconnecting = false;
+    this.connected = false;
+
+    this.errors = 0;
+
+    this.paused = false;
+
+    this._data = [];
+    this._uniqueId = 0;
+
+    this._sasl_success_handler = null;
+    this._sasl_failure_handler = null;
+    this._sasl_challenge_handler = null;
+
+    // Max retries before disconnecting
+    this.maxRetries = 5;
+
+    // setup onIdle callback every 1/10th of a second
+    this._idleTimeout = setTimeout(this._onIdle.bind(this), 100);
+
+    // initialize plugins
+    for (var k in Strophe._connectionPlugins) {
+        if (Strophe._connectionPlugins.hasOwnProperty(k)) {
+            var ptype = Strophe._connectionPlugins[k];
+            // jslint complaints about the below line, but this is fine
+            var F = function () {}; // jshint ignore:line
+            F.prototype = ptype;
+            this[k] = new F();
+            this[k].init(this);
+        }
+    }
+};
+
+Strophe.Connection.prototype = {
+    /** Function: reset
+     *  Reset the connection.
+     *
+     *  This function should be called after a connection is disconnected
+     *  before that connection is reused.
+     */
+    reset: function ()
+    {
+        this._proto._reset();
+
+        // SASL
+        this.do_session = false;
+        this.do_bind = false;
+
+        // handler lists
+        this.timedHandlers = [];
+        this.handlers = [];
+        this.removeTimeds = [];
+        this.removeHandlers = [];
+        this.addTimeds = [];
+        this.addHandlers = [];
+        this._authentication = {};
+
+        this.authenticated = false;
+        this.disconnecting = false;
+        this.connected = false;
+
+        this.errors = 0;
+
+        this._requests = [];
+        this._uniqueId = 0;
+    },
+
+    /** Function: pause
+     *  Pause the request manager.
+     *
+     *  This will prevent Strophe from sending any more requests to the
+     *  server.  This is very useful for temporarily pausing
+     *  BOSH-Connections while a lot of send() calls are happening quickly.
+     *  This causes Strophe to send the data in a single request, saving
+     *  many request trips.
+     */
+    pause: function ()
+    {
+        this.paused = true;
+    },
+
+    /** Function: resume
+     *  Resume the request manager.
+     *
+     *  This resumes after pause() has been called.
+     */
+    resume: function ()
+    {
+        this.paused = false;
+    },
+
+    /** Function: getUniqueId
+     *  Generate a unique ID for use in <iq/> elements.
+     *
+     *  All <iq/> stanzas are required to have unique id attributes.  This
+     *  function makes creating these easy.  Each connection instance has
+     *  a counter which starts from zero, and the value of this counter
+     *  plus a colon followed by the suffix becomes the unique id. If no
+     *  suffix is supplied, the counter is used as the unique id.
+     *
+     *  Suffixes are used to make debugging easier when reading the stream
+     *  data, and their use is recommended.  The counter resets to 0 for
+     *  every new connection for the same reason.  For connections to the
+     *  same server that authenticate the same way, all the ids should be
+     *  the same, which makes it easy to see changes.  This is useful for
+     *  automated testing as well.
+     *
+     *  Parameters:
+     *    (String) suffix - A optional suffix to append to the id.
+     *
+     *  Returns:
+     *    A unique string to be used for the id attribute.
+     */
+    getUniqueId: function (suffix)
+    {
+        if (typeof(suffix) == "string" || typeof(suffix) == "number") {
+            return ++this._uniqueId + ":" + suffix;
+        } else {
+            return ++this._uniqueId + "";
+        }
+    },
+
+    /** Function: connect
+     *  Starts the connection process.
+     *
+     *  As the connection process proceeds, the user supplied callback will
+     *  be triggered multiple times with status updates.  The callback
+     *  should take two arguments - the status code and the error condition.
+     *
+     *  The status code will be one of the values in the Strophe.Status
+     *  constants.  The error condition will be one of the conditions
+     *  defined in RFC 3920 or the condition 'strophe-parsererror'.
+     *
+     *  The Parameters _wait_, _hold_ and _route_ are optional and only relevant
+     *  for BOSH connections. Please see XEP 124 for a more detailed explanation
+     *  of the optional parameters.
+     *
+     *  Parameters:
+     *    (String) jid - The user's JID.  This may be a bare JID,
+     *      or a full JID.  If a node is not supplied, SASL ANONYMOUS
+     *      authentication will be attempted.
+     *    (String) pass - The user's password.
+     *    (Function) callback - The connect callback function.
+     *    (Integer) wait - The optional HTTPBIND wait value.  This is the
+     *      time the server will wait before returning an empty result for
+     *      a request.  The default setting of 60 seconds is recommended.
+     *    (Integer) hold - The optional HTTPBIND hold value.  This is the
+     *      number of connections the server will hold at one time.  This
+     *      should almost always be set to 1 (the default).
+     *    (String) route - The optional route value.
+     */
+    connect: function (jid, pass, callback, wait, hold, route)
+    {
+        this.jid = jid;
+        /** Variable: authzid
+         *  Authorization identity.
+         */
+        this.authzid = Strophe.getBareJidFromJid(this.jid);
+        /** Variable: authcid
+         *  Authentication identity (User name).
+         */
+        this.authcid = Strophe.getNodeFromJid(this.jid);
+        /** Variable: pass
+         *  Authentication identity (User password).
+         */
+        this.pass = pass;
+        /** Variable: servtype
+         *  Digest MD5 compatibility.
+         */
+        this.servtype = "xmpp";
+        this.connect_callback = callback;
+        this.disconnecting = false;
+        this.connected = false;
+        this.authenticated = false;
+        this.errors = 0;
+
+        // parse jid for domain
+        this.domain = Strophe.getDomainFromJid(this.jid);
+
+        this._changeConnectStatus(Strophe.Status.CONNECTING, null);
+
+        this._proto._connect(wait, hold, route);
+    },
+
+    /** Function: attach
+     *  Attach to an already created and authenticated BOSH session.
+     *
+     *  This function is provided to allow Strophe to attach to BOSH
+     *  sessions which have been created externally, perhaps by a Web
+     *  application.  This is often used to support auto-login type features
+     *  without putting user credentials into the page.
+     *
+     *  Parameters:
+     *    (String) jid - The full JID that is bound by the session.
+     *    (String) sid - The SID of the BOSH session.
+     *    (String) rid - The current RID of the BOSH session.  This RID
+     *      will be used by the next request.
+     *    (Function) callback The connect callback function.
+     *    (Integer) wait - The optional HTTPBIND wait value.  This is the
+     *      time the server will wait before returning an empty result for
+     *      a request.  The default setting of 60 seconds is recommended.
+     *      Other settings will require tweaks to the Strophe.TIMEOUT value.
+     *    (Integer) hold - The optional HTTPBIND hold value.  This is the
+     *      number of connections the server will hold at one time.  This
+     *      should almost always be set to 1 (the default).
+     *    (Integer) wind - The optional HTTBIND window value.  This is the
+     *      allowed range of request ids that are valid.  The default is 5.
+     */
+    attach: function (jid, sid, rid, callback, wait, hold, wind)
+    {
+        this._proto._attach(jid, sid, rid, callback, wait, hold, wind);
+    },
+
+    /** Function: xmlInput
+     *  User overrideable function that receives XML data coming into the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.xmlInput = function (elem) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Due to limitations of current Browsers' XML-Parsers the opening and closing
+     *  <stream> tag for WebSocket-Connoctions will be passed as selfclosing here.
+     *
+     *  BOSH-Connections will have all stanzas wrapped in a <body> tag. See
+     *  <Strophe.Bosh.strip> if you want to strip this tag.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The XML data received by the connection.
+     */
+    /* jshint unused:false */
+    xmlInput: function (elem)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: xmlOutput
+     *  User overrideable function that receives XML data sent to the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.xmlOutput = function (elem) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Due to limitations of current Browsers' XML-Parsers the opening and closing
+     *  <stream> tag for WebSocket-Connoctions will be passed as selfclosing here.
+     *
+     *  BOSH-Connections will have all stanzas wrapped in a <body> tag. See
+     *  <Strophe.Bosh.strip> if you want to strip this tag.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The XMLdata sent by the connection.
+     */
+    /* jshint unused:false */
+    xmlOutput: function (elem)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: rawInput
+     *  User overrideable function that receives raw data coming into the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.rawInput = function (data) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Parameters:
+     *    (String) data - The data received by the connection.
+     */
+    /* jshint unused:false */
+    rawInput: function (data)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: rawOutput
+     *  User overrideable function that receives raw data sent to the
+     *  connection.
+     *
+     *  The default function does nothing.  User code can override this with
+     *  > Strophe.Connection.rawOutput = function (data) {
+     *  >   (user code)
+     *  > };
+     *
+     *  Parameters:
+     *    (String) data - The data sent by the connection.
+     */
+    /* jshint unused:false */
+    rawOutput: function (data)
+    {
+        return;
+    },
+    /* jshint unused:true */
+
+    /** Function: send
+     *  Send a stanza.
+     *
+     *  This function is called to push data onto the send queue to
+     *  go out over the wire.  Whenever a request is sent to the BOSH
+     *  server, all pending data is sent and the queue is flushed.
+     *
+     *  Parameters:
+     *    (XMLElement |
+     *     [XMLElement] |
+     *     Strophe.Builder) elem - The stanza to send.
+     */
+    send: function (elem)
+    {
+        if (elem === null) { return ; }
+        if (typeof(elem.sort) === "function") {
+            for (var i = 0; i < elem.length; i++) {
+                this._queueData(elem[i]);
+            }
+        } else if (typeof(elem.tree) === "function") {
+            this._queueData(elem.tree());
+        } else {
+            this._queueData(elem);
+        }
+
+        this._proto._send();
+    },
+
+    /** Function: flush
+     *  Immediately send any pending outgoing data.
+     *
+     *  Normally send() queues outgoing data until the next idle period
+     *  (100ms), which optimizes network use in the common cases when
+     *  several send()s are called in succession. flush() can be used to
+     *  immediately send all pending data.
+     */
+    flush: function ()
+    {
+        // cancel the pending idle period and run the idle function
+        // immediately
+        clearTimeout(this._idleTimeout);
+        this._onIdle();
+    },
+
+    /** Function: sendIQ
+     *  Helper function to send IQ stanzas.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The stanza to send.
+     *    (Function) callback - The callback function for a successful request.
+     *    (Function) errback - The callback function for a failed or timed
+     *      out request.  On timeout, the stanza will be null.
+     *    (Integer) timeout - The time specified in milliseconds for a
+     *      timeout to occur.
+     *
+     *  Returns:
+     *    The id used to send the IQ.
+    */
+    sendIQ: function(elem, callback, errback, timeout) {
+        var timeoutHandler = null;
+        var that = this;
+
+        if (typeof(elem.tree) === "function") {
+            elem = elem.tree();
+        }
+        var id = elem.getAttribute('id');
+
+        // inject id if not found
+        if (!id) {
+            id = this.getUniqueId("sendIQ");
+            elem.setAttribute("id", id);
+        }
+
+        var handler = this.addHandler(function (stanza) {
+            // remove timeout handler if there is one
+            if (timeoutHandler) {
+                that.deleteTimedHandler(timeoutHandler);
+            }
+
+            var iqtype = stanza.getAttribute('type');
+            if (iqtype == 'result') {
+                if (callback) {
+                    callback(stanza);
+                }
+            } else if (iqtype == 'error') {
+                if (errback) {
+                    errback(stanza);
+                }
+            } else {
+                throw {
+                    name: "StropheError",
+            message: "Got bad IQ type of " + iqtype
+                };
+            }
+        }, null, 'iq', null, id);
+
+        // if timeout specified, setup timeout handler.
+        if (timeout) {
+            timeoutHandler = this.addTimedHandler(timeout, function () {
+                // get rid of normal handler
+                that.deleteHandler(handler);
+
+                // call errback on timeout with null stanza
+                if (errback) {
+                    errback(null);
+                }
+                return false;
+            });
+        }
+
+        this.send(elem);
+
+        return id;
+    },
+
+    /** PrivateFunction: _queueData
+     *  Queue outgoing data for later sending.  Also ensures that the data
+     *  is a DOMElement.
+     */
+    _queueData: function (element) {
+        if (element === null ||
+            !element.tagName ||
+            !element.childNodes) {
+            throw {
+                name: "StropheError",
+                message: "Cannot queue non-DOMElement."
+            };
+        }
+
+        this._data.push(element);
+    },
+
+    /** PrivateFunction: _sendRestart
+     *  Send an xmpp:restart stanza.
+     */
+    _sendRestart: function ()
+    {
+        this._data.push("restart");
+
+        this._proto._sendRestart();
+
+        this._idleTimeout = setTimeout(this._onIdle.bind(this), 100);
+    },
+
+    /** Function: addTimedHandler
+     *  Add a timed handler to the connection.
+     *
+     *  This function adds a timed handler.  The provided handler will
+     *  be called every period milliseconds until it returns false,
+     *  the connection is terminated, or the handler is removed.  Handlers
+     *  that wish to continue being invoked should return true.
+     *
+     *  Because of method binding it is necessary to save the result of
+     *  this function if you wish to remove a handler with
+     *  deleteTimedHandler().
+     *
+     *  Note that user handlers are not active until authentication is
+     *  successful.
+     *
+     *  Parameters:
+     *    (Integer) period - The period of the handler.
+     *    (Function) handler - The callback function.
+     *
+     *  Returns:
+     *    A reference to the handler that can be used to remove it.
+     */
+    addTimedHandler: function (period, handler)
+    {
+        var thand = new Strophe.TimedHandler(period, handler);
+        this.addTimeds.push(thand);
+        return thand;
+    },
+
+    /** Function: deleteTimedHandler
+     *  Delete a timed handler for a connection.
+     *
+     *  This function removes a timed handler from the connection.  The
+     *  handRef parameter is *not* the function passed to addTimedHandler(),
+     *  but is the reference returned from addTimedHandler().
+     *
+     *  Parameters:
+     *    (Strophe.TimedHandler) handRef - The handler reference.
+     */
+    deleteTimedHandler: function (handRef)
+    {
+        // this must be done in the Idle loop so that we don't change
+        // the handlers during iteration
+        this.removeTimeds.push(handRef);
+    },
+
+    /** Function: addHandler
+     *  Add a stanza handler for the connection.
+     *
+     *  This function adds a stanza handler to the connection.  The
+     *  handler callback will be called for any stanza that matches
+     *  the parameters.  Note that if multiple parameters are supplied,
+     *  they must all match for the handler to be invoked.
+     *
+     *  The handler will receive the stanza that triggered it as its argument.
+     *  The handler should return true if it is to be invoked again;
+     *  returning false will remove the handler after it returns.
+     *
+     *  As a convenience, the ns parameters applies to the top level element
+     *  and also any of its immediate children.  This is primarily to make
+     *  matching /iq/query elements easy.
+     *
+     *  The options argument contains handler matching flags that affect how
+     *  matches are determined. Currently the only flag is matchBare (a
+     *  boolean). When matchBare is true, the from parameter and the from
+     *  attribute on the stanza will be matched as bare JIDs instead of
+     *  full JIDs. To use this, pass {matchBare: true} as the value of
+     *  options. The default value for matchBare is false.
+     *
+     *  The return value should be saved if you wish to remove the handler
+     *  with deleteHandler().
+     *
+     *  Parameters:
+     *    (Function) handler - The user callback.
+     *    (String) ns - The namespace to match.
+     *    (String) name - The stanza name to match.
+     *    (String) type - The stanza type attribute to match.
+     *    (String) id - The stanza id attribute to match.
+     *    (String) from - The stanza from attribute to match.
+     *    (String) options - The handler options
+     *
+     *  Returns:
+     *    A reference to the handler that can be used to remove it.
+     */
+    addHandler: function (handler, ns, name, type, id, from, options)
+    {
+        var hand = new Strophe.Handler(handler, ns, name, type, id, from, options);
+        this.addHandlers.push(hand);
+        return hand;
+    },
+
+    /** Function: deleteHandler
+     *  Delete a stanza handler for a connection.
+     *
+     *  This function removes a stanza handler from the connection.  The
+     *  handRef parameter is *not* the function passed to addHandler(),
+     *  but is the reference returned from addHandler().
+     *
+     *  Parameters:
+     *    (Strophe.Handler) handRef - The handler reference.
+     */
+    deleteHandler: function (handRef)
+    {
+        // this must be done in the Idle loop so that we don't change
+        // the handlers during iteration
+        this.removeHandlers.push(handRef);
+    },
+
+    /** Function: disconnect
+     *  Start the graceful disconnection process.
+     *
+     *  This function starts the disconnection process.  This process starts
+     *  by sending unavailable presence and sending BOSH body of type
+     *  terminate.  A timeout handler makes sure that disconnection happens
+     *  even if the BOSH server does not respond.
+     *
+     *  The user supplied connection callback will be notified of the
+     *  progress as this process happens.
+     *
+     *  Parameters:
+     *    (String) reason - The reason the disconnect is occuring.
+     */
+    disconnect: function (reason)
+    {
+        this._changeConnectStatus(Strophe.Status.DISCONNECTING, reason);
+
+        Strophe.info("Disconnect was called because: " + reason);
+        if (this.connected) {
+            var pres = false;
+            this.disconnecting = true;
+            if (this.authenticated) {
+                pres = $pres({
+                    xmlns: Strophe.NS.CLIENT,
+                    type: 'unavailable'
+                });
+            }
+            // setup timeout handler
+            this._disconnectTimeout = this._addSysTimedHandler(
+                3000, this._onDisconnectTimeout.bind(this));
+            this._proto._disconnect(pres);
+        }
+    },
+
+    /** PrivateFunction: _changeConnectStatus
+     *  _Private_ helper function that makes sure plugins and the user's
+     *  callback are notified of connection status changes.
+     *
+     *  Parameters:
+     *    (Integer) status - the new connection status, one of the values
+     *      in Strophe.Status
+     *    (String) condition - the error condition or null
+     */
+    _changeConnectStatus: function (status, condition)
+    {
+        // notify all plugins listening for status changes
+        for (var k in Strophe._connectionPlugins) {
+            if (Strophe._connectionPlugins.hasOwnProperty(k)) {
+                var plugin = this[k];
+                if (plugin.statusChanged) {
+                    try {
+                        plugin.statusChanged(status, condition);
+                    } catch (err) {
+                        Strophe.error("" + k + " plugin caused an exception " +
+                                      "changing status: " + err);
+                    }
+                }
+            }
+        }
+
+        // notify the user's callback
+        if (this.connect_callback) {
+            try {
+                this.connect_callback(status, condition);
+            } catch (e) {
+                Strophe.error("User connection callback caused an " +
+                              "exception: " + e);
+            }
+        }
+    },
+
+    /** PrivateFunction: _doDisconnect
+     *  _Private_ function to disconnect.
+     *
+     *  This is the last piece of the disconnection logic.  This resets the
+     *  connection and alerts the user's connection callback.
+     */
+    _doDisconnect: function ()
+    {
+        // Cancel Disconnect Timeout
+        if (this._disconnectTimeout !== null) {
+            this.deleteTimedHandler(this._disconnectTimeout);
+            this._disconnectTimeout = null;
+        }
+
+        Strophe.info("_doDisconnect was called");
+        this._proto._doDisconnect();
+
+        this.authenticated = false;
+        this.disconnecting = false;
+
+        // delete handlers
+        this.handlers = [];
+        this.timedHandlers = [];
+        this.removeTimeds = [];
+        this.removeHandlers = [];
+        this.addTimeds = [];
+        this.addHandlers = [];
+
+        // tell the parent we disconnected
+        this._changeConnectStatus(Strophe.Status.DISCONNECTED, null);
+        this.connected = false;
+    },
+
+    /** PrivateFunction: _dataRecv
+     *  _Private_ handler to processes incoming data from the the connection.
+     *
+     *  Except for _connect_cb handling the initial connection request,
+     *  this function handles the incoming data for all requests.  This
+     *  function also fires stanza handlers that match each incoming
+     *  stanza.
+     *
+     *  Parameters:
+     *    (Strophe.Request) req - The request that has data ready.
+     *    (string) req - The stanza a raw string (optiona).
+     */
+    _dataRecv: function (req, raw)
+    {
+        Strophe.info("_dataRecv called");
+        var elem = this._proto._reqToData(req);
+        if (elem === null) { return; }
+
+        if (this.xmlInput !== Strophe.Connection.prototype.xmlInput) {
+            if (elem.nodeName === this._proto.strip && elem.childNodes.length) {
+                this.xmlInput(elem.childNodes[0]);
+            } else {
+                this.xmlInput(elem);
+            }
+        }
+        if (this.rawInput !== Strophe.Connection.prototype.rawInput) {
+            if (raw) {
+                this.rawInput(raw);
+            } else {
+                this.rawInput(Strophe.serialize(elem));
+            }
+        }
+
+        // remove handlers scheduled for deletion
+        var i, hand;
+        while (this.removeHandlers.length > 0) {
+            hand = this.removeHandlers.pop();
+            i = this.handlers.indexOf(hand);
+            if (i >= 0) {
+                this.handlers.splice(i, 1);
+            }
+        }
+
+        // add handlers scheduled for addition
+        while (this.addHandlers.length > 0) {
+            this.handlers.push(this.addHandlers.pop());
+        }
+
+        // handle graceful disconnect
+        if (this.disconnecting && this._proto._emptyQueue()) {
+            this._doDisconnect();
+            return;
+        }
+
+        var typ = elem.getAttribute("type");
+        var cond, conflict;
+        if (typ !== null && typ == "terminate") {
+            // Don't process stanzas that come in after disconnect
+            if (this.disconnecting) {
+                return;
+            }
+
+            // an error occurred
+            cond = elem.getAttribute("condition");
+            conflict = elem.getElementsByTagName("conflict");
+            if (cond !== null) {
+                if (cond == "remote-stream-error" && conflict.length > 0) {
+                    cond = "conflict";
+                }
+                this._changeConnectStatus(Strophe.Status.CONNFAIL, cond);
+            } else {
+                this._changeConnectStatus(Strophe.Status.CONNFAIL, "unknown");
+            }
+            this.disconnect('unknown stream-error');
+            return;
+        }
+
+        // send each incoming stanza through the handler chain
+        var that = this;
+        Strophe.forEachChild(elem, null, function (child) {
+            var i, newList;
+            // process handlers
+            newList = that.handlers;
+            that.handlers = [];
+            for (i = 0; i < newList.length; i++) {
+                var hand = newList[i];
+                // encapsulate 'handler.run' not to lose the whole handler list if
+                // one of the handlers throws an exception
+                try {
+                    if (hand.isMatch(child) &&
+                        (that.authenticated || !hand.user)) {
+                        if (hand.run(child)) {
+                            that.handlers.push(hand);
+                        }
+                    } else {
+                        that.handlers.push(hand);
+                    }
+                } catch(e) {
+                    // if the handler throws an exception, we consider it as false
+                    Strophe.warn('Removing Strophe handlers due to uncaught exception: ' + e.message);
+                }
+            }
+        });
+    },
+
+
+    /** Attribute: mechanisms
+     *  SASL Mechanisms available for Conncection.
+     */
+    mechanisms: {},
+
+    /** PrivateFunction: _connect_cb
+     *  _Private_ handler for initial connection request.
+     *
+     *  This handler is used to process the initial connection request
+     *  response from the BOSH server. It is used to set up authentication
+     *  handlers and start the authentication process.
+     *
+     *  SASL authentication will be attempted if available, otherwise
+     *  the code will fall back to legacy authentication.
+     *
+     *  Parameters:
+     *    (Strophe.Request) req - The current request.
+     *    (Function) _callback - low level (xmpp) connect callback function.
+     *      Useful for plugins with their own xmpp connect callback (when their)
+     *      want to do something special).
+     */
+    _connect_cb: function (req, _callback, raw)
+    {
+        Strophe.info("_connect_cb was called");
+
+        this.connected = true;
+
+        var bodyWrap = this._proto._reqToData(req);
+        if (!bodyWrap) { return; }
+
+        if (this.xmlInput !== Strophe.Connection.prototype.xmlInput) {
+            if (bodyWrap.nodeName === this._proto.strip && bodyWrap.childNodes.length) {
+                this.xmlInput(bodyWrap.childNodes[0]);
+            } else {
+                this.xmlInput(bodyWrap);
+            }
+        }
+        if (this.rawInput !== Strophe.Connection.prototype.rawInput) {
+            if (raw) {
+                this.rawInput(raw);
+            } else {
+                this.rawInput(Strophe.serialize(bodyWrap));
+            }
+        }
+
+        var conncheck = this._proto._connect_cb(bodyWrap);
+        if (conncheck === Strophe.Status.CONNFAIL) {
+            return;
+        }
+
+        this._authentication.sasl_scram_sha1 = false;
+        this._authentication.sasl_plain = false;
+        this._authentication.sasl_digest_md5 = false;
+        this._authentication.sasl_anonymous = false;
+
+        this._authentication.legacy_auth = false;
+
+        // Check for the stream:features tag
+        var hasFeatures = bodyWrap.getElementsByTagName("stream:features").length > 0;
+        if (!hasFeatures) {
+            hasFeatures = bodyWrap.getElementsByTagName("features").length > 0;
+        }
+        var mechanisms = bodyWrap.getElementsByTagName("mechanism");
+        var matched = [];
+        var i, mech, found_authentication = false;
+        if (!hasFeatures) {
+            this._proto._no_auth_received(_callback);
+            return;
+        }
+        if (mechanisms.length > 0) {
+            for (i = 0; i < mechanisms.length; i++) {
+                mech = Strophe.getText(mechanisms[i]);
+                if (this.mechanisms[mech]) matched.push(this.mechanisms[mech]);
+            }
+        }
+        this._authentication.legacy_auth =
+            bodyWrap.getElementsByTagName("auth").length > 0;
+        found_authentication = this._authentication.legacy_auth ||
+            matched.length > 0;
+        if (!found_authentication) {
+            this._proto._no_auth_received(_callback);
+            return;
+        }
+        if (this.do_authentication !== false)
+            this.authenticate(matched);
+    },
+
+    /** Function: authenticate
+     * Set up authentication
+     *
+     *  Contiunues the initial connection request by setting up authentication
+     *  handlers and start the authentication process.
+     *
+     *  SASL authentication will be attempted if available, otherwise
+     *  the code will fall back to legacy authentication.
+     *
+     */
+    authenticate: function (matched)
+    {
+      var i;
+      // Sorting matched mechanisms according to priority.
+      for (i = 0; i < matched.length - 1; ++i) {
+        var higher = i;
+        for (var j = i + 1; j < matched.length; ++j) {
+          if (matched[j].prototype.priority > matched[higher].prototype.priority) {
+            higher = j;
+          }
+        }
+        if (higher != i) {
+          var swap = matched[i];
+          matched[i] = matched[higher];
+          matched[higher] = swap;
+        }
+      }
+
+      // run each mechanism
+      var mechanism_found = false;
+      for (i = 0; i < matched.length; ++i) {
+        if (!matched[i].test(this)) continue;
+
+        this._sasl_success_handler = this._addSysHandler(
+          this._sasl_success_cb.bind(this), null,
+          "success", null, null);
+        this._sasl_failure_handler = this._addSysHandler(
+          this._sasl_failure_cb.bind(this), null,
+          "failure", null, null);
+        this._sasl_challenge_handler = this._addSysHandler(
+          this._sasl_challenge_cb.bind(this), null,
+          "challenge", null, null);
+
+        this._sasl_mechanism = new matched[i]();
+        this._sasl_mechanism.onStart(this);
+
+        var request_auth_exchange = $build("auth", {
+          xmlns: Strophe.NS.SASL,
+          mechanism: this._sasl_mechanism.name
+        });
+
+        if (this._sasl_mechanism.isClientFirst) {
+          var response = this._sasl_mechanism.onChallenge(this, null);
+          request_auth_exchange.t(Base64.encode(response));
+        }
+
+        this.send(request_auth_exchange.tree());
+
+        mechanism_found = true;
+        break;
+      }
+
+      if (!mechanism_found) {
+        // if none of the mechanism worked
+        if (Strophe.getNodeFromJid(this.jid) === null) {
+            // we don't have a node, which is required for non-anonymous
+            // client connections
+            this._changeConnectStatus(Strophe.Status.CONNFAIL,
+                                      'x-strophe-bad-non-anon-jid');
+            this.disconnect('x-strophe-bad-non-anon-jid');
+        } else {
+          // fall back to legacy authentication
+          this._changeConnectStatus(Strophe.Status.AUTHENTICATING, null);
+          this._addSysHandler(this._auth1_cb.bind(this), null, null,
+                              null, "_auth_1");
+
+          this.send($iq({
+            type: "get",
+            to: this.domain,
+            id: "_auth_1"
+          }).c("query", {
+            xmlns: Strophe.NS.AUTH
+          }).c("username", {}).t(Strophe.getNodeFromJid(this.jid)).tree());
+        }
+      }
+
+    },
+
+    _sasl_challenge_cb: function(elem) {
+      var challenge = Base64.decode(Strophe.getText(elem));
+      var response = this._sasl_mechanism.onChallenge(this, challenge);
+
+      var stanza = $build('response', {
+          xmlns: Strophe.NS.SASL
+      });
+      if (response !== "") {
+        stanza.t(Base64.encode(response));
+      }
+      this.send(stanza.tree());
+
+      return true;
+    },
+
+    /** PrivateFunction: _auth1_cb
+     *  _Private_ handler for legacy authentication.
+     *
+     *  This handler is called in response to the initial <iq type='get'/>
+     *  for legacy authentication.  It builds an authentication <iq/> and
+     *  sends it, creating a handler (calling back to _auth2_cb()) to
+     *  handle the result
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The stanza that triggered the callback.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    /* jshint unused:false */
+    _auth1_cb: function (elem)
+    {
+        // build plaintext auth iq
+        var iq = $iq({type: "set", id: "_auth_2"})
+            .c('query', {xmlns: Strophe.NS.AUTH})
+            .c('username', {}).t(Strophe.getNodeFromJid(this.jid))
+            .up()
+            .c('password').t(this.pass);
+
+        if (!Strophe.getResourceFromJid(this.jid)) {
+            // since the user has not supplied a resource, we pick
+            // a default one here.  unlike other auth methods, the server
+            // cannot do this for us.
+            this.jid = Strophe.getBareJidFromJid(this.jid) + '/strophe';
+        }
+        iq.up().c('resource', {}).t(Strophe.getResourceFromJid(this.jid));
+
+        this._addSysHandler(this._auth2_cb.bind(this), null,
+                            null, null, "_auth_2");
+
+        this.send(iq.tree());
+
+        return false;
+    },
+    /* jshint unused:true */
+
+    /** PrivateFunction: _sasl_success_cb
+     *  _Private_ handler for succesful SASL authentication.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_success_cb: function (elem)
+    {
+        if (this._sasl_data["server-signature"]) {
+            var serverSignature;
+            var success = Base64.decode(Strophe.getText(elem));
+            var attribMatch = /([a-z]+)=([^,]+)(,|$)/;
+            var matches = success.match(attribMatch);
+            if (matches[1] == "v") {
+                serverSignature = matches[2];
+            }
+
+            if (serverSignature != this._sasl_data["server-signature"]) {
+              // remove old handlers
+              this.deleteHandler(this._sasl_failure_handler);
+              this._sasl_failure_handler = null;
+              if (this._sasl_challenge_handler) {
+                this.deleteHandler(this._sasl_challenge_handler);
+                this._sasl_challenge_handler = null;
+              }
+
+              this._sasl_data = {};
+              return this._sasl_failure_cb(null);
+            }
+        }
+
+        Strophe.info("SASL authentication succeeded.");
+
+        if(this._sasl_mechanism)
+          this._sasl_mechanism.onSuccess();
+
+        // remove old handlers
+        this.deleteHandler(this._sasl_failure_handler);
+        this._sasl_failure_handler = null;
+        if (this._sasl_challenge_handler) {
+            this.deleteHandler(this._sasl_challenge_handler);
+            this._sasl_challenge_handler = null;
+        }
+
+        this._addSysHandler(this._sasl_auth1_cb.bind(this), null,
+                            "stream:features", null, null);
+
+        // we must send an xmpp:restart now
+        this._sendRestart();
+
+        return false;
+    },
+
+    /** PrivateFunction: _sasl_auth1_cb
+     *  _Private_ handler to start stream binding.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_auth1_cb: function (elem)
+    {
+        // save stream:features for future usage
+        this.features = elem;
+
+        var i, child;
+
+        for (i = 0; i < elem.childNodes.length; i++) {
+            child = elem.childNodes[i];
+            if (child.nodeName == 'bind') {
+                this.do_bind = true;
+            }
+
+            if (child.nodeName == 'session') {
+                this.do_session = true;
+            }
+        }
+
+        if (!this.do_bind) {
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            return false;
+        } else {
+            this._addSysHandler(this._sasl_bind_cb.bind(this), null, null,
+                                null, "_bind_auth_2");
+
+            var resource = Strophe.getResourceFromJid(this.jid);
+            if (resource) {
+                this.send($iq({type: "set", id: "_bind_auth_2"})
+                          .c('bind', {xmlns: Strophe.NS.BIND})
+                          .c('resource', {}).t(resource).tree());
+            } else {
+                this.send($iq({type: "set", id: "_bind_auth_2"})
+                          .c('bind', {xmlns: Strophe.NS.BIND})
+                          .tree());
+            }
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: _sasl_bind_cb
+     *  _Private_ handler for binding result and session start.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_bind_cb: function (elem)
+    {
+        if (elem.getAttribute("type") == "error") {
+            Strophe.info("SASL binding failed.");
+            var conflict = elem.getElementsByTagName("conflict"), condition;
+            if (conflict.length > 0) {
+                condition = 'conflict';
+            }
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, condition);
+            return false;
+        }
+
+        // TODO - need to grab errors
+        var bind = elem.getElementsByTagName("bind");
+        var jidNode;
+        if (bind.length > 0) {
+            // Grab jid
+            jidNode = bind[0].getElementsByTagName("jid");
+            if (jidNode.length > 0) {
+                this.jid = Strophe.getText(jidNode[0]);
+
+                if (this.do_session) {
+                    this._addSysHandler(this._sasl_session_cb.bind(this),
+                                        null, null, null, "_session_auth_2");
+
+                    this.send($iq({type: "set", id: "_session_auth_2"})
+                                  .c('session', {xmlns: Strophe.NS.SESSION})
+                                  .tree());
+                } else {
+                    this.authenticated = true;
+                    this._changeConnectStatus(Strophe.Status.CONNECTED, null);
+                }
+            }
+        } else {
+            Strophe.info("SASL binding failed.");
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            return false;
+        }
+    },
+
+    /** PrivateFunction: _sasl_session_cb
+     *  _Private_ handler to finish successful SASL connection.
+     *
+     *  This sets Connection.authenticated to true on success, which
+     *  starts the processing of user handlers.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _sasl_session_cb: function (elem)
+    {
+        if (elem.getAttribute("type") == "result") {
+            this.authenticated = true;
+            this._changeConnectStatus(Strophe.Status.CONNECTED, null);
+        } else if (elem.getAttribute("type") == "error") {
+            Strophe.info("Session creation failed.");
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            return false;
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: _sasl_failure_cb
+     *  _Private_ handler for SASL authentication failure.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The matching stanza.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    /* jshint unused:false */
+    _sasl_failure_cb: function (elem)
+    {
+        // delete unneeded handlers
+        if (this._sasl_success_handler) {
+            this.deleteHandler(this._sasl_success_handler);
+            this._sasl_success_handler = null;
+        }
+        if (this._sasl_challenge_handler) {
+            this.deleteHandler(this._sasl_challenge_handler);
+            this._sasl_challenge_handler = null;
+        }
+
+        if(this._sasl_mechanism)
+          this._sasl_mechanism.onFailure();
+        this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+        return false;
+    },
+    /* jshint unused:true */
+
+    /** PrivateFunction: _auth2_cb
+     *  _Private_ handler to finish legacy authentication.
+     *
+     *  This handler is called when the result from the jabber:iq:auth
+     *  <iq/> stanza is returned.
+     *
+     *  Parameters:
+     *    (XMLElement) elem - The stanza that triggered the callback.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _auth2_cb: function (elem)
+    {
+        if (elem.getAttribute("type") == "result") {
+            this.authenticated = true;
+            this._changeConnectStatus(Strophe.Status.CONNECTED, null);
+        } else if (elem.getAttribute("type") == "error") {
+            this._changeConnectStatus(Strophe.Status.AUTHFAIL, null);
+            this.disconnect('authentication failed');
+        }
+
+        return false;
+    },
+
+    /** PrivateFunction: _addSysTimedHandler
+     *  _Private_ function to add a system level timed handler.
+     *
+     *  This function is used to add a Strophe.TimedHandler for the
+     *  library code.  System timed handlers are allowed to run before
+     *  authentication is complete.
+     *
+     *  Parameters:
+     *    (Integer) period - The period of the handler.
+     *    (Function) handler - The callback function.
+     */
+    _addSysTimedHandler: function (period, handler)
+    {
+        var thand = new Strophe.TimedHandler(period, handler);
+        thand.user = false;
+        this.addTimeds.push(thand);
+        return thand;
+    },
+
+    /** PrivateFunction: _addSysHandler
+     *  _Private_ function to add a system level stanza handler.
+     *
+     *  This function is used to add a Strophe.Handler for the
+     *  library code.  System stanza handlers are allowed to run before
+     *  authentication is complete.
+     *
+     *  Parameters:
+     *    (Function) handler - The callback function.
+     *    (String) ns - The namespace to match.
+     *    (String) name - The stanza name to match.
+     *    (String) type - The stanza type attribute to match.
+     *    (String) id - The stanza id attribute to match.
+     */
+    _addSysHandler: function (handler, ns, name, type, id)
+    {
+        var hand = new Strophe.Handler(handler, ns, name, type, id);
+        hand.user = false;
+        this.addHandlers.push(hand);
+        return hand;
+    },
+
+    /** PrivateFunction: _onDisconnectTimeout
+     *  _Private_ timeout handler for handling non-graceful disconnection.
+     *
+     *  If the graceful disconnect process does not complete within the
+     *  time allotted, this handler finishes the disconnect anyway.
+     *
+     *  Returns:
+     *    false to remove the handler.
+     */
+    _onDisconnectTimeout: function ()
+    {
+        Strophe.info("_onDisconnectTimeout was called");
+
+        this._proto._onDisconnectTimeout();
+
+        // actually disconnect
+        this._doDisconnect();
+
+        return false;
+    },
+
+    /** PrivateFunction: _onIdle
+     *  _Private_ handler to process events during idle cycle.
+     *
+     *  This handler is called every 100ms to fire timed handlers that
+     *  are ready and keep poll requests going.
+     */
+    _onIdle: function ()
+    {
+        var i, thand, since, newList;
+
+        // add timed handlers scheduled for addition
+        // NOTE: we add before remove in the case a timed handler is
+        // added and then deleted before the next _onIdle() call.
+        while (this.addTimeds.length > 0) {
+            this.timedHandlers.push(this.addTimeds.pop());
+        }
+
+        // remove timed handlers that have been scheduled for deletion
+        while (this.removeTimeds.length > 0) {
+            thand = this.removeTimeds.pop();
+            i = this.timedHandlers.indexOf(thand);
+            if (i >= 0) {
+                this.timedHandlers.splice(i, 1);
+            }
+        }
+
+        // call ready timed handlers
+        var now = new Date().getTime();
+        newList = [];
+        for (i = 0; i < this.timedHandlers.length; i++) {
+            thand = this.timedHandlers[i];
+            if (this.authenticated || !thand.user) {
+                since = thand.lastCalled + thand.period;
+                if (since - now <= 0) {
+                    if (thand.run()) {
+                        newList.push(thand);
+                    }
+                } else {
+                    newList.push(thand);
+                }
+            }
+        }
+        this.timedHandlers = newList;
+
+        clearTimeout(this._idleTimeout);
+
+        this._proto._onIdle();
+
+        // reactivate the timer only if connected
+        if (this.connected) {
+            this._idleTimeout = setTimeout(this._onIdle.bind(this), 100);
+        }
+    }
+};
+
+if (callback) {
+    callback(Strophe, $build, $msg, $iq, $pres);
+}
+
+/** Class: Strophe.SASLMechanism
+ *
+ *  encapsulates SASL authentication mechanisms.
+ *
+ *  User code may override the priority for each mechanism or disable it completely.
+ *  See <priority> for information about changing priority and <test> for informatian on
+ *  how to disable a mechanism.
+ *
+ *  By default, all mechanisms are enabled and the priorities are
+ *
+ *  SCRAM-SHA1 - 40
+ *  DIGEST-MD5 - 30
+ *  Plain - 20
+ */
+
+/**
+ * PrivateConstructor: Strophe.SASLMechanism
+ * SASL auth mechanism abstraction.
+ *
+ *  Parameters:
+ *    (String) name - SASL Mechanism name.
+ *    (Boolean) isClientFirst - If client should send response first without challenge.
+ *    (Number) priority - Priority.
+ *
+ *  Returns:
+ *    A new Strophe.SASLMechanism object.
+ */
+Strophe.SASLMechanism = function(name, isClientFirst, priority) {
+  /** PrivateVariable: name
+   *  Mechanism name.
+   */
+  this.name = name;
+  /** PrivateVariable: isClientFirst
+   *  If client sends response without initial server challenge.
+   */
+  this.isClientFirst = isClientFirst;
+  /** Variable: priority
+   *  Determines which <SASLMechanism> is chosen for authentication (Higher is better).
+   *  Users may override this to prioritize mechanisms differently.
+   *
+   *  In the default configuration the priorities are
+   *
+   *  SCRAM-SHA1 - 40
+   *  DIGEST-MD5 - 30
+   *  Plain - 20
+   *
+   *  Example: (This will cause Strophe to choose the mechanism that the server sent first)
+   *
+   *  > Strophe.SASLMD5.priority = Strophe.SASLSHA1.priority;
+   *
+   *  See <SASL mechanisms> for a list of available mechanisms.
+   *
+   */
+  this.priority = priority;
+};
+
+Strophe.SASLMechanism.prototype = {
+  /**
+   *  Function: test
+   *  Checks if mechanism able to run.
+   *  To disable a mechanism, make this return false;
+   *
+   *  To disable plain authentication run
+   *  > Strophe.SASLPlain.test = function() {
+   *  >   return false;
+   *  > }
+   *
+   *  See <SASL mechanisms> for a list of available mechanisms.
+   *
+   *  Parameters:
+   *    (Strophe.Connection) connection - Target Connection.
+   *
+   *  Returns:
+   *    (Boolean) If mechanism was able to run.
+   */
+  /* jshint unused:false */
+  test: function(connection) {
+    return true;
+  },
+  /* jshint unused:true */
+
+  /** PrivateFunction: onStart
+   *  Called before starting mechanism on some connection.
+   *
+   *  Parameters:
+   *    (Strophe.Connection) connection - Target Connection.
+   */
+  onStart: function(connection)
+  {
+    this._connection = connection;
+  },
+
+  /** PrivateFunction: onChallenge
+   *  Called by protocol implementation on incoming challenge. If client is
+   *  first (isClientFirst == true) challenge will be null on the first call.
+   *
+   *  Parameters:
+   *    (Strophe.Connection) connection - Target Connection.
+   *    (String) challenge - current challenge to handle.
+   *
+   *  Returns:
+   *    (String) Mechanism response.
+   */
+  /* jshint unused:false */
+  onChallenge: function(connection, challenge) {
+    throw new Error("You should implement challenge handling!");
+  },
+  /* jshint unused:true */
+
+  /** PrivateFunction: onFailure
+   *  Protocol informs mechanism implementation about SASL failure.
+   */
+  onFailure: function() {
+    this._connection = null;
+  },
+
+  /** PrivateFunction: onSuccess
+   *  Protocol informs mechanism implementation about SASL success.
+   */
+  onSuccess: function() {
+    this._connection = null;
+  }
+};
+
+  /** Constants: SASL mechanisms
+   *  Available authentication mechanisms
+   *
+   *  Strophe.SASLAnonymous - SASL Anonymous authentication.
+   *  Strophe.SASLPlain - SASL Plain authentication.
+   *  Strophe.SASLMD5 - SASL Digest-MD5 authentication
+   *  Strophe.SASLSHA1 - SASL SCRAM-SHA1 authentication
+   */
+
+// Building SASL callbacks
+
+/** PrivateConstructor: SASLAnonymous
+ *  SASL Anonymous authentication.
+ */
+Strophe.SASLAnonymous = function() {};
+
+Strophe.SASLAnonymous.prototype = new Strophe.SASLMechanism("ANONYMOUS", false, 10);
+
+Strophe.SASLAnonymous.test = function(connection) {
+  return connection.authcid === null;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLAnonymous.prototype.name] = Strophe.SASLAnonymous;
+
+/** PrivateConstructor: SASLPlain
+ *  SASL Plain authentication.
+ */
+Strophe.SASLPlain = function() {};
+
+Strophe.SASLPlain.prototype = new Strophe.SASLMechanism("PLAIN", true, 20);
+
+Strophe.SASLPlain.test = function(connection) {
+  return connection.authcid !== null;
+};
+
+Strophe.SASLPlain.prototype.onChallenge = function(connection) {
+  var auth_str = connection.authzid;
+  auth_str = auth_str + "\u0000";
+  auth_str = auth_str + connection.authcid;
+  auth_str = auth_str + "\u0000";
+  auth_str = auth_str + connection.pass;
+  return auth_str;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLPlain.prototype.name] = Strophe.SASLPlain;
+
+/** PrivateConstructor: SASLSHA1
+ *  SASL SCRAM SHA 1 authentication.
+ */
+Strophe.SASLSHA1 = function() {};
+
+/* TEST:
+ * This is a simple example of a SCRAM-SHA-1 authentication exchange
+ * when the client doesn't support channel bindings (username 'user' and
+ * password 'pencil' are used):
+ *
+ * C: n,,n=user,r=fyko+d2lbbFgONRv9qkxdawL
+ * S: r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j,s=QSXCR+Q6sek8bf92,
+ * i=4096
+ * C: c=biws,r=fyko+d2lbbFgONRv9qkxdawL3rfcNHYJY1ZVvWVs7j,
+ * p=v0X8v3Bz2T0CJGbJQyF0X+HI4Ts=
+ * S: v=rmF9pqV8S7suAoZWja4dJRkFsKQ=
+ *
+ */
+
+Strophe.SASLSHA1.prototype = new Strophe.SASLMechanism("SCRAM-SHA-1", true, 40);
+
+Strophe.SASLSHA1.test = function(connection) {
+  return connection.authcid !== null;
+};
+
+Strophe.SASLSHA1.prototype.onChallenge = function(connection, challenge, test_cnonce) {
+  var cnonce = test_cnonce || MD5.hexdigest(Math.random() * 1234567890);
+
+  var auth_str = "n=" + connection.authcid;
+  auth_str += ",r=";
+  auth_str += cnonce;
+
+  connection._sasl_data.cnonce = cnonce;
+  connection._sasl_data["client-first-message-bare"] = auth_str;
+
+  auth_str = "n,," + auth_str;
+
+  this.onChallenge = function (connection, challenge)
+  {
+    var nonce, salt, iter, Hi, U, U_old, i, k;
+    var clientKey, serverKey, clientSignature;
+    var responseText = "c=biws,";
+    var authMessage = connection._sasl_data["client-first-message-bare"] + "," +
+      challenge + ",";
+    var cnonce = connection._sasl_data.cnonce;
+    var attribMatch = /([a-z]+)=([^,]+)(,|$)/;
+
+    while (challenge.match(attribMatch)) {
+      var matches = challenge.match(attribMatch);
+      challenge = challenge.replace(matches[0], "");
+      switch (matches[1]) {
+      case "r":
+        nonce = matches[2];
+        break;
+      case "s":
+        salt = matches[2];
+        break;
+      case "i":
+        iter = matches[2];
+        break;
+      }
+    }
+
+    if (nonce.substr(0, cnonce.length) !== cnonce) {
+      connection._sasl_data = {};
+      return connection._sasl_failure_cb();
+    }
+
+    responseText += "r=" + nonce;
+    authMessage += responseText;
+
+    salt = Base64.decode(salt);
+    salt += "\x00\x00\x00\x01";
+
+    Hi = U_old = core_hmac_sha1(connection.pass, salt);
+    for (i = 1; i < iter; i++) {
+      U = core_hmac_sha1(connection.pass, binb2str(U_old));
+      for (k = 0; k < 5; k++) {
+        Hi[k] ^= U[k];
+      }
+      U_old = U;
+    }
+    Hi = binb2str(Hi);
+
+    clientKey = core_hmac_sha1(Hi, "Client Key");
+    serverKey = str_hmac_sha1(Hi, "Server Key");
+    clientSignature = core_hmac_sha1(str_sha1(binb2str(clientKey)), authMessage);
+    connection._sasl_data["server-signature"] = b64_hmac_sha1(serverKey, authMessage);
+
+    for (k = 0; k < 5; k++) {
+      clientKey[k] ^= clientSignature[k];
+    }
+
+    responseText += ",p=" + Base64.encode(binb2str(clientKey));
+
+    return responseText;
+  }.bind(this);
+
+  return auth_str;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLSHA1.prototype.name] = Strophe.SASLSHA1;
+
+/** PrivateConstructor: SASLMD5
+ *  SASL DIGEST MD5 authentication.
+ */
+Strophe.SASLMD5 = function() {};
+
+Strophe.SASLMD5.prototype = new Strophe.SASLMechanism("DIGEST-MD5", false, 30);
+
+Strophe.SASLMD5.test = function(connection) {
+  return connection.authcid !== null;
+};
+
+/** PrivateFunction: _quote
+ *  _Private_ utility function to backslash escape and quote strings.
+ *
+ *  Parameters:
+ *    (String) str - The string to be quoted.
+ *
+ *  Returns:
+ *    quoted string
+ */
+Strophe.SASLMD5.prototype._quote = function (str)
+  {
+    return '"' + str.replace(/\\/g, "\\\\").replace(/"/g, '\\"') + '"';
+    //" end string workaround for emacs
+  };
+
+
+Strophe.SASLMD5.prototype.onChallenge = function(connection, challenge, test_cnonce) {
+  var attribMatch = /([a-z]+)=("[^"]+"|[^,"]+)(?:,|$)/;
+  var cnonce = test_cnonce || MD5.hexdigest("" + (Math.random() * 1234567890));
+  var realm = "";
+  var host = null;
+  var nonce = "";
+  var qop = "";
+  var matches;
+
+  while (challenge.match(attribMatch)) {
+    matches = challenge.match(attribMatch);
+    challenge = challenge.replace(matches[0], "");
+    matches[2] = matches[2].replace(/^"(.+)"$/, "$1");
+    switch (matches[1]) {
+    case "realm":
+      realm = matches[2];
+      break;
+    case "nonce":
+      nonce = matches[2];
+      break;
+    case "qop":
+      qop = matches[2];
+      break;
+    case "host":
+      host = matches[2];
+      break;
+    }
+  }
+
+  var digest_uri = connection.servtype + "/" + connection.domain;
+  if (host !== null) {
+    digest_uri = digest_uri + "/" + host;
+  }
+
+  var A1 = MD5.hash(connection.authcid +
+                    ":" + realm + ":" + this._connection.pass) +
+    ":" + nonce + ":" + cnonce;
+  var A2 = 'AUTHENTICATE:' + digest_uri;
+
+  var responseText = "";
+  responseText += 'charset=utf-8,';
+  responseText += 'username=' +
+    this._quote(connection.authcid) + ',';
+  responseText += 'realm=' + this._quote(realm) + ',';
+  responseText += 'nonce=' + this._quote(nonce) + ',';
+  responseText += 'nc=00000001,';
+  responseText += 'cnonce=' + this._quote(cnonce) + ',';
+  responseText += 'digest-uri=' + this._quote(digest_uri) + ',';
+  responseText += 'response=' + MD5.hexdigest(MD5.hexdigest(A1) + ":" +
+                                              nonce + ":00000001:" +
+                                              cnonce + ":auth:" +
+                                              MD5.hexdigest(A2)) + ",";
+  responseText += 'qop=auth';
+
+  this.onChallenge = function ()
+  {
+    return "";
+  }.bind(this);
+
+  return responseText;
+};
+
+Strophe.Connection.prototype.mechanisms[Strophe.SASLMD5.prototype.name] = Strophe.SASLMD5;
+
+})(function () {
+    window.Strophe = arguments[0];
+    window.$build = arguments[1];
+    window.$msg = arguments[2];
+    window.$iq = arguments[3];
+    window.$pres = arguments[4];
+});
+
+/*
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+
+/* jshint undef: true, unused: true:, noarg: true, latedef: true */
+/*global window, setTimeout, clearTimeout,
+    XMLHttpRequest, ActiveXObject,
+    Strophe, $build */
+
+
+/** PrivateClass: Strophe.Request
+ *  _Private_ helper class that provides a cross implementation abstraction
+ *  for a BOSH related XMLHttpRequest.
+ *
+ *  The Strophe.Request class is used internally to encapsulate BOSH request
+ *  information.  It is not meant to be used from user's code.
+ */
+
+/** PrivateConstructor: Strophe.Request
+ *  Create and initialize a new Strophe.Request object.
+ *
+ *  Parameters:
+ *    (XMLElement) elem - The XML data to be sent in the request.
+ *    (Function) func - The function that will be called when the
+ *      XMLHttpRequest readyState changes.
+ *    (Integer) rid - The BOSH rid attribute associated with this request.
+ *    (Integer) sends - The number of times this same request has been
+ *      sent.
+ */
+Strophe.Request = function (elem, func, rid, sends)
+{
+    this.id = ++Strophe._requestId;
+    this.xmlData = elem;
+    this.data = Strophe.serialize(elem);
+    // save original function in case we need to make a new request
+    // from this one.
+    this.origFunc = func;
+    this.func = func;
+    this.rid = rid;
+    this.date = NaN;
+    this.sends = sends || 0;
+    this.abort = false;
+    this.dead = null;
+
+    this.age = function () {
+        if (!this.date) { return 0; }
+        var now = new Date();
+        return (now - this.date) / 1000;
+    };
+    this.timeDead = function () {
+        if (!this.dead) { return 0; }
+        var now = new Date();
+        return (now - this.dead) / 1000;
+    };
+    this.xhr = this._newXHR();
+};
+
+Strophe.Request.prototype = {
+    /** PrivateFunction: getResponse
+     *  Get a response from the underlying XMLHttpRequest.
+     *
+     *  This function attempts to get a response from the request and checks
+     *  for errors.
+     *
+     *  Throws:
+     *    "parsererror" - A parser error occured.
+     *
+     *  Returns:
+     *    The DOM element tree of the response.
+     */
+    getResponse: function ()
+    {
+        var node = null;
+        if (this.xhr.responseXML && this.xhr.responseXML.documentElement) {
+            node = this.xhr.responseXML.documentElement;
+            if (node.tagName == "parsererror") {
+                Strophe.error("invalid response received");
+                Strophe.error("responseText: " + this.xhr.responseText);
+                Strophe.error("responseXML: " +
+                              Strophe.serialize(this.xhr.responseXML));
+                throw "parsererror";
+            }
+        } else if (this.xhr.responseText) {
+            Strophe.error("invalid response received");
+            Strophe.error("responseText: " + this.xhr.responseText);
+            Strophe.error("responseXML: " +
+                          Strophe.serialize(this.xhr.responseXML));
+        }
+
+        return node;
+    },
+
+    /** PrivateFunction: _newXHR
+     *  _Private_ helper function to create XMLHttpRequests.
+     *
+     *  This function creates XMLHttpRequests across all implementations.
+     *
+     *  Returns:
+     *    A new XMLHttpRequest.
+     */
+    _newXHR: function ()
+    {
+        var xhr = null;
+        if (window.XMLHttpRequest) {
+            xhr = new XMLHttpRequest();
+            if (xhr.overrideMimeType) {
+                xhr.overrideMimeType("text/xml");
+            }
+        } else if (window.ActiveXObject) {
+            xhr = new ActiveXObject("Microsoft.XMLHTTP");
+        }
+
+        // use Function.bind() to prepend ourselves as an argument
+        xhr.onreadystatechange = this.func.bind(null, this);
+
+        return xhr;
+    }
+};
+
+/** Class: Strophe.Bosh
+ *  _Private_ helper class that handles BOSH Connections
+ *
+ *  The Strophe.Bosh class is used internally by Strophe.Connection
+ *  to encapsulate BOSH sessions. It is not meant to be used from user's code.
+ */
+
+/** File: bosh.js
+ *  A JavaScript library to enable BOSH in Strophejs.
+ *
+ *  this library uses Bidirectional-streams Over Synchronous HTTP (BOSH)
+ *  to emulate a persistent, stateful, two-way connection to an XMPP server.
+ *  More information on BOSH can be found in XEP 124.
+ */
+
+/** PrivateConstructor: Strophe.Bosh
+ *  Create and initialize a Strophe.Bosh object.
+ *
+ *  Parameters:
+ *    (Strophe.Connection) connection - The Strophe.Connection that will use BOSH.
+ *
+ *  Returns:
+ *    A new Strophe.Bosh object.
+ */
+Strophe.Bosh = function(connection) {
+    this._conn = connection;
+    /* request id for body tags */
+    this.rid = Math.floor(Math.random() * 4294967295);
+    /* The current session ID. */
+    this.sid = null;
+
+    // default BOSH values
+    this.hold = 1;
+    this.wait = 60;
+    this.window = 5;
+
+    this._requests = [];
+};
+
+Strophe.Bosh.prototype = {
+    /** Variable: strip
+     *
+     *  BOSH-Connections will have all stanzas wrapped in a <body> tag when
+     *  passed to <Strophe.Connection.xmlInput> or <Strophe.Connection.xmlOutput>.
+     *  To strip this tag, User code can set <Strophe.Bosh.strip> to "body":
+     *
+     *  > Strophe.Bosh.prototype.strip = "body";
+     *
+     *  This will enable stripping of the body tag in both
+     *  <Strophe.Connection.xmlInput> and <Strophe.Connection.xmlOutput>.
+     */
+    strip: null,
+
+    /** PrivateFunction: _buildBody
+     *  _Private_ helper function to generate the <body/> wrapper for BOSH.
+     *
+     *  Returns:
+     *    A Strophe.Builder with a <body/> element.
+     */
+    _buildBody: function ()
+    {
+        var bodyWrap = $build('body', {
+            rid: this.rid++,
+            xmlns: Strophe.NS.HTTPBIND
+        });
+
+        if (this.sid !== null) {
+            bodyWrap.attrs({sid: this.sid});
+        }
+
+        return bodyWrap;
+    },
+
+    /** PrivateFunction: _reset
+     *  Reset the connection.
+     *
+     *  This function is called by the reset function of the Strophe Connection
+     */
+    _reset: function ()
+    {
+        this.rid = Math.floor(Math.random() * 4294967295);
+        this.sid = null;
+        
+        jQuery(document).trigger('ridChange', {rid: this.rid});
+    },
+
+    /** PrivateFunction: _connect
+     *  _Private_ function that initializes the BOSH connection.
+     *
+     *  Creates and sends the Request that initializes the BOSH connection.
+     */
+    _connect: function (wait, hold, route)
+    {
+        this.wait = wait || this.wait;
+        this.hold = hold || this.hold;
+
+        // build the body tag
+        var body = this._buildBody().attrs({
+            to: this._conn.domain,
+            "xml:lang": "en",
+            wait: this.wait,
+            hold: this.hold,
+            content: "text/xml; charset=utf-8",
+            ver: "1.6",
+            "xmpp:version": "1.0",
+            "xmlns:xmpp": Strophe.NS.BOSH
+        });
+
+        if(route){
+            body.attrs({
+                route: route
+            });
+        }
+
+        var _connect_cb = this._conn._connect_cb;
+
+        this._requests.push(
+            new Strophe.Request(body.tree(),
+                                this._onRequestStateChange.bind(
+                                    this, _connect_cb.bind(this._conn)),
+                                body.tree().getAttribute("rid")));
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _attach
+     *  Attach to an already created and authenticated BOSH session.
+     *
+     *  This function is provided to allow Strophe to attach to BOSH
+     *  sessions which have been created externally, perhaps by a Web
+     *  application.  This is often used to support auto-login type features
+     *  without putting user credentials into the page.
+     *
+     *  Parameters:
+     *    (String) jid - The full JID that is bound by the session.
+     *    (String) sid - The SID of the BOSH session.
+     *    (String) rid - The current RID of the BOSH session.  This RID
+     *      will be used by the next request.
+     *    (Function) callback The connect callback function.
+     *    (Integer) wait - The optional HTTPBIND wait value.  This is the
+     *      time the server will wait before returning an empty result for
+     *      a request.  The default setting of 60 seconds is recommended.
+     *      Other settings will require tweaks to the Strophe.TIMEOUT value.
+     *    (Integer) hold - The optional HTTPBIND hold value.  This is the
+     *      number of connections the server will hold at one time.  This
+     *      should almost always be set to 1 (the default).
+     *    (Integer) wind - The optional HTTBIND window value.  This is the
+     *      allowed range of request ids that are valid.  The default is 5.
+     */
+    _attach: function (jid, sid, rid, callback, wait, hold, wind)
+    {
+        this._conn.jid = jid;
+        this.sid = sid;
+        this.rid = rid;
+
+        this._conn.connect_callback = callback;
+
+        this._conn.domain = Strophe.getDomainFromJid(this._conn.jid);
+
+        this._conn.authenticated = true;
+        this._conn.connected = true;
+
+        this.wait = wait || this.wait;
+        this.hold = hold || this.hold;
+        this.window = wind || this.window;
+
+        this._conn._changeConnectStatus(Strophe.Status.ATTACHED, null);
+    },
+
+    /** PrivateFunction: _connect_cb
+     *  _Private_ handler for initial connection request.
+     *
+     *  This handler is used to process the Bosh-part of the initial request.
+     *  Parameters:
+     *    (Strophe.Request) bodyWrap - The received stanza.
+     */
+    _connect_cb: function (bodyWrap)
+    {
+        var typ = bodyWrap.getAttribute("type");
+        var cond, conflict;
+        if (typ !== null && typ == "terminate") {
+            // an error occurred
+            Strophe.error("BOSH-Connection failed: " + cond);
+            cond = bodyWrap.getAttribute("condition");
+            conflict = bodyWrap.getElementsByTagName("conflict");
+            if (cond !== null) {
+                if (cond == "remote-stream-error" && conflict.length > 0) {
+                    cond = "conflict";
+                }
+                this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, cond);
+            } else {
+                this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "unknown");
+            }
+            this._conn._doDisconnect();
+            return Strophe.Status.CONNFAIL;
+        }
+
+        // check to make sure we don't overwrite these if _connect_cb is
+        // called multiple times in the case of missing stream:features
+        if (!this.sid) {
+            this.sid = bodyWrap.getAttribute("sid");
+        }
+        var wind = bodyWrap.getAttribute('requests');
+        if (wind) { this.window = parseInt(wind, 10); }
+        var hold = bodyWrap.getAttribute('hold');
+        if (hold) { this.hold = parseInt(hold, 10); }
+        var wait = bodyWrap.getAttribute('wait');
+        if (wait) { this.wait = parseInt(wait, 10); }
+    },
+
+    /** PrivateFunction: _disconnect
+     *  _Private_ part of Connection.disconnect for Bosh
+     *
+     *  Parameters:
+     *    (Request) pres - This stanza will be sent before disconnecting.
+     */
+    _disconnect: function (pres)
+    {
+        this._sendTerminate(pres);
+    },
+
+    /** PrivateFunction: _doDisconnect
+     *  _Private_ function to disconnect.
+     *
+     *  Resets the SID and RID.
+     */
+    _doDisconnect: function ()
+    {
+        this.sid = null;
+        this.rid = Math.floor(Math.random() * 4294967295);
+        
+        jQuery(document).trigger('ridChange', {rid: this.rid});
+    },
+
+    /** PrivateFunction: _emptyQueue
+     * _Private_ function to check if the Request queue is empty.
+     *
+     *  Returns:
+     *    True, if there are no Requests queued, False otherwise.
+     */
+    _emptyQueue: function ()
+    {
+        return this._requests.length === 0;
+    },
+
+    /** PrivateFunction: _hitError
+     *  _Private_ function to handle the error count.
+     *
+     *  Requests are resent automatically until their error count reaches
+     *  5.  Each time an error is encountered, this function is called to
+     *  increment the count and disconnect if the count is too high.
+     *
+     *  Parameters:
+     *    (Integer) reqStatus - The request status.
+     */
+    _hitError: function (reqStatus)
+    {
+        this.errors++;
+        Strophe.warn("request errored, status: " + reqStatus +
+                     ", number of errors: " + this.errors);
+        if (this.errors > 4) {
+            this._onDisconnectTimeout();
+        }
+    },
+
+    /** PrivateFunction: _no_auth_received
+     *
+     * Called on stream start/restart when no stream:features
+     * has been received and sends a blank poll request.
+     */
+    _no_auth_received: function (_callback)
+    {
+        if (_callback) {
+            _callback = _callback.bind(this._conn);
+        } else {
+            _callback = this._conn._connect_cb.bind(this._conn);
+        }
+        var body = this._buildBody();
+        this._requests.push(
+                new Strophe.Request(body.tree(),
+                    this._onRequestStateChange.bind(
+                        this, _callback.bind(this._conn)),
+                    body.tree().getAttribute("rid")));
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _onDisconnectTimeout
+     *  _Private_ timeout handler for handling non-graceful disconnection.
+     *
+     *  Cancels all remaining Requests and clears the queue.
+     */
+    _onDisconnectTimeout: function ()
+    {
+        var req;
+        while (this._requests.length > 0) {
+            req = this._requests.pop();
+            req.abort = true;
+            req.xhr.abort();
+            // jslint complains, but this is fine. setting to empty func
+            // is necessary for IE6
+            req.xhr.onreadystatechange = function () {}; // jshint ignore:line
+        }
+    },
+
+    /** PrivateFunction: _onIdle
+     *  _Private_ handler called by Strophe.Connection._onIdle
+     *
+     *  Sends all queued Requests or polls with empty Request if there are none.
+     */
+    _onIdle: function () {
+        var data = this._conn._data;
+
+        // if no requests are in progress, poll
+        if (this._conn.authenticated && this._requests.length === 0 &&
+            data.length === 0 && !this._conn.disconnecting) {
+            Strophe.info("no requests during idle cycle, sending " +
+                         "blank request");
+            data.push(null);
+        }
+
+        if (this._requests.length < 2 && data.length > 0 &&
+            !this._conn.paused) {
+            var body = this._buildBody();
+            for (var i = 0; i < data.length; i++) {
+                if (data[i] !== null) {
+                    if (data[i] === "restart") {
+                        body.attrs({
+                            to: this._conn.domain,
+                            "xml:lang": "en",
+                            "xmpp:restart": "true",
+                            "xmlns:xmpp": Strophe.NS.BOSH
+                        });
+                    } else {
+                        body.cnode(data[i]).up();
+                    }
+                }
+            }
+            delete this._conn._data;
+            this._conn._data = [];
+            this._requests.push(
+                new Strophe.Request(body.tree(),
+                                    this._onRequestStateChange.bind(
+                                        this, this._conn._dataRecv.bind(this._conn)),
+                                    body.tree().getAttribute("rid")));
+            this._processRequest(this._requests.length - 1);
+        }
+
+        if (this._requests.length > 0) {
+            var time_elapsed = this._requests[0].age();
+            if (this._requests[0].dead !== null) {
+                if (this._requests[0].timeDead() >
+                    Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait)) {
+                    this._throttledRequestHandler();
+                }
+            }
+
+            if (time_elapsed > Math.floor(Strophe.TIMEOUT * this.wait)) {
+                Strophe.warn("Request " +
+                             this._requests[0].id +
+                             " timed out, over " + Math.floor(Strophe.TIMEOUT * this.wait) +
+                             " seconds since last activity");
+                this._throttledRequestHandler();
+            }
+        }
+    },
+
+    /** PrivateFunction: _onRequestStateChange
+     *  _Private_ handler for Strophe.Request state changes.
+     *
+     *  This function is called when the XMLHttpRequest readyState changes.
+     *  It contains a lot of error handling logic for the many ways that
+     *  requests can fail, and calls the request callback when requests
+     *  succeed.
+     *
+     *  Parameters:
+     *    (Function) func - The handler for the request.
+     *    (Strophe.Request) req - The request that is changing readyState.
+     */
+    _onRequestStateChange: function (func, req)
+    {
+        Strophe.debug("request id " + req.id +
+                      "." + req.sends + " state changed to " +
+                      req.xhr.readyState);
+
+        if (req.abort) {
+            req.abort = false;
+            return;
+        }
+
+        if(req.xhr.readyState == 2){ 
+           jQuery(document).trigger('ridChange', {rid: Number(req.rid)+1});
+        }
+        
+        // request complete
+        var reqStatus;
+        if (req.xhr.readyState == 4) {
+            reqStatus = 0;
+            try {
+                reqStatus = req.xhr.status;
+            } catch (e) {
+                // ignore errors from undefined status attribute.  works
+                // around a browser bug
+            }
+
+            if (typeof(reqStatus) == "undefined") {
+                reqStatus = 0;
+            }
+
+            if (this.disconnecting) {
+                if (reqStatus >= 400) {
+                    this._hitError(reqStatus);
+                    return;
+                }
+            }
+
+            var reqIs0 = (this._requests[0] == req);
+            var reqIs1 = (this._requests[1] == req);
+
+            if ((reqStatus > 0 && reqStatus < 500) || req.sends > 5) {
+                // remove from internal queue
+                this._removeRequest(req);
+                Strophe.debug("request id " +
+                              req.id +
+                              " should now be removed");
+            }
+
+            // request succeeded
+            if (reqStatus == 200) {
+                // if request 1 finished, or request 0 finished and request
+                // 1 is over Strophe.SECONDARY_TIMEOUT seconds old, we need to
+                // restart the other - both will be in the first spot, as the
+                // completed request has been removed from the queue already
+                if (reqIs1 ||
+                    (reqIs0 && this._requests.length > 0 &&
+                     this._requests[0].age() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait))) {
+                    this._restartRequest(0);
+                }
+                // call handler
+                Strophe.debug("request id " +
+                              req.id + "." +
+                              req.sends + " got 200");
+                func(req);
+                this.errors = 0;
+            } else {
+                Strophe.error("request id " +
+                              req.id + "." +
+                              req.sends + " error " + reqStatus +
+                              " happened");
+                if (reqStatus === 0 ||
+                    (reqStatus >= 400 && reqStatus < 600) ||
+                    reqStatus >= 12000) {
+                    this._hitError(reqStatus);
+                    if (reqStatus >= 400 && reqStatus < 500) {
+                        this._conn._changeConnectStatus(Strophe.Status.DISCONNECTING,
+                                                  null);
+                        this._conn._doDisconnect();
+                    }
+                }
+            }
+
+            if (!((reqStatus > 0 && reqStatus < 500) ||
+                  req.sends > 5)) {
+                this._throttledRequestHandler();
+            }
+        }
+    },
+
+    /** PrivateFunction: _processRequest
+     *  _Private_ function to process a request in the queue.
+     *
+     *  This function takes requests off the queue and sends them and
+     *  restarts dead requests.
+     *
+     *  Parameters:
+     *    (Integer) i - The index of the request in the queue.
+     */
+    _processRequest: function (i)
+    {
+        var self = this;
+        var req = this._requests[i];
+        var reqStatus = -1;
+
+        try {
+            if (req.xhr.readyState == 4) {
+                reqStatus = req.xhr.status;
+            }
+        } catch (e) {
+            Strophe.error("caught an error in _requests[" + i +
+                          "], reqStatus: " + reqStatus);
+        }
+
+        if (typeof(reqStatus) == "undefined") {
+            reqStatus = -1;
+        }
+
+        // make sure we limit the number of retries
+        if (req.sends > this.maxRetries) {
+            this._onDisconnectTimeout();
+            return;
+        }
+
+        var time_elapsed = req.age();
+        var primaryTimeout = (!isNaN(time_elapsed) &&
+                              time_elapsed > Math.floor(Strophe.TIMEOUT * this.wait));
+        var secondaryTimeout = (req.dead !== null &&
+                                req.timeDead() > Math.floor(Strophe.SECONDARY_TIMEOUT * this.wait));
+        var requestCompletedWithServerError = (req.xhr.readyState == 4 &&
+                                               (reqStatus < 1 ||
+                                                reqStatus >= 500));
+        if (primaryTimeout || secondaryTimeout ||
+            requestCompletedWithServerError) {
+            if (secondaryTimeout) {
+                Strophe.error("Request " +
+                              this._requests[i].id +
+                              " timed out (secondary), restarting");
+            }
+            req.abort = true;
+            req.xhr.abort();
+            // setting to null fails on IE6, so set to empty function
+            req.xhr.onreadystatechange = function () {};
+            this._requests[i] = new Strophe.Request(req.xmlData,
+                                                    req.origFunc,
+                                                    req.rid,
+                                                    req.sends);
+            req = this._requests[i];
+        }
+
+        if (req.xhr.readyState === 0) {
+            Strophe.debug("request id " + req.id +
+                          "." + req.sends + " posting");
+
+            try {
+                req.xhr.open("POST", this._conn.service, this._conn.options.sync ? false : true);
+            } catch (e2) {
+                Strophe.error("XHR open failed.");
+                if (!this._conn.connected) {
+                    this._conn._changeConnectStatus(Strophe.Status.CONNFAIL,
+                                              "bad-service");
+                }
+                this._conn.disconnect();
+                return;
+            }
+
+            // Fires the XHR request -- may be invoked immediately
+            // or on a gradually expanding retry window for reconnects
+            var sendFunc = function () {
+                req.date = new Date();
+                if (self._conn.options.customHeaders){
+                    var headers = self._conn.options.customHeaders;
+                    for (var header in headers) {
+                        if (headers.hasOwnProperty(header)) {
+                            req.xhr.setRequestHeader(header, headers[header]);
+                        }
+                    }
+                }
+                req.xhr.send(req.data);
+            };
+
+            // Implement progressive backoff for reconnects --
+            // First retry (send == 1) should also be instantaneous
+            if (req.sends > 1) {
+                // Using a cube of the retry number creates a nicely
+                // expanding retry window
+                var backoff = Math.min(Math.floor(Strophe.TIMEOUT * this.wait),
+                                       Math.pow(req.sends, 3)) * 1000;
+                setTimeout(sendFunc, backoff);
+            } else {
+                sendFunc();
+            }
+
+            req.sends++;
+
+            if (this._conn.xmlOutput !== Strophe.Connection.prototype.xmlOutput) {
+                if (req.xmlData.nodeName === this.strip && req.xmlData.childNodes.length) {
+                    this._conn.xmlOutput(req.xmlData.childNodes[0]);
+                } else {
+                    this._conn.xmlOutput(req.xmlData);
+                }
+            }
+            if (this._conn.rawOutput !== Strophe.Connection.prototype.rawOutput) {
+                this._conn.rawOutput(req.data);
+            }
+        } else {
+            Strophe.debug("_processRequest: " +
+                          (i === 0 ? "first" : "second") +
+                          " request has readyState of " +
+                          req.xhr.readyState);
+        }
+    },
+
+    /** PrivateFunction: _removeRequest
+     *  _Private_ function to remove a request from the queue.
+     *
+     *  Parameters:
+     *    (Strophe.Request) req - The request to remove.
+     */
+    _removeRequest: function (req)
+    {
+        Strophe.debug("removing request");
+
+        var i;
+        for (i = this._requests.length - 1; i >= 0; i--) {
+            if (req == this._requests[i]) {
+                this._requests.splice(i, 1);
+            }
+        }
+
+        // IE6 fails on setting to null, so set to empty function
+        req.xhr.onreadystatechange = function () {};
+
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _restartRequest
+     *  _Private_ function to restart a request that is presumed dead.
+     *
+     *  Parameters:
+     *    (Integer) i - The index of the request in the queue.
+     */
+    _restartRequest: function (i)
+    {
+        var req = this._requests[i];
+        if (req.dead === null) {
+            req.dead = new Date();
+        }
+
+        this._processRequest(i);
+    },
+
+    /** PrivateFunction: _reqToData
+     * _Private_ function to get a stanza out of a request.
+     *
+     * Tries to extract a stanza out of a Request Object.
+     * When this fails the current connection will be disconnected.
+     *
+     *  Parameters:
+     *    (Object) req - The Request.
+     *
+     *  Returns:
+     *    The stanza that was passed.
+     */
+    _reqToData: function (req)
+    {
+        try {
+            return req.getResponse();
+        } catch (e) {
+            if (e != "parsererror") { throw e; }
+            this._conn.disconnect("strophe-parsererror");
+        }
+    },
+
+    /** PrivateFunction: _sendTerminate
+     *  _Private_ function to send initial disconnect sequence.
+     *
+     *  This is the first step in a graceful disconnect.  It sends
+     *  the BOSH server a terminate body and includes an unavailable
+     *  presence if authentication has completed.
+     */
+    _sendTerminate: function (pres)
+    {
+        Strophe.info("_sendTerminate was called");
+        var body = this._buildBody().attrs({type: "terminate"});
+
+        if (pres) {
+            body.cnode(pres.tree());
+        }
+
+        var req = new Strophe.Request(body.tree(),
+                                      this._onRequestStateChange.bind(
+                                          this, this._conn._dataRecv.bind(this._conn)),
+                                      body.tree().getAttribute("rid"));
+
+        this._requests.push(req);
+        this._throttledRequestHandler();
+    },
+
+    /** PrivateFunction: _send
+     *  _Private_ part of the Connection.send function for BOSH
+     *
+     * Just triggers the RequestHandler to send the messages that are in the queue
+     */
+    _send: function () {
+        clearTimeout(this._conn._idleTimeout);
+        this._throttledRequestHandler();
+        this._conn._idleTimeout = setTimeout(this._conn._onIdle.bind(this._conn), 100);
+    },
+
+    /** PrivateFunction: _sendRestart
+     *
+     *  Send an xmpp:restart stanza.
+     */
+    _sendRestart: function ()
+    {
+        this._throttledRequestHandler();
+        clearTimeout(this._conn._idleTimeout);
+    },
+
+    /** PrivateFunction: _throttledRequestHandler
+     *  _Private_ function to throttle requests to the connection window.
+     *
+     *  This function makes sure we don't send requests so fast that the
+     *  request ids overflow the connection window in the case that one
+     *  request died.
+     */
+    _throttledRequestHandler: function ()
+    {
+        if (!this._requests) {
+            Strophe.debug("_throttledRequestHandler called with " +
+                          "undefined requests");
+        } else {
+            Strophe.debug("_throttledRequestHandler called with " +
+                          this._requests.length + " requests");
+        }
+
+        if (!this._requests || this._requests.length === 0) {
+            return;
+        }
+
+        if (this._requests.length > 0) {
+            this._processRequest(0);
+        }
+
+        if (this._requests.length > 1 &&
+            Math.abs(this._requests[0].rid -
+                     this._requests[1].rid) < this.window) {
+            this._processRequest(1);
+        }
+    }
+};
+
+/*
+    This program is distributed under the terms of the MIT license.
+    Please see the LICENSE file for details.
+
+    Copyright 2006-2008, OGG, LLC
+*/
+
+/* jshint undef: true, unused: true:, noarg: true, latedef: true */
+/*global document, window, clearTimeout, WebSocket,
+    DOMParser, Strophe, $build */
+
+/** Class: Strophe.WebSocket
+ *  _Private_ helper class that handles WebSocket Connections
+ *
+ *  The Strophe.WebSocket class is used internally by Strophe.Connection
+ *  to encapsulate WebSocket sessions. It is not meant to be used from user's code.
+ */
+
+/** File: websocket.js
+ *  A JavaScript library to enable XMPP over Websocket in Strophejs.
+ *
+ *  This file implements XMPP over WebSockets for Strophejs.
+ *  If a Connection is established with a Websocket url (ws://...)
+ *  Strophe will use WebSockets.
+ *  For more information on XMPP-over WebSocket see this RFC draft:
+ *  http://tools.ietf.org/html/draft-ietf-xmpp-websocket-00
+ *
+ *  WebSocket support implemented by Andreas Guth (andreas.guth at rwth-aachen.de)
+ */
+
+/** PrivateConstructor: Strophe.Websocket
+ *  Create and initialize a Strophe.WebSocket object.
+ *  Currently only sets the connection Object.
+ *
+ *  Parameters:
+ *    (Strophe.Connection) connection - The Strophe.Connection that will use WebSockets.
+ *
+ *  Returns:
+ *    A new Strophe.WebSocket object.
+ */
+Strophe.Websocket = function(connection) {
+    this._conn = connection;
+    this.strip = "stream:stream";
+
+    var service = connection.service;
+    if (service.indexOf("ws:") !== 0 && service.indexOf("wss:") !== 0) {
+        // If the service is not an absolute URL, assume it is a path and put the absolute
+        // URL together from options, current URL and the path.
+        var new_service = "";
+
+        if (connection.options.protocol === "ws" && window.location.protocol !== "https:") {
+            new_service += "ws";
+        } else {
+            new_service += "wss";
+        }
+
+        new_service += "://" + window.location.host;
+
+        if (service.indexOf("/") !== 0) {
+            new_service += window.location.pathname + service;
+        } else {
+            new_service += service;
+        }
+
+        connection.service = new_service;
+    }
+};
+
+Strophe.Websocket.prototype = {
+    /** PrivateFunction: _buildStream
+     *  _Private_ helper function to generate the <stream> start tag for WebSockets
+     *
+     *  Returns:
+     *    A Strophe.Builder with a <stream> element.
+     */
+    _buildStream: function ()
+    {
+        return $build("stream:stream", {
+            "to": this._conn.domain,
+            "xmlns": Strophe.NS.CLIENT,
+            "xmlns:stream": Strophe.NS.STREAM,
+            "version": '1.0'
+        });
+    },
+
+    /** PrivateFunction: _check_streamerror
+     * _Private_ checks a message for stream:error
+     *
+     *  Parameters:
+     *    (Strophe.Request) bodyWrap - The received stanza.
+     *    connectstatus - The ConnectStatus that will be set on error.
+     *  Returns:
+     *     true if there was a streamerror, false otherwise.
+     */
+    _check_streamerror: function (bodyWrap, connectstatus) {
+        var errors = bodyWrap.getElementsByTagName("stream:error");
+        if (errors.length === 0) {
+            return false;
+        }
+        var error = errors[0];
+
+        var condition = "";
+        var text = "";
+
+        var ns = "urn:ietf:params:xml:ns:xmpp-streams";
+        for (var i = 0; i < error.childNodes.length; i++) {
+            var e = error.childNodes[i];
+            if (e.getAttribute("xmlns") !== ns) {
+                break;
+            } if (e.nodeName === "text") {
+                text = e.textContent;
+            } else {
+                condition = e.nodeName;
+            }
+        }
+
+        var errorString = "WebSocket stream error: ";
+
+        if (condition) {
+            errorString += condition;
+        } else {
+            errorString += "unknown";
+        }
+
+        if (text) {
+            errorString += " - " + condition;
+        }
+
+        Strophe.error(errorString);
+
+        // close the connection on stream_error
+        this._conn._changeConnectStatus(connectstatus, condition);
+        this._conn._doDisconnect();
+        return true;
+    },
+
+    /** PrivateFunction: _reset
+     *  Reset the connection.
+     *
+     *  This function is called by the reset function of the Strophe Connection.
+     *  Is not needed by WebSockets.
+     */
+    _reset: function ()
+    {
+        return;
+    },
+
+    /** PrivateFunction: _connect
+     *  _Private_ function called by Strophe.Connection.connect
+     *
+     *  Creates a WebSocket for a connection and assigns Callbacks to it.
+     *  Does nothing if there already is a WebSocket.
+     */
+    _connect: function () {
+        // Ensure that there is no open WebSocket from a previous Connection.
+        this._closeSocket();
+
+        // Create the new WobSocket
+        this.socket = new WebSocket(this._conn.service, "xmpp");
+        this.socket.onopen = this._onOpen.bind(this);
+        this.socket.onerror = this._onError.bind(this);
+        this.socket.onclose = this._onClose.bind(this);
+        this.socket.onmessage = this._connect_cb_wrapper.bind(this);
+    },
+
+    /** PrivateFunction: _connect_cb
+     *  _Private_ function called by Strophe.Connection._connect_cb
+     *
+     * checks for stream:error
+     *
+     *  Parameters:
+     *    (Strophe.Request) bodyWrap - The received stanza.
+     */
+    _connect_cb: function(bodyWrap) {
+        var error = this._check_streamerror(bodyWrap, Strophe.Status.CONNFAIL);
+        if (error) {
+            return Strophe.Status.CONNFAIL;
+        }
+    },
+
+    /** PrivateFunction: _handleStreamStart
+     * _Private_ function that checks the opening stream:stream tag for errors.
+     *
+     * Disconnects if there is an error and returns false, true otherwise.
+     *
+     *  Parameters:
+     *    (Node) message - Stanza containing the stream:stream.
+     */
+    _handleStreamStart: function(message) {
+        var error = false;
+        // Check for errors in the stream:stream tag
+        var ns = message.getAttribute("xmlns");
+        if (typeof ns !== "string") {
+            error = "Missing xmlns in stream:stream";
+        } else if (ns !== Strophe.NS.CLIENT) {
+            error = "Wrong xmlns in stream:stream: " + ns;
+        }
+
+        var ns_stream = message.namespaceURI;
+        if (typeof ns_stream !== "string") {
+            error = "Missing xmlns:stream in stream:stream";
+        } else if (ns_stream !== Strophe.NS.STREAM) {
+            error = "Wrong xmlns:stream in stream:stream: " + ns_stream;
+        }
+
+        var ver = message.getAttribute("version");
+        if (typeof ver !== "string") {
+            error = "Missing version in stream:stream";
+        } else if (ver !== "1.0") {
+            error = "Wrong version in stream:stream: " + ver;
+        }
+
+        if (error) {
+            this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, error);
+            this._conn._doDisconnect();
+            return false;
+        }
+
+        return true;
+    },
+
+    /** PrivateFunction: _connect_cb_wrapper
+     * _Private_ function that handles the first connection messages.
+     *
+     * On receiving an opening stream tag this callback replaces itself with the real
+     * message handler. On receiving a stream error the connection is terminated.
+     */
+    _connect_cb_wrapper: function(message) {
+        if (message.data.indexOf("<stream:stream ") === 0 || message.data.indexOf("<?xml") === 0) {
+            // Strip the XML Declaration, if there is one
+            var data = message.data.replace(/^(<\?.*?\?>\s*)*/, "");
+            if (data === '') return;
+
+            //Make the initial stream:stream selfclosing to parse it without a SAX parser.
+            data = message.data.replace(/<stream:stream (.*[^\/])>/, "<stream:stream $1/>");
+
+            var streamStart = new DOMParser().parseFromString(data, "text/xml").documentElement;
+            this._conn.xmlInput(streamStart);
+            this._conn.rawInput(message.data);
+
+            //_handleStreamSteart will check for XML errors and disconnect on error
+            if (this._handleStreamStart(streamStart)) {
+
+                //_connect_cb will check for stream:error and disconnect on error
+                this._connect_cb(streamStart);
+
+                // ensure received stream:stream is NOT selfclosing and save it for following messages
+                this.streamStart = message.data.replace(/^<stream:(.*)\/>$/, "<stream:$1>");
+            }
+        } else if (message.data === "</stream:stream>") {
+            this._conn.rawInput(message.data);
+            this._conn.xmlInput(document.createElement("stream:stream"));
+            this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "Received closing stream");
+            this._conn._doDisconnect();
+            return;
+        } else {
+            var string = this._streamWrap(message.data);
+            var elem = new DOMParser().parseFromString(string, "text/xml").documentElement;
+            this.socket.onmessage = this._onMessage.bind(this);
+            this._conn._connect_cb(elem, null, message.data);
+        }
+    },
+
+    /** PrivateFunction: _disconnect
+     *  _Private_ function called by Strophe.Connection.disconnect
+     *
+     *  Disconnects and sends a last stanza if one is given
+     *
+     *  Parameters:
+     *    (Request) pres - This stanza will be sent before disconnecting.
+     */
+    _disconnect: function (pres)
+    {
+        if (this.socket.readyState !== WebSocket.CLOSED) {
+            if (pres) {
+                this._conn.send(pres);
+            }
+            var close = '</stream:stream>';
+            this._conn.xmlOutput(document.createElement("stream:stream"));
+            this._conn.rawOutput(close);
+            try {
+                this.socket.send(close);
+            } catch (e) {
+                Strophe.info("Couldn't send closing stream tag.");
+            }
+        }
+
+        this._conn._doDisconnect();
+    },
+
+    /** PrivateFunction: _doDisconnect
+     *  _Private_ function to disconnect.
+     *
+     *  Just closes the Socket for WebSockets
+     */
+    _doDisconnect: function ()
+    {
+        Strophe.info("WebSockets _doDisconnect was called");
+        this._closeSocket();
+    },
+
+    /** PrivateFunction _streamWrap
+     *  _Private_ helper function to wrap a stanza in a <stream> tag.
+     *  This is used so Strophe can process stanzas from WebSockets like BOSH
+     */
+    _streamWrap: function (stanza)
+    {
+        return this.streamStart + stanza + '</stream:stream>';
+    },
+
+
+    /** PrivateFunction: _closeSocket
+     *  _Private_ function to close the WebSocket.
+     *
+     *  Closes the socket if it is still open and deletes it
+     */
+    _closeSocket: function ()
+    {
+        if (this.socket) { try {
+            this.socket.close();
+        } catch (e) {} }
+        this.socket = null;
+    },
+
+    /** PrivateFunction: _emptyQueue
+     * _Private_ function to check if the message queue is empty.
+     *
+     *  Returns:
+     *    True, because WebSocket messages are send immediately after queueing.
+     */
+    _emptyQueue: function ()
+    {
+        return true;
+    },
+
+    /** PrivateFunction: _onClose
+     * _Private_ function to handle websockets closing.
+     *
+     * Nothing to do here for WebSockets
+     */
+    _onClose: function() {
+        if(this._conn.connected && !this._conn.disconnecting) {
+            Strophe.error("Websocket closed unexcectedly");
+            this._conn._doDisconnect();
+        } else {
+            Strophe.info("Websocket closed");
+        }
+    },
+
+    /** PrivateFunction: _no_auth_received
+     *
+     * Called on stream start/restart when no stream:features
+     * has been received.
+     */
+    _no_auth_received: function (_callback)
+    {
+        Strophe.error("Server did not send any auth methods");
+        this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "Server did not send any auth methods");
+        if (_callback) {
+            _callback = _callback.bind(this._conn);
+            _callback();
+        }
+        this._conn._doDisconnect();
+    },
+
+    /** PrivateFunction: _onDisconnectTimeout
+     *  _Private_ timeout handler for handling non-graceful disconnection.
+     *
+     *  This does nothing for WebSockets
+     */
+    _onDisconnectTimeout: function () {},
+
+    /** PrivateFunction: _onError
+     * _Private_ function to handle websockets errors.
+     *
+     * Parameters:
+     * (Object) error - The websocket error.
+     */
+    _onError: function(error) {
+        Strophe.error("Websocket error " + error);
+        this._conn._changeConnectStatus(Strophe.Status.CONNFAIL, "The WebSocket connection could not be established was disconnected.");
+        this._disconnect();
+    },
+
+    /** PrivateFunction: _onIdle
+     *  _Private_ function called by Strophe.Connection._onIdle
+     *
+     *  sends all queued stanzas
+     */
+    _onIdle: function () {
+        var data = this._conn._data;
+        if (data.length > 0 && !this._conn.paused) {
+            for (var i = 0; i < data.length; i++) {
+                if (data[i] !== null) {
+                    var stanza, rawStanza;
+                    if (data[i] === "restart") {
+                        stanza = this._buildStream();
+                        rawStanza = this._removeClosingTag(stanza);
+                        stanza = stanza.tree();
+                    } else {
+                        stanza = data[i];
+                        rawStanza = Strophe.serialize(stanza);
+                    }
+                    this._conn.xmlOutput(stanza);
+                    this._conn.rawOutput(rawStanza);
+                    this.socket.send(rawStanza);
+                }
+            }
+            this._conn._data = [];
+        }
+    },
+
+    /** PrivateFunction: _onMessage
+     * _Private_ function to handle websockets messages.
+     *
+     * This function parses each of the messages as if they are full documents. [TODO : We may actually want to use a SAX Push parser].
+     *
+     * Since all XMPP traffic starts with "<stream:stream version='1.0' xml:lang='en' xmlns='jabber:client' xmlns:stream='http://etherx.jabber.org/streams' id='3697395463' from='SERVER'>"
+     * The first stanza will always fail to be parsed...
+     * Addtionnaly, the seconds stanza will always be a <stream:features> with the stream NS defined in the previous stanza... so we need to 'force' the inclusion of the NS in this stanza!
+     *
+     * Parameters:
+     * (string) message - The websocket message.
+     */
+    _onMessage: function(message) {
+        var elem, data;
+        // check for closing stream
+        if (message.data === "</stream:stream>") {
+            var close = "</stream:stream>";
+            this._conn.rawInput(close);
+            this._conn.xmlInput(document.createElement("stream:stream"));
+            if (!this._conn.disconnecting) {
+                this._conn._doDisconnect();
+            }
+            return;
+        } else if (message.data.search("<stream:stream ") === 0) {
+            //Make the initial stream:stream selfclosing to parse it without a SAX parser.
+            data = message.data.replace(/<stream:stream (.*[^\/])>/, "<stream:stream $1/>");
+            elem = new DOMParser().parseFromString(data, "text/xml").documentElement;
+
+            if (!this._handleStreamStart(elem)) {
+                return;
+            }
+        } else {
+            data = this._streamWrap(message.data);
+            elem = new DOMParser().parseFromString(data, "text/xml").documentElement;
+        }
+
+        if (this._check_streamerror(elem, Strophe.Status.ERROR)) {
+            return;
+        }
+
+        //handle unavailable presence stanza before disconnecting
+        if (this._conn.disconnecting &&
+                elem.firstChild.nodeName === "presence" &&
+                elem.firstChild.getAttribute("type") === "unavailable") {
+            this._conn.xmlInput(elem);
+            this._conn.rawInput(Strophe.serialize(elem));
+            // if we are already disconnecting we will ignore the unavailable stanza and
+            // wait for the </stream:stream> tag before we close the connection
+            return;
+        }
+        this._conn._dataRecv(elem, message.data);
+    },
+
+    /** PrivateFunction: _onOpen
+     * _Private_ function to handle websockets connection setup.
+     *
+     * The opening stream tag is sent here.
+     */
+    _onOpen: function() {
+        Strophe.info("Websocket open");
+        var start = this._buildStream();
+        this._conn.xmlOutput(start.tree());
+
+        var startString = this._removeClosingTag(start);
+        this._conn.rawOutput(startString);
+        this.socket.send(startString);
+    },
+
+    /** PrivateFunction: _removeClosingTag
+     *  _Private_ function to Make the first <stream:stream> non-selfclosing
+     *
+     *  Parameters:
+     *      (Object) elem - The <stream:stream> tag.
+     *
+     *  Returns:
+     *      The stream:stream tag as String
+     */
+    _removeClosingTag: function(elem) {
+        var string = Strophe.serialize(elem);
+        string = string.replace(/<(stream:stream .*[^\/])\/>$/, "<$1>");
+        return string;
+    },
+
+    /** PrivateFunction: _reqToData
+     * _Private_ function to get a stanza out of a request.
+     *
+     * WebSockets don't use requests, so the passed argument is just returned.
+     *
+     *  Parameters:
+     *    (Object) stanza - The stanza.
+     *
+     *  Returns:
+     *    The stanza that was passed.
+     */
+    _reqToData: function (stanza)
+    {
+        return stanza;
+    },
+
+    /** PrivateFunction: _send
+     *  _Private_ part of the Connection.send function for WebSocket
+     *
+     * Just flushes the messages that are in the queue
+     */
+    _send: function () {
+        this._conn.flush();
+    },
+
+    /** PrivateFunction: _sendRestart
+     *
+     *  Send an xmpp:restart stanza.
+     */
+    _sendRestart: function ()
+    {
+        clearTimeout(this._conn._idleTimeout);
+        this._conn._onIdle.bind(this._conn)();
+    }
+};
diff --git a/lib/strophe.muc.js b/lib/strophe.muc.js
new file mode 100644
index 0000000..fe7100e
--- /dev/null
+++ b/lib/strophe.muc.js
@@ -0,0 +1,1020 @@
+// Generated by CoffeeScript 1.3.3
+/*
+ *Plugin to implement the MUC extension.
+   http://xmpp.org/extensions/xep-0045.html
+ *Previous Author:
+    Nathan Zorn <nathan.zorn at gmail.com>
+ *Complete CoffeeScript rewrite:
+    Andreas Guth <guth at dbis.rwth-aachen.de>
+*/
+
+var Occupant, RoomConfig, XmppRoom,
+  __bind = function(fn, me){ return function(){ return fn.apply(me, arguments); }; };
+
+Strophe.addConnectionPlugin('muc', {
+  _connection: null,
+  rooms: {},
+  roomNames: [],
+  /*Function 
+  Initialize the MUC plugin. Sets the correct connection object and
+  extends the namesace.
+  */
+
+  init: function(conn) {
+    this._connection = conn;
+    this._muc_handler = null;
+    Strophe.addNamespace('MUC_OWNER', Strophe.NS.MUC + "#owner");
+    Strophe.addNamespace('MUC_ADMIN', Strophe.NS.MUC + "#admin");
+    Strophe.addNamespace('MUC_USER', Strophe.NS.MUC + "#user");
+    return Strophe.addNamespace('MUC_ROOMCONF', Strophe.NS.MUC + "#roomconfig");
+  },
+  /*Function
+  Join a multi-user chat room
+  Parameters:
+  (String) room - The multi-user chat room to join.
+  (String) nick - The nickname to use in the chat room. Optional
+  (Function) msg_handler_cb - The function call to handle messages from the
+  specified chat room.
+  (Function) pres_handler_cb - The function call back to handle presence
+  in the chat room.
+  (Function) roster_cb - The function call to handle roster info in the chat room
+  (String) password - The optional password to use. (password protected
+  rooms only)
+  (Object) history_attrs - Optional attributes for retrieving history
+  (XML DOM Element) extended_presence - Optional XML for extending presence
+  */
+
+  join: function(room, nick, msg_handler_cb, pres_handler_cb, roster_cb, password, history_attrs, extended_presence) {
+    var msg, room_nick, _ref,
+      _this = this;
+    room_nick = this.test_append_nick(room, nick);
+    msg = $pres({
+      from: this._connection.jid,
+      to: room_nick
+    }).c("x", {
+      xmlns: Strophe.NS.MUC
+    });
+    if (history_attrs != null) {
+      msg = msg.c("history", history_attrs).up();
+    }
+    if (password != null) {
+      msg.cnode(Strophe.xmlElement("password", [], password));
+    }
+    if (extended_presence != null) {
+      msg.up().cnode(extended_presence);
+    }
+    if ((_ref = this._muc_handler) == null) {
+      this._muc_handler = this._connection.addHandler(function(stanza) {
+        var from, handler, handlers, id, roomname, x, xmlns, xquery, _i, _len;
+        from = stanza.getAttribute('from');
+        if (!from) {
+          return true;
+        }
+        roomname = from.split("/")[0];
+        if (!_this.rooms[roomname]) {
+          return true;
+        }
+        room = _this.rooms[roomname];
+        handlers = {};
+        if (stanza.nodeName === "message") {
+          handlers = room._message_handlers;
+        } else if (stanza.nodeName === "presence") {
+          xquery = stanza.getElementsByTagName("x");
+          if (xquery.length > 0) {
+            for (_i = 0, _len = xquery.length; _i < _len; _i++) {
+              x = xquery[_i];
+              xmlns = x.getAttribute("xmlns");
+              if (xmlns && xmlns.match(Strophe.NS.MUC)) {
+                handlers = room._presence_handlers;
+                break;
+              }
+            }
+          }
+        }
+        for (id in handlers) {
+          handler = handlers[id];
+          if (!handler(stanza, room)) {
+            delete handlers[id];
+          }
+        }
+        return true;
+      });
+    }
+    if (!this.rooms.hasOwnProperty(room)) {
+      this.rooms[room] = new XmppRoom(this, room, nick, password);
+      this.roomNames.push(room);
+    }
+    if (pres_handler_cb) {
+      this.rooms[room].addHandler('presence', pres_handler_cb);
+    }
+    if (msg_handler_cb) {
+      this.rooms[room].addHandler('message', msg_handler_cb);
+    }
+    if (roster_cb) {
+      this.rooms[room].addHandler('roster', roster_cb);
+    }
+    return this._connection.send(msg);
+  },
+  /*Function
+  Leave a multi-user chat room
+  Parameters:
+  (String) room - The multi-user chat room to leave.
+  (String) nick - The nick name used in the room.
+  (Function) handler_cb - Optional function to handle the successful leave.
+  (String) exit_msg - optional exit message.
+  Returns:
+  iqid - The unique id for the room leave.
+  */
+
+  leave: function(room, nick, handler_cb, exit_msg) {
+    var id, presence, presenceid, room_nick;
+    id = this.roomNames.indexOf(room);
+    delete this.rooms[room];
+    if (id >= 0) {
+      this.roomNames.splice(id, 1);
+      if (this.roomNames.length === 0) {
+        this._connection.deleteHandler(this._muc_handler);
+        this._muc_handler = null;
+      }
+    }
+    room_nick = this.test_append_nick(room, nick);
+    presenceid = this._connection.getUniqueId();
+    presence = $pres({
+      type: "unavailable",
+      id: presenceid,
+      from: this._connection.jid,
+      to: room_nick
+    });
+    if (exit_msg != null) {
+      presence.c("status", exit_msg);
+    }
+    if (handler_cb != null) {
+      this._connection.addHandler(handler_cb, null, "presence", null, presenceid);
+    }
+    this._connection.send(presence);
+    return presenceid;
+  },
+  /*Function
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) nick - The nick name used in the chat room.
+  (String) message - The plaintext message to send to the room.
+  (String) html_message - The message to send to the room with html markup.
+  (String) type - "groupchat" for group chat messages o
+                  "chat" for private chat messages
+  Returns:
+  msgiq - the unique id used to send the message
+  */
+
+  message: function(room, nick, message, html_message, type) {
+    var msg, msgid, parent, room_nick;
+    room_nick = this.test_append_nick(room, nick);
+    type = type || (nick != null ? "chat" : "groupchat");
+    msgid = this._connection.getUniqueId();
+    msg = $msg({
+      to: room_nick,
+      from: this._connection.jid,
+      type: type,
+      id: msgid
+    }).c("body", {
+      xmlns: Strophe.NS.CLIENT
+    }).t(message);
+    msg.up();
+    if (html_message != null) {
+      msg.c("html", {
+        xmlns: Strophe.NS.XHTML_IM
+      }).c("body", {
+        xmlns: Strophe.NS.XHTML
+      }).t(html_message);
+      if (msg.node.childNodes.length === 0) {
+        parent = msg.node.parentNode;
+        msg.up().up();
+        msg.node.removeChild(parent);
+      } else {
+        msg.up().up();
+      }
+    }
+    msg.c("x", {
+      xmlns: "jabber:x:event"
+    }).c("composing");
+    this._connection.send(msg);
+    return msgid;
+  },
+  /*Function
+  Convenience Function to send a Message to all Occupants
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) message - The plaintext message to send to the room.
+  (String) html_message - The message to send to the room with html markup.
+  Returns:
+  msgiq - the unique id used to send the message
+  */
+
+  groupchat: function(room, message, html_message) {
+    return this.message(room, null, message, html_message);
+  },
+  /*Function
+  Send a mediated invitation.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) receiver - The invitation's receiver.
+  (String) reason - Optional reason for joining the room.
+  Returns:
+  msgiq - the unique id used to send the invitation
+  */
+
+  invite: function(room, receiver, reason) {
+    var invitation, msgid;
+    msgid = this._connection.getUniqueId();
+    invitation = $msg({
+      from: this._connection.jid,
+      to: room,
+      id: msgid
+    }).c('x', {
+      xmlns: Strophe.NS.MUC_USER
+    }).c('invite', {
+      to: receiver
+    });
+    if (reason != null) {
+      invitation.c('reason', reason);
+    }
+    this._connection.send(invitation);
+    return msgid;
+  },
+  /*Function
+  Send a direct invitation.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) receiver - The invitation's receiver.
+  (String) reason - Optional reason for joining the room.
+  (String) password - Optional password for the room.
+  Returns:
+  msgiq - the unique id used to send the invitation
+  */
+
+  directInvite: function(room, receiver, reason, password) {
+    var attrs, invitation, msgid;
+    msgid = this._connection.getUniqueId();
+    attrs = {
+      xmlns: 'jabber:x:conference',
+      jid: room
+    };
+    if (reason != null) {
+      attrs.reason = reason;
+    }
+    if (password != null) {
+      attrs.password = password;
+    }
+    invitation = $msg({
+      from: this._connection.jid,
+      to: receiver,
+      id: msgid
+    }).c('x', attrs);
+    this._connection.send(invitation);
+    return msgid;
+  },
+  /*Function
+  Queries a room for a list of occupants
+  (String) room - The multi-user chat room name.
+  (Function) success_cb - Optional function to handle the info.
+  (Function) error_cb - Optional function to handle an error.
+  Returns:
+  id - the unique id used to send the info request
+  */
+
+  queryOccupants: function(room, success_cb, error_cb) {
+    var attrs, info;
+    attrs = {
+      xmlns: Strophe.NS.DISCO_ITEMS
+    };
+    info = $iq({
+      from: this._connection.jid,
+      to: room,
+      type: 'get'
+    }).c('query', attrs);
+    return this._connection.sendIQ(info, success_cb, error_cb);
+  },
+  /*Function
+  Start a room configuration.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (Function) handler_cb - Optional function to handle the config form.
+  Returns:
+  id - the unique id used to send the configuration request
+  */
+
+  configure: function(room, handler_cb, error_cb) {
+    var config, stanza;
+    config = $iq({
+      to: room,
+      type: "get"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    });
+    stanza = config.tree();
+    return this._connection.sendIQ(stanza, handler_cb, error_cb);
+  },
+  /*Function
+  Cancel the room configuration
+  Parameters:
+  (String) room - The multi-user chat room name.
+  Returns:
+  id - the unique id used to cancel the configuration.
+  */
+
+  cancelConfigure: function(room) {
+    var config, stanza;
+    config = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    }).c("x", {
+      xmlns: "jabber:x:data",
+      type: "cancel"
+    });
+    stanza = config.tree();
+    return this._connection.sendIQ(stanza);
+  },
+  /*Function
+  Save a room configuration.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (Array) config- Form Object or an array of form elements used to configure the room.
+  Returns:
+  id - the unique id used to save the configuration.
+  */
+
+  saveConfiguration: function(room, config, success_cb, error_cb) {
+    var conf, iq, stanza, _i, _len;
+    iq = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    });
+    if (config instanceof Form) {
+      config.type = "submit";
+      iq.cnode(config.toXML());
+    } else {
+      iq.c("x", {
+        xmlns: "jabber:x:data",
+        type: "submit"
+      });
+      for (_i = 0, _len = config.length; _i < _len; _i++) {
+        conf = config[_i];
+        iq.cnode(conf).up();
+      }
+    }
+    stanza = iq.tree();
+    return this._connection.sendIQ(stanza, success_cb, error_cb);
+  },
+  /*Function
+  Parameters:
+  (String) room - The multi-user chat room name.
+  Returns:
+  id - the unique id used to create the chat room.
+  */
+
+  createInstantRoom: function(room, success_cb, error_cb) {
+    var roomiq;
+    roomiq = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_OWNER
+    }).c("x", {
+      xmlns: "jabber:x:data",
+      type: "submit"
+    });
+    return this._connection.sendIQ(roomiq.tree(), success_cb, error_cb);
+  },
+  /*Function
+  Set the topic of the chat room.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) topic - Topic message.
+  */
+
+  setTopic: function(room, topic) {
+    var msg;
+    msg = $msg({
+      to: room,
+      from: this._connection.jid,
+      type: "groupchat"
+    }).c("subject", {
+      xmlns: "jabber:client"
+    }).t(topic);
+    return this._connection.send(msg.tree());
+  },
+  /*Function
+  Internal Function that Changes the role or affiliation of a member
+  of a MUC room. This function is used by modifyRole and modifyAffiliation.
+  The modification can only be done by a room moderator. An error will be
+  returned if the user doesn't have permission.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (Object) item - Object with nick and role or jid and affiliation attribute
+  (String) reason - Optional reason for the change.
+  (Function) handler_cb - Optional callback for success
+  (Function) error_cb - Optional callback for error
+  Returns:
+  iq - the id of the mode change request.
+  */
+
+  _modifyPrivilege: function(room, item, reason, handler_cb, error_cb) {
+    var iq;
+    iq = $iq({
+      to: room,
+      type: "set"
+    }).c("query", {
+      xmlns: Strophe.NS.MUC_ADMIN
+    }).cnode(item.node);
+    if (reason != null) {
+      iq.c("reason", reason);
+    }
+    return this._connection.sendIQ(iq.tree(), handler_cb, error_cb);
+  },
+  /*Function
+  Changes the role of a member of a MUC room.
+  The modification can only be done by a room moderator. An error will be
+  returned if the user doesn't have permission.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) nick - The nick name of the user to modify.
+  (String) role - The new role of the user.
+  (String) affiliation - The new affiliation of the user.
+  (String) reason - Optional reason for the change.
+  (Function) handler_cb - Optional callback for success
+  (Function) error_cb - Optional callback for error
+  Returns:
+  iq - the id of the mode change request.
+  */
+
+  modifyRole: function(room, nick, role, reason, handler_cb, error_cb) {
+    var item;
+    item = $build("item", {
+      nick: nick,
+      role: role
+    });
+    return this._modifyPrivilege(room, item, reason, handler_cb, error_cb);
+  },
+  kick: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'none', reason, handler_cb, error_cb);
+  },
+  voice: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'participant', reason, handler_cb, error_cb);
+  },
+  mute: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'visitor', reason, handler_cb, error_cb);
+  },
+  op: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'moderator', reason, handler_cb, error_cb);
+  },
+  deop: function(room, nick, reason, handler_cb, error_cb) {
+    return this.modifyRole(room, nick, 'participant', reason, handler_cb, error_cb);
+  },
+  /*Function
+  Changes the affiliation of a member of a MUC room.
+  The modification can only be done by a room moderator. An error will be
+  returned if the user doesn't have permission.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) jid  - The jid of the user to modify.
+  (String) affiliation - The new affiliation of the user.
+  (String) reason - Optional reason for the change.
+  (Function) handler_cb - Optional callback for success
+  (Function) error_cb - Optional callback for error
+  Returns:
+  iq - the id of the mode change request.
+  */
+
+  modifyAffiliation: function(room, jid, affiliation, reason, handler_cb, error_cb) {
+    var item;
+    item = $build("item", {
+      jid: jid,
+      affiliation: affiliation
+    });
+    return this._modifyPrivilege(room, item, reason, handler_cb, error_cb);
+  },
+  ban: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'outcast', reason, handler_cb, error_cb);
+  },
+  member: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'member', reason, handler_cb, error_cb);
+  },
+  revoke: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'none', reason, handler_cb, error_cb);
+  },
+  owner: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'owner', reason, handler_cb, error_cb);
+  },
+  admin: function(room, jid, reason, handler_cb, error_cb) {
+    return this.modifyAffiliation(room, jid, 'admin', reason, handler_cb, error_cb);
+  },
+  /*Function
+  Change the current users nick name.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) user - The new nick name.
+  */
+
+  changeNick: function(room, user) {
+    var presence, room_nick;
+    room_nick = this.test_append_nick(room, user);
+    presence = $pres({
+      from: this._connection.jid,
+      to: room_nick,
+      id: this._connection.getUniqueId()
+    });
+    return this._connection.send(presence.tree());
+  },
+  /*Function
+  Change the current users status.
+  Parameters:
+  (String) room - The multi-user chat room name.
+  (String) user - The current nick.
+  (String) show - The new show-text.
+  (String) status - The new status-text.
+  */
+
+  setStatus: function(room, user, show, status) {
+    var presence, room_nick;
+    room_nick = this.test_append_nick(room, user);
+    presence = $pres({
+      from: this._connection.jid,
+      to: room_nick
+    });
+    if (show != null) {
+      presence.c('show', show).up();
+    }
+    if (status != null) {
+      presence.c('status', status);
+    }
+    return this._connection.send(presence.tree());
+  },
+  /*Function
+  List all chat room available on a server.
+  Parameters:
+  (String) server - name of chat server.
+  (String) handle_cb - Function to call for room list return.
+  (String) error_cb - Function to call on error.
+  */
+
+  listRooms: function(server, handle_cb, error_cb) {
+    var iq;
+    iq = $iq({
+      to: server,
+      from: this._connection.jid,
+      type: "get"
+    }).c("query", {
+      xmlns: Strophe.NS.DISCO_ITEMS
+    });
+    return this._connection.sendIQ(iq, handle_cb, error_cb);
+  },
+  test_append_nick: function(room, nick) {
+    return room + (nick != null ? "/" + (Strophe.escapeNode(nick)) : "");
+  }
+});
+
+XmppRoom = (function() {
+
+  function XmppRoom(client, name, nick, password) {
+    this.client = client;
+    this.name = name;
+    this.nick = nick;
+    this.password = password;
+    this._roomRosterHandler = __bind(this._roomRosterHandler, this);
+
+    this._addOccupant = __bind(this._addOccupant, this);
+
+    this.roster = {};
+    this._message_handlers = {};
+    this._presence_handlers = {};
+    this._roster_handlers = {};
+    this._handler_ids = 0;
+    if (client.muc) {
+      this.client = client.muc;
+    }
+    this.name = Strophe.getBareJidFromJid(name);
+    this.addHandler('presence', this._roomRosterHandler);
+  }
+
+  XmppRoom.prototype.join = function(msg_handler_cb, pres_handler_cb, roster_cb) {
+    return this.client.join(this.name, this.nick, msg_handler_cb, pres_handler_cb, roster_cb, this.password);
+  };
+
+  XmppRoom.prototype.leave = function(handler_cb, message) {
+    this.client.leave(this.name, this.nick, handler_cb, message);
+    return delete this.client.rooms[this.name];
+  };
+
+  XmppRoom.prototype.message = function(nick, message, html_message, type) {
+    return this.client.message(this.name, nick, message, html_message, type);
+  };
+
+  XmppRoom.prototype.groupchat = function(message, html_message) {
+    return this.client.groupchat(this.name, message, html_message);
+  };
+
+  XmppRoom.prototype.invite = function(receiver, reason) {
+    return this.client.invite(this.name, receiver, reason);
+  };
+
+  XmppRoom.prototype.directInvite = function(receiver, reason) {
+    return this.client.directInvite(this.name, receiver, reason, this.password);
+  };
+
+  XmppRoom.prototype.configure = function(handler_cb) {
+    return this.client.configure(this.name, handler_cb);
+  };
+
+  XmppRoom.prototype.cancelConfigure = function() {
+    return this.client.cancelConfigure(this.name);
+  };
+
+  XmppRoom.prototype.saveConfiguration = function(config) {
+    return this.client.saveConfiguration(this.name, config);
+  };
+
+  XmppRoom.prototype.queryOccupants = function(success_cb, error_cb) {
+    return this.client.queryOccupants(this.name, success_cb, error_cb);
+  };
+
+  XmppRoom.prototype.setTopic = function(topic) {
+    return this.client.setTopic(this.name, topic);
+  };
+
+  XmppRoom.prototype.modifyRole = function(nick, role, reason, success_cb, error_cb) {
+    return this.client.modifyRole(this.name, nick, role, reason, success_cb, error_cb);
+  };
+
+  XmppRoom.prototype.kick = function(nick, reason, handler_cb, error_cb) {
+    return this.client.kick(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.voice = function(nick, reason, handler_cb, error_cb) {
+    return this.client.voice(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.mute = function(nick, reason, handler_cb, error_cb) {
+    return this.client.mute(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.op = function(nick, reason, handler_cb, error_cb) {
+    return this.client.op(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.deop = function(nick, reason, handler_cb, error_cb) {
+    return this.client.deop(this.name, nick, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.modifyAffiliation = function(jid, affiliation, reason, success_cb, error_cb) {
+    return this.client.modifyAffiliation(this.name, jid, affiliation, reason, success_cb, error_cb);
+  };
+
+  XmppRoom.prototype.ban = function(jid, reason, handler_cb, error_cb) {
+    return this.client.ban(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.member = function(jid, reason, handler_cb, error_cb) {
+    return this.client.member(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.revoke = function(jid, reason, handler_cb, error_cb) {
+    return this.client.revoke(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.owner = function(jid, reason, handler_cb, error_cb) {
+    return this.client.owner(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.admin = function(jid, reason, handler_cb, error_cb) {
+    return this.client.admin(this.name, jid, reason, handler_cb, error_cb);
+  };
+
+  XmppRoom.prototype.changeNick = function(nick) {
+    this.nick = nick;
+    return this.client.changeNick(this.name, nick);
+  };
+
+  XmppRoom.prototype.setStatus = function(show, status) {
+    return this.client.setStatus(this.name, this.nick, show, status);
+  };
+
+  /*Function
+  Adds a handler to the MUC room.
+    Parameters:
+  (String) handler_type - 'message', 'presence' or 'roster'.
+  (Function) handler - The handler function.
+  Returns:
+  id - the id of handler.
+  */
+
+
+  XmppRoom.prototype.addHandler = function(handler_type, handler) {
+    var id;
+    id = this._handler_ids++;
+    switch (handler_type) {
+      case 'presence':
+        this._presence_handlers[id] = handler;
+        break;
+      case 'message':
+        this._message_handlers[id] = handler;
+        break;
+      case 'roster':
+        this._roster_handlers[id] = handler;
+        break;
+      default:
+        this._handler_ids--;
+        return null;
+    }
+    return id;
+  };
+
+  /*Function
+  Removes a handler from the MUC room.
+  This function takes ONLY ids returned by the addHandler function
+  of this room. passing handler ids returned by connection.addHandler
+  may brake things!
+    Parameters:
+  (number) id - the id of the handler
+  */
+
+
+  XmppRoom.prototype.removeHandler = function(id) {
+    delete this._presence_handlers[id];
+    delete this._message_handlers[id];
+    return delete this._roster_handlers[id];
+  };
+
+  /*Function
+  Creates and adds an Occupant to the Room Roster.
+    Parameters:
+  (Object) data - the data the Occupant is filled with
+  Returns:
+  occ - the created Occupant.
+  */
+
+
+  XmppRoom.prototype._addOccupant = function(data) {
+    var occ;
+    occ = new Occupant(data, this);
+    this.roster[occ.nick] = occ;
+    return occ;
+  };
+
+  /*Function
+  The standard handler that managed the Room Roster.
+    Parameters:
+  (Object) pres - the presence stanza containing user information
+  */
+
+
+  XmppRoom.prototype._roomRosterHandler = function(pres) {
+    var data, handler, id, newnick, nick, _ref;
+    data = XmppRoom._parsePresence(pres);
+    nick = data.nick;
+    newnick = data.newnick || null;
+    switch (data.type) {
+      case 'error':
+        return;
+      case 'unavailable':
+        if (newnick) {
+          data.nick = newnick;
+          if (this.roster[nick] && this.roster[newnick]) {
+            this.roster[nick].update(this.roster[newnick]);
+            this.roster[newnick] = this.roster[nick];
+          }
+          if (this.roster[nick] && !this.roster[newnick]) {
+            this.roster[newnick] = this.roster[nick].update(data);
+          }
+        }
+        delete this.roster[nick];
+        break;
+      default:
+        if (this.roster[nick]) {
+          this.roster[nick].update(data);
+        } else {
+          this._addOccupant(data);
+        }
+    }
+    _ref = this._roster_handlers;
+    for (id in _ref) {
+      handler = _ref[id];
+      if (!handler(this.roster, this)) {
+        delete this._roster_handlers[id];
+      }
+    }
+    return true;
+  };
+
+  /*Function
+  Parses a presence stanza
+    Parameters:
+  (Object) data - the data extracted from the presence stanza
+  */
+
+
+  XmppRoom._parsePresence = function(pres) {
+    var a, c, c2, data, _i, _j, _len, _len1, _ref, _ref1, _ref2, _ref3, _ref4, _ref5, _ref6, _ref7;
+    data = {};
+    a = pres.attributes;
+    data.nick = Strophe.getResourceFromJid(a.from.textContent);
+    data.type = ((_ref = a.type) != null ? _ref.textContent : void 0) || null;
+    data.states = [];
+    _ref1 = pres.childNodes;
+    for (_i = 0, _len = _ref1.length; _i < _len; _i++) {
+      c = _ref1[_i];
+      switch (c.nodeName) {
+        case "status":
+          data.status = c.textContent || null;
+          break;
+        case "show":
+          data.show = c.textContent || null;
+          break;
+        case "x":
+          a = c.attributes;
+          if (((_ref2 = a.xmlns) != null ? _ref2.textContent : void 0) === Strophe.NS.MUC_USER) {
+            _ref3 = c.childNodes;
+            for (_j = 0, _len1 = _ref3.length; _j < _len1; _j++) {
+              c2 = _ref3[_j];
+              switch (c2.nodeName) {
+                case "item":
+                  a = c2.attributes;
+                  data.affiliation = ((_ref4 = a.affiliation) != null ? _ref4.textContent : void 0) || null;
+                  data.role = ((_ref5 = a.role) != null ? _ref5.textContent : void 0) || null;
+                  data.jid = ((_ref6 = a.jid) != null ? _ref6.textContent : void 0) || null;
+                  data.newnick = ((_ref7 = a.nick) != null ? _ref7.textContent : void 0) || null;
+                  break;
+                case "status":
+                  if (c2.attributes.code) {
+                    data.states.push(c2.attributes.code.textContent);
+                  }
+              }
+            }
+          }
+      }
+    }
+    return data;
+  };
+
+  return XmppRoom;
+
+})();
+
+RoomConfig = (function() {
+
+  function RoomConfig(info) {
+    this.parse = __bind(this.parse, this);
+    if (info != null) {
+      this.parse(info);
+    }
+  }
+
+  RoomConfig.prototype.parse = function(result) {
+    var attr, attrs, child, field, identity, query, _i, _j, _k, _len, _len1, _len2, _ref;
+    query = result.getElementsByTagName("query")[0].childNodes;
+    this.identities = [];
+    this.features = [];
+    this.x = [];
+    for (_i = 0, _len = query.length; _i < _len; _i++) {
+      child = query[_i];
+      attrs = child.attributes;
+      switch (child.nodeName) {
+        case "identity":
+          identity = {};
+          for (_j = 0, _len1 = attrs.length; _j < _len1; _j++) {
+            attr = attrs[_j];
+            identity[attr.name] = attr.textContent;
+          }
+          this.identities.push(identity);
+          break;
+        case "feature":
+          this.features.push(attrs["var"].textContent);
+          break;
+        case "x":
+          attrs = child.childNodes[0].attributes;
+          if ((!attrs["var"].textContent === 'FORM_TYPE') || (!attrs.type.textContent === 'hidden')) {
+            break;
+          }
+          _ref = child.childNodes;
+          for (_k = 0, _len2 = _ref.length; _k < _len2; _k++) {
+            field = _ref[_k];
+            if (!(!field.attributes.type)) {
+              continue;
+            }
+            attrs = field.attributes;
+            this.x.push({
+              "var": attrs["var"].textContent,
+              label: attrs.label.textContent || "",
+              value: field.firstChild.textContent || ""
+            });
+          }
+      }
+    }
+    return {
+      "identities": this.identities,
+      "features": this.features,
+      "x": this.x
+    };
+  };
+
+  return RoomConfig;
+
+})();
+
+Occupant = (function() {
+
+  function Occupant(data, room) {
+    this.room = room;
+    this.update = __bind(this.update, this);
+
+    this.admin = __bind(this.admin, this);
+
+    this.owner = __bind(this.owner, this);
+
+    this.revoke = __bind(this.revoke, this);
+
+    this.member = __bind(this.member, this);
+
+    this.ban = __bind(this.ban, this);
+
+    this.modifyAffiliation = __bind(this.modifyAffiliation, this);
+
+    this.deop = __bind(this.deop, this);
+
+    this.op = __bind(this.op, this);
+
+    this.mute = __bind(this.mute, this);
+
+    this.voice = __bind(this.voice, this);
+
+    this.kick = __bind(this.kick, this);
+
+    this.modifyRole = __bind(this.modifyRole, this);
+
+    this.update(data);
+  }
+
+  Occupant.prototype.modifyRole = function(role, reason, success_cb, error_cb) {
+    return this.room.modifyRole(this.nick, role, reason, success_cb, error_cb);
+  };
+
+  Occupant.prototype.kick = function(reason, handler_cb, error_cb) {
+    return this.room.kick(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.voice = function(reason, handler_cb, error_cb) {
+    return this.room.voice(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.mute = function(reason, handler_cb, error_cb) {
+    return this.room.mute(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.op = function(reason, handler_cb, error_cb) {
+    return this.room.op(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.deop = function(reason, handler_cb, error_cb) {
+    return this.room.deop(this.nick, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.modifyAffiliation = function(affiliation, reason, success_cb, error_cb) {
+    return this.room.modifyAffiliation(this.jid, affiliation, reason, success_cb, error_cb);
+  };
+
+  Occupant.prototype.ban = function(reason, handler_cb, error_cb) {
+    return this.room.ban(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.member = function(reason, handler_cb, error_cb) {
+    return this.room.member(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.revoke = function(reason, handler_cb, error_cb) {
+    return this.room.revoke(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.owner = function(reason, handler_cb, error_cb) {
+    return this.room.owner(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.admin = function(reason, handler_cb, error_cb) {
+    return this.room.admin(this.jid, reason, handler_cb, error_cb);
+  };
+
+  Occupant.prototype.update = function(data) {
+    this.nick = data.nick || null;
+    this.affiliation = data.affiliation || null;
+    this.role = data.role || null;
+    this.jid = data.jid || null;
+    this.status = data.status || null;
+    this.show = data.show || null;
+    return this;
+  };
+
+  return Occupant;
+
+})();
\ No newline at end of file
diff --git a/lib/strophe.vcard.js b/lib/strophe.vcard.js
new file mode 100644
index 0000000..9ad1c13
--- /dev/null
+++ b/lib/strophe.vcard.js
@@ -0,0 +1,66 @@
+// Generated by CoffeeScript 1.3.3
+/*
+Plugin to implement the vCard extension.
+http://xmpp.org/extensions/xep-0054.html
+
+Author: Nathan Zorn (nathan.zorn at gmail.com)
+CoffeeScript port: Andreas Guth (guth at dbis.rwth-aachen.de)
+*/
+
+/* jslint configuration:
+*/
+
+/* global document, window, setTimeout, clearTimeout, console,
+    XMLHttpRequest, ActiveXObject,
+    Base64, MD5,
+    Strophe, $build, $msg, $iq, $pres
+*/
+
+var buildIq;
+
+buildIq = function(type, jid, vCardEl) {
+  var iq;
+  iq = $iq(jid ? {
+    type: type,
+    to: jid
+  } : {
+    type: type
+  });
+  iq.c("vCard", {
+    xmlns: Strophe.NS.VCARD
+  });
+  if (vCardEl) {
+    iq.cnode(vCardEl);
+  }
+  return iq;
+};
+
+Strophe.addConnectionPlugin('vcard', {
+  _connection: null,
+  init: function(conn) {
+    this._connection = conn;
+    return Strophe.addNamespace('VCARD', 'vcard-temp');
+  },
+  /*Function
+    Retrieve a vCard for a JID/Entity
+    Parameters:
+    (Function) handler_cb - The callback function used to handle the request.
+    (String) jid - optional - The name of the entity to request the vCard
+       If no jid is given, this function retrieves the current user's vcard.
+  */
+
+  get: function(handler_cb, jid, error_cb) {
+    var iq;
+    iq = buildIq("get", jid);
+    return this._connection.sendIQ(iq, handler_cb, error_cb);
+  },
+  /* Function
+      Set an entity's vCard.
+  */
+
+  set: function(handler_cb, vCardEl, jid, error_cb) {
+    var iq;
+    iq = buildIq("set", jid, vCardEl);
+    return this._connection.sendIQ(iq, handler_cb, error_rb);
+  }
+});
\ No newline at end of file
diff --git a/sound/Ping1.mp3 b/sound/Ping1.mp3
new file mode 100644
index 0000000..d429eee
Binary files /dev/null and b/sound/Ping1.mp3 differ
diff --git a/sound/Rotary-Phone6.mp3 b/sound/Rotary-Phone6.mp3
new file mode 100644
index 0000000..c978c8d
Binary files /dev/null and b/sound/Rotary-Phone6.mp3 differ
diff --git a/sound/credential b/sound/credential
new file mode 100644
index 0000000..522d7ce
--- /dev/null
+++ b/sound/credential
@@ -0,0 +1,11 @@
+Rotary-Phone6.mp3
+===
+Creator: David English
+Source: http://www.beepzoid.com/old-phones/
+License: unknown
+
+Ping1.mp3
+===
+Creator: CameronMusic
+Source: https://soundcloud.com/freefilmandgamemusic/ping-1?in=freefilmandgamemusic/sets/free-notification-sounds-and
+License: cc 3.0, http://creativecommons.org/licenses/by/3.0/
\ No newline at end of file
diff --git a/sound/incomingMessage.wav b/sound/incomingMessage.wav
new file mode 100644
index 0000000..012dbab
Binary files /dev/null and b/sound/incomingMessage.wav differ

-- 
Alioth's /usr/local/bin/git-commit-notice on /srv/git.debian.org/git/pkg-javascript/libjs-jsxc.git



More information about the Pkg-javascript-commits mailing list