├── LICENSE-MIT.txt ├── TODO ├── changelog ├── demo ├── chat.js └── web │ ├── css │ ├── layout.css │ └── reset.css │ ├── images │ ├── background.png │ ├── button.png │ ├── footer.png │ ├── glows.png │ ├── header-bg.png │ ├── inset-border-l.png │ ├── inset-border.png │ ├── metal.jpg │ ├── node-chat.png │ └── send.png │ ├── index.html │ └── js │ ├── client.js │ └── jquery-1.4.2.js ├── lib ├── channel.js ├── router │ ├── index.js │ └── mime.js ├── server.js └── session.js ├── package-lock.json ├── package.json ├── readme.md └── web └── nodechat.js /LICENSE-MIT.txt: -------------------------------------------------------------------------------- 1 | Copyright 2013 Scott González http://scottgonzalez.com 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining 4 | a copy of this software and associated documentation files (the 5 | "Software"), to deal in the Software without restriction, including 6 | without limitation the rights to use, copy, modify, merge, publish, 7 | distribute, sublicense, and/or sell copies of the Software, and to 8 | permit persons to whom the Software is furnished to do so, subject to 9 | the following conditions: 10 | 11 | The above copyright notice and this permission notice shall be 12 | included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 21 | -------------------------------------------------------------------------------- /TODO: -------------------------------------------------------------------------------- 1 | - find a better router module 2 | -- update relevant requests to POSTs 3 | 4 | Demo: 5 | - style the scrollbar 6 | - handle long user lists in the UI 7 | - add current user's message before hitting the server 8 | - run a /who on login to find all users 9 | - indicate dropped connections and auto-reconnect in demo 10 | -------------------------------------------------------------------------------- /changelog: -------------------------------------------------------------------------------- 1 | 2010.05.18, v0.2.1, 156141845042d0ac6d231823c311b2a1c7ffa86d 2 | - Added better error handling. 3 | - Added unique id to messages/events. 4 | - Changed since parameter for channel.query (/recv) to use the unique id instead of the timestamp. 5 | - Fixed demo so it doesn't auto-scroll if the user has scrolled up. 6 | - Fixed demo to reset unread count on focus. 7 | 8 | 9 | 2010.04.27, v0.2.0, 0aeea97488bf599bca877b057ca1356ed4530be2 10 | - Moved stdout logging from the channel to the demo (no built-in logging). 11 | - Removed event prefixes in client, e.g., join instead of nodechat-join. 12 | - Added events to the channel instances on the server to match the events on the client. 13 | - Added error handling for invalid channels. 14 | - Moved callback flushing and session expiration to each channel to reduce blocking. 15 | - Updated to work with node 0.1.9x with no warnings. 16 | - Simplified demo by serving all files in /web as static files instead of explicitly handling each file. 17 | - Added error handling for dropped connections to the demo. 18 | - Added success and error callbacks for nodeChat.join (client). 19 | - Added a readme with documentation. 20 | 21 | 22 | 2010.04.12, v0.1.0, 95467b513366ba71e220528262b376b21b16df98 23 | - Initial release. 24 | -------------------------------------------------------------------------------- /demo/chat.js: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env node 2 | var fs = require("fs"), 3 | chat = require('../lib/server'), 4 | router = require("../lib/router"); 5 | 6 | // create chat server 7 | var chatServer = chat.createServer(); 8 | chatServer.listen(8001, function() { 9 | console.log("Server is running on port 8001"); 10 | }); 11 | 12 | // create a channel and log all activity to stdout 13 | chatServer.addChannel({ 14 | basePath: "/chat" 15 | }).addListener("msg", function(msg) { 16 | console.log("<" + msg.nick + "> " + msg.text); 17 | }).addListener("join", function(msg) { 18 | console.log(msg.nick + " join"); 19 | }).addListener("part", function(msg) { 20 | console.log(msg.nick + " part"); 21 | }); 22 | 23 | // server static web files 24 | function serveFiles(localDir, webDir) { 25 | fs.readdirSync(localDir).forEach(function(file) { 26 | var local = localDir + "/" + file, 27 | web = webDir + "/" + file; 28 | 29 | if (fs.statSync(local).isDirectory()) { 30 | serveFiles(local, web); 31 | } else { 32 | chatServer.passThru(web, router.staticHandler(local)); 33 | } 34 | }); 35 | } 36 | serveFiles(__dirname + "/web", ""); 37 | chatServer.passThru("/js/nodechat.js", router.staticHandler(__dirname + "/../web/nodechat.js")); 38 | chatServer.passThru("/", router.staticHandler(__dirname + "/web/index.html")); 39 | -------------------------------------------------------------------------------- /demo/web/css/layout.css: -------------------------------------------------------------------------------- 1 | @import "reset.css"; 2 | 3 | html { 4 | background: #3c3c3c url(../images/background.png); 5 | } 6 | 7 | html, body { 8 | width: 100%; 9 | height: 100%; 10 | } 11 | 12 | /* Fonts */ 13 | 14 | /* Normal type */ 15 | body, #users li, #entry input[type=text], a#submit, #chat-log { 16 | font-family: Helvetica, Arial, sans-serif; 17 | } 18 | 19 | /* Special Type */ 20 | footer p.credits, #login h1 { 21 | font-family: "Gill Sans", "Gill Sans MT", Helvetica, Arial, sans-serif; 22 | } 23 | 24 | /** HEADER */ 25 | 26 | body > header { 27 | background: #000 url(../images/header-bg.png) repeat-x; 28 | 29 | -webkit-box-shadow: rgba(255,255,255,0.1) 0 1px 0; 30 | -moz-box-shadow: rgba(255,255,255,0.1) 0 1px 0; 31 | box-shadow: rgba(255,255,255,0.1) 0 1px 0; 32 | position: fixed; 33 | 34 | top: 0; 35 | right: 0; 36 | left: 0; 37 | 38 | height: 23px; 39 | 40 | display: block; 41 | } 42 | 43 | body > header img { 44 | position: relative; 45 | top: 3px; 46 | left: 6px; 47 | } 48 | 49 | 50 | 51 | /** CHAT LOG */ 52 | #frame { 53 | display: block; 54 | position: fixed; 55 | 56 | top: 45px; 57 | right: 200px; 58 | bottom: 110px; 59 | left: 20px; 60 | 61 | border-width: 5px 7px 7px 5px; 62 | -webkit-border-image: url(../images/inset-border.png) 5 7 7 5 stretch stretch; 63 | -moz-border-image: url(../images/inset-border.png) 5 7 7 5 stretch stretch; 64 | border-image: url(../images/inset-border.png) 5 7 7 5 stretch stretch; 65 | } 66 | 67 | #chat-log { 68 | position: absolute; 69 | left: 0.5%; 70 | bottom: 0.5%; 71 | width: 99%; 72 | overflow: auto; 73 | max-height: 99%; 74 | color: #ccc; 75 | } 76 | 77 | #chat-log div.chat-msg { 78 | margin-right: 10px; 79 | padding: 6px 70px 6px 30px; 80 | font-size: 10pt; 81 | border-bottom: solid 1px #111; 82 | -webkit-box-shadow: #444 0 1px 0; 83 | -moz-box-shadow: #444 0 1px 0; 84 | box-shadow: #444 0 1px 0; 85 | margin-bottom: 1px; 86 | border-width: 0 0 1px 0; 87 | position: relative; 88 | line-height: 140%; 89 | text-indent: -20px; 90 | } 91 | 92 | #chat-log div.chat-system-msg { 93 | text-align: center; 94 | font-style: italic; 95 | color: #888; 96 | text-shadow: #000 0 -1px 0; 97 | } 98 | 99 | 100 | 101 | .chat-time { 102 | position: absolute; 103 | font-size: 10px; 104 | height: 10pt; 105 | line-height: 12px; 106 | right: 6px; 107 | top: 7px; 108 | color: #111; 109 | font-weight: bold; 110 | font-style: normal; 111 | text-shadow: #4a4a4a 0 1px 0; 112 | } 113 | 114 | #chat-log div.chat-msg .chat-nick { 115 | color: #fff; 116 | margin-right: 10px; 117 | } 118 | 119 | #chat-log div.chat-msg .chat-nick:after { 120 | content: ":"; 121 | } 122 | 123 | #chat-log div.chat-system-msg .chat-nick { 124 | margin-right: 4px; 125 | color: #888; 126 | font-weight: bold; 127 | } 128 | 129 | #chat-log div.chat-system-msg .chat-nick:after { 130 | content: ""; 131 | } 132 | 133 | 134 | #chat-log div.chat-msg:nth-child(2n){ 135 | background: rgba(0,0,0,0.05); 136 | } 137 | 138 | 139 | /** USER LIST */ 140 | 141 | #users { 142 | position: fixed; 143 | width: 155px; 144 | top: 45px; 145 | right: 25px; 146 | } 147 | 148 | #users li { 149 | color: #ccc; 150 | background: url(../images/button.png) no-repeat; 151 | 152 | font-size: 14px; 153 | text-align: center; 154 | text-indent: -1px; 155 | text-shadow: rgba(0,0,0,1) 0 -1px 0, rgba(0,0,0,0.4) 0 0 1px; 156 | line-height: 30px; 157 | 158 | margin: 0 0 5px 0; 159 | width: 155px; 160 | height: 35px; 161 | 162 | position: relative; 163 | cursor: pointer; 164 | } 165 | 166 | #users li:hover { 167 | color: #fff; 168 | background-position: 0 -36px; 169 | } 170 | 171 | 172 | #users li:active { 173 | color: #fff; 174 | line-height: 33px; 175 | text-indent: -1px; 176 | background-position: 0 -72px; 177 | } 178 | 179 | /* Adds the glow */ 180 | #users li:after { 181 | content: ""; 182 | display: block; 183 | width: 57px; 184 | height: 12px; 185 | position: absolute; 186 | left: 50%; 187 | margin-left: -29px; 188 | top: 25px; 189 | background: url(../images/glows.png) no-repeat; 190 | } 191 | 192 | #users li:active:after { 193 | top: 26px; 194 | } 195 | 196 | #users li.green:after { background-position: 0 0;} 197 | #users li.orange:after { background-position: -57px 0;} 198 | #users li.yellow:after { background-position: -114px 0;} 199 | #users li.red:after { background-position: -171px 0;} 200 | #users li.fuschia:after { background-position: -228px 0;} 201 | #users li.blue:after { background-position: -285px 0;} 202 | 203 | /** ENTRY FORM */ 204 | 205 | #entry { 206 | border-top: solid 1px #c2c2c2; 207 | border-bottom: solid 1px #646464; 208 | -webkit-box-shadow: #000 0 -1px 0; 209 | -moz-box-shadow: #000 0 -1px 0; 210 | box-shadow: #000 0 -1px 0; 211 | height: 60px; 212 | position: fixed; 213 | left: 0; 214 | right: 0; 215 | bottom: 26px; 216 | background: #a7a7a7 url(../images/metal.jpg) center -3px repeat-x; 217 | } 218 | 219 | #entry p { 220 | position: fixed; 221 | height: 20px; 222 | left: 27px; 223 | right: 270px; 224 | 225 | width: auto; 226 | bottom: 46px; 227 | 228 | } 229 | 230 | #entry input[type=text]{ 231 | background: transparent; 232 | padding: 0; 233 | outline: none; 234 | border: none; 235 | display: block; 236 | color: #fff; 237 | width: 100%; 238 | height: 100%; 239 | text-shadow: #000 0 1px 0; 240 | font-size: 11pt; 241 | } 242 | 243 | #entry input[type=submit]{ 244 | display: none; 245 | } 246 | 247 | #entry a#submit { 248 | position: absolute; 249 | background: url(../images/send.png) no-repeat; 250 | right: -6px; 251 | top: -4px; 252 | text-align: center; 253 | width: 68px; 254 | height: 27px; 255 | line-height: 25px; 256 | border: none; 257 | padding: 0; 258 | outline: none; 259 | font-size: 14px; 260 | text-indent: -1px; 261 | color: #111; 262 | text-shadow: rgba(255,255,255,0.3) 0 1px 0; 263 | text-decoration: none; 264 | } 265 | 266 | #entry a#submit:hover { 267 | background-position: 0 -27px; 268 | text-shadow: rgba(255,255,255,0.3) 0 1px 0, rgba(255,255,255,0.8) 0 0 6px; 269 | } 270 | 271 | #entry a#submit:active { 272 | background-position: 0 -54px; 273 | line-height: 27px; 274 | text-indent: 1px; 275 | } 276 | 277 | #entry fieldset { 278 | position: absolute; 279 | left: 20px; 280 | top: 14px; 281 | height: 17px; 282 | right: 200px; 283 | border-width: 7px 8px 8px 8px; 284 | -webkit-border-image: url(../images/inset-border-l.png) 7 8 8 8 repeat repeat; 285 | -moz-border-image: url(../images/inset-border-l.png) 7 8 8 8 repeat repeat; 286 | -webkit-border-radius: 6px; 287 | -moz-border-radius: 4px; 288 | border-radius: 4px; 289 | } 290 | 291 | /** FOOTER */ 292 | 293 | footer { 294 | height: 26px; 295 | position: fixed; 296 | left: 0; 297 | right: 0; 298 | bottom: 0; 299 | background: url(../images/footer.png) repeat-x; 300 | } 301 | 302 | footer p.credits { 303 | text-transform: uppercase; 304 | height: 26px; 305 | line-height: 26px; 306 | padding: 0 15px; 307 | font-size: 10px; 308 | text-align: right; 309 | color: #7b7b7b; 310 | text-shadow: #000 0 1px 0, #000 0 0 1px; 311 | } 312 | 313 | footer p.credits a { 314 | color: #aaa; 315 | text-decoration: none; 316 | } 317 | 318 | footer p.credits a:hover { 319 | color: #ccc; 320 | } 321 | 322 | span.pipe { 323 | margin: 0 3px; 324 | } 325 | 326 | 327 | /** LOGIN FORM */ 328 | 329 | #login { 330 | position: fixed; 331 | height: 70px; 332 | left: 0; 333 | right: 0; 334 | margin-top: -75px; 335 | top: 50%; 336 | z-index: 200; 337 | background: #a7a7a7 url(../images/metal.jpg) 0 -3px repeat-x; 338 | border-top: solid 1px #bbb; 339 | border-bottom: solid 1px #666; 340 | -webkit-box-shadow: rgba(0,0,0,0.4) 0 2px 4px; 341 | } 342 | 343 | #login h1 { 344 | text-transform: uppercase; 345 | color: #333; 346 | text-align: center; 347 | padding: 4px 0 0 0; 348 | font-size: 16px; 349 | letter-spacing: 1px; 350 | text-shadow: rgba(255,255,255,0.3) 0 1px 0; 351 | } 352 | 353 | #login p { 354 | position: absolute; 355 | content: ""; 356 | display: block; 357 | height: 35px; 358 | line-height: 35px; 359 | left: 0; 360 | right: 0; 361 | bottom: 10px; 362 | text-align: center; 363 | background: url(../images/footer.png) repeat-x; 364 | } 365 | 366 | #login input { 367 | width: 150px; 368 | margin-left: 5px; 369 | background: #fff; 370 | border: none; 371 | padding: 3px; 372 | -webkit-border-radius: 3px; 373 | -moz-border-radius: 3px; 374 | border-radius: 3px; 375 | } 376 | 377 | #login.error { 378 | -webkit-box-shadow: red 0 0 100px; 379 | -moz-box-shadow: red 0 0 50px; 380 | box-shadow: red 0 0 100px; 381 | } 382 | 383 | #login p label { 384 | color: #ddd; 385 | font-size: 12px; 386 | font-style: italic; 387 | text-shadow: #000 0 1px 0; 388 | } 389 | 390 | body.login header:after { 391 | content: ""; 392 | display: block; 393 | position: fixed; 394 | top: 0; 395 | left: 0; 396 | bottom: 0; 397 | right: 0; 398 | background: url(../images/background.png); 399 | opacity: 0.8; 400 | z-index: 100; 401 | } 402 | 403 | body.login header:before { 404 | content: ""; 405 | display: block; 406 | position: fixed; 407 | top: 0; 408 | left: 0; 409 | bottom: 0; 410 | right: 0; 411 | background: rgba(0,0,0,0.5); 412 | opacity: 0.8; 413 | z-index: 101; 414 | } 415 | 416 | body #login { display: none;} 417 | body.login #login { display: block;} 418 | -------------------------------------------------------------------------------- /demo/web/css/reset.css: -------------------------------------------------------------------------------- 1 | /* 2 | Written by Eric Myer 3 | Source: http://meyerweb.com/eric/thoughts/2007/05/01/reset-reloaded/ 4 | */ 5 | html, body, div, span, applet, object, iframe, 6 | h1, h2, h3, h4, h5, h6, p, blockquote, pre, 7 | a, abbr, acronym, address, big, cite, code, 8 | del, dfn, em, font, img, ins, kbd, q, s, samp, 9 | small, strike, strong, sub, sup, tt, var, 10 | dl, dt, dd, ol, ul, li, 11 | fieldset, form, label, legend, 12 | table, caption, tbody, tfoot, thead, tr, th, td { 13 | margin: 0; 14 | padding: 0; 15 | border: 0; 16 | outline: 0; 17 | font-weight: inherit; 18 | font-style: inherit; 19 | font-size: 100%; 20 | font-family: inherit; 21 | vertical-align: baseline; 22 | } 23 | 24 | /* remember to define focus styles! */ 25 | :focus { 26 | outline: 0; 27 | } 28 | body { 29 | line-height: 1; 30 | color: black; 31 | background: transparent; 32 | } 33 | ol, ul { 34 | list-style: none; 35 | } 36 | 37 | /* tables still need 'cellspacing="0"' in the markup */ 38 | table { 39 | border-collapse: separate; 40 | border-spacing: 0; 41 | } 42 | caption, th, td { 43 | text-align: left; 44 | font-weight: normal; 45 | } 46 | blockquote:before, blockquote:after, 47 | q:before, q:after { 48 | content: ""; 49 | } 50 | blockquote, q { 51 | quotes: "" ""; 52 | } -------------------------------------------------------------------------------- /demo/web/images/background.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/background.png -------------------------------------------------------------------------------- /demo/web/images/button.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/button.png -------------------------------------------------------------------------------- /demo/web/images/footer.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/footer.png -------------------------------------------------------------------------------- /demo/web/images/glows.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/glows.png -------------------------------------------------------------------------------- /demo/web/images/header-bg.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/header-bg.png -------------------------------------------------------------------------------- /demo/web/images/inset-border-l.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/inset-border-l.png -------------------------------------------------------------------------------- /demo/web/images/inset-border.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/inset-border.png -------------------------------------------------------------------------------- /demo/web/images/metal.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/metal.jpg -------------------------------------------------------------------------------- /demo/web/images/node-chat.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/node-chat.png -------------------------------------------------------------------------------- /demo/web/images/send.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/scottgonzalez/node-chat/2cf364335d71fdad93efa433930abe5d9df26cf8/demo/web/images/send.png -------------------------------------------------------------------------------- /demo/web/index.html: -------------------------------------------------------------------------------- 1 | 2 | 3 |
4 | 5 |=0))l||m.push(v);else if(l)h[p]=false;return false},ID:function(g){return g[1].replace(/\\/g,"")},TAG:function(g){return g[1].toLowerCase()}, 80 | CHILD:function(g){if(g[1]==="nth"){var h=/(-?)(\d*)n((?:\+|-)?\d*)/.exec(g[2]==="even"&&"2n"||g[2]==="odd"&&"2n+1"||!/\D/.test(g[2])&&"0n+"+g[2]||g[2]);g[2]=h[1]+(h[2]||1)-0;g[3]=h[3]-0}g[0]=e++;return g},ATTR:function(g,h,l,m,q,p){h=g[1].replace(/\\/g,"");if(!p&&n.attrMap[h])g[1]=n.attrMap[h];if(g[2]==="~=")g[4]=" "+g[4]+" ";return g},PSEUDO:function(g,h,l,m,q){if(g[1]==="not")if((f.exec(g[3])||"").length>1||/^\w/.test(g[3]))g[3]=k(g[3],null,null,h);else{g=k.filter(g[3],h,l,true^q);l||m.push.apply(m, 81 | g);return false}else if(n.match.POS.test(g[0])||n.match.CHILD.test(g[0]))return true;return g},POS:function(g){g.unshift(true);return g}},filters:{enabled:function(g){return g.disabled===false&&g.type!=="hidden"},disabled:function(g){return g.disabled===true},checked:function(g){return g.checked===true},selected:function(g){return g.selected===true},parent:function(g){return!!g.firstChild},empty:function(g){return!g.firstChild},has:function(g,h,l){return!!k(l[3],g).length},header:function(g){return/h\d/i.test(g.nodeName)}, 82 | text:function(g){return"text"===g.type},radio:function(g){return"radio"===g.type},checkbox:function(g){return"checkbox"===g.type},file:function(g){return"file"===g.type},password:function(g){return"password"===g.type},submit:function(g){return"submit"===g.type},image:function(g){return"image"===g.type},reset:function(g){return"reset"===g.type},button:function(g){return"button"===g.type||g.nodeName.toLowerCase()==="button"},input:function(g){return/input|select|textarea|button/i.test(g.nodeName)}}, 83 | setFilters:{first:function(g,h){return h===0},last:function(g,h,l,m){return h===m.length-1},even:function(g,h){return h%2===0},odd:function(g,h){return h%2===1},lt:function(g,h,l){return hl[3]-0},nth:function(g,h,l){return l[3]-0===h},eq:function(g,h,l){return l[3]-0===h}},filter:{PSEUDO:function(g,h,l,m){var q=h[1],p=n.filters[q];if(p)return p(g,l,h,m);else if(q==="contains")return(g.textContent||g.innerText||a([g])||"").indexOf(h[3])>=0;else if(q==="not"){h= 84 | h[3];l=0;for(m=h.length;l =0}},ID:function(g,h){return g.nodeType===1&&g.getAttribute("id")===h},TAG:function(g,h){return h==="*"&&g.nodeType===1||g.nodeName.toLowerCase()===h},CLASS:function(g,h){return(" "+(g.className||g.getAttribute("class"))+" ").indexOf(h)>-1},ATTR:function(g,h){var l=h[1];g=n.attrHandle[l]?n.attrHandle[l](g):g[l]!=null?g[l]:g.getAttribute(l);l=g+"";var m=h[2];h=h[4];return g==null?m==="!=":m=== 86 | "="?l===h:m==="*="?l.indexOf(h)>=0:m==="~="?(" "+l+" ").indexOf(h)>=0:!h?l&&g!==false:m==="!="?l!==h:m==="^="?l.indexOf(h)===0:m==="$="?l.substr(l.length-h.length)===h:m==="|="?l===h||l.substr(0,h.length+1)===h+"-":false},POS:function(g,h,l,m){var q=n.setFilters[h[2]];if(q)return q(g,l,h,m)}}},r=n.match.POS;for(var u in n.match){n.match[u]=new RegExp(n.match[u].source+/(?![^\[]*\])(?![^\(]*\))/.source);n.leftMatch[u]=new RegExp(/(^(?:.|\r|\n)*?)/.source+n.match[u].source.replace(/\\(\d+)/g,function(g, 87 | h){return"\\"+(h-0+1)}))}var z=function(g,h){g=Array.prototype.slice.call(g,0);if(h){h.push.apply(h,g);return h}return g};try{Array.prototype.slice.call(s.documentElement.childNodes,0)}catch(C){z=function(g,h){h=h||[];if(j.call(g)==="[object Array]")Array.prototype.push.apply(h,g);else if(typeof g.length==="number")for(var l=0,m=g.length;l ";var l=s.documentElement;l.insertBefore(g,l.firstChild);if(s.getElementById(h)){n.find.ID=function(m,q,p){if(typeof q.getElementById!=="undefined"&&!p)return(q=q.getElementById(m[1]))?q.id===m[1]||typeof q.getAttributeNode!=="undefined"&& 90 | q.getAttributeNode("id").nodeValue===m[1]?[q]:w:[]};n.filter.ID=function(m,q){var p=typeof m.getAttributeNode!=="undefined"&&m.getAttributeNode("id");return m.nodeType===1&&p&&p.nodeValue===q}}l.removeChild(g);l=g=null})();(function(){var g=s.createElement("div");g.appendChild(s.createComment(""));if(g.getElementsByTagName("*").length>0)n.find.TAG=function(h,l){l=l.getElementsByTagName(h[1]);if(h[1]==="*"){h=[];for(var m=0;l[m];m++)l[m].nodeType===1&&h.push(l[m]);l=h}return l};g.innerHTML=""; 91 | if(g.firstChild&&typeof g.firstChild.getAttribute!=="undefined"&&g.firstChild.getAttribute("href")!=="#")n.attrHandle.href=function(h){return h.getAttribute("href",2)};g=null})();s.querySelectorAll&&function(){var g=k,h=s.createElement("div");h.innerHTML="";if(!(h.querySelectorAll&&h.querySelectorAll(".TEST").length===0)){k=function(m,q,p,v){q=q||s;if(!v&&q.nodeType===9&&!x(q))try{return z(q.querySelectorAll(m),p)}catch(t){}return g(m,q,p,v)};for(var l in g)k[l]=g[l];h=null}}(); 92 | (function(){var g=s.createElement("div");g.innerHTML="";if(!(!g.getElementsByClassName||g.getElementsByClassName("e").length===0)){g.lastChild.className="e";if(g.getElementsByClassName("e").length!==1){n.order.splice(1,0,"CLASS");n.find.CLASS=function(h,l,m){if(typeof l.getElementsByClassName!=="undefined"&&!m)return l.getElementsByClassName(h[1])};g=null}}})();var E=s.compareDocumentPosition?function(g,h){return!!(g.compareDocumentPosition(h)&16)}: 93 | function(g,h){return g!==h&&(g.contains?g.contains(h):true)},x=function(g){return(g=(g?g.ownerDocument||g:0).documentElement)?g.nodeName!=="HTML":false},ga=function(g,h){var l=[],m="",q;for(h=h.nodeType?[h]:h;q=n.match.PSEUDO.exec(g);){m+=q[0];g=g.replace(n.match.PSEUDO,"")}g=n.relative[g]?g+"*":g;q=0;for(var p=h.length;q =0===d})};c.fn.extend({find:function(a){for(var b=this.pushStack("","find",a),d=0,f=0,e=this.length;f
0)for(var j=d;j 0},closest:function(a,b){if(c.isArray(a)){var d=[],f=this[0],e,j= 96 | {},i;if(f&&a.length){e=0;for(var o=a.length;e -1:c(f).is(e)){d.push({selector:i,elem:f});delete j[i]}}f=f.parentNode}}return d}var k=c.expr.match.POS.test(a)?c(a,b||this.context):null;return this.map(function(n,r){for(;r&&r.ownerDocument&&r!==b;){if(k?k.index(r)>-1:c(r).is(a))return r;r=r.parentNode}return null})},index:function(a){if(!a||typeof a=== 97 | "string")return c.inArray(this[0],a?c(a):this.parent().children());return c.inArray(a.jquery?a[0]:a,this)},add:function(a,b){a=typeof a==="string"?c(a,b||this.context):c.makeArray(a);b=c.merge(this.get(),a);return this.pushStack(qa(a[0])||qa(b[0])?b:c.unique(b))},andSelf:function(){return this.add(this.prevObject)}});c.each({parent:function(a){return(a=a.parentNode)&&a.nodeType!==11?a:null},parents:function(a){return c.dir(a,"parentNode")},parentsUntil:function(a,b,d){return c.dir(a,"parentNode", 98 | d)},next:function(a){return c.nth(a,2,"nextSibling")},prev:function(a){return c.nth(a,2,"previousSibling")},nextAll:function(a){return c.dir(a,"nextSibling")},prevAll:function(a){return c.dir(a,"previousSibling")},nextUntil:function(a,b,d){return c.dir(a,"nextSibling",d)},prevUntil:function(a,b,d){return c.dir(a,"previousSibling",d)},siblings:function(a){return c.sibling(a.parentNode.firstChild,a)},children:function(a){return c.sibling(a.firstChild)},contents:function(a){return c.nodeName(a,"iframe")? 99 | a.contentDocument||a.contentWindow.document:c.makeArray(a.childNodes)}},function(a,b){c.fn[a]=function(d,f){var e=c.map(this,b,d);eb.test(a)||(f=d);if(f&&typeof f==="string")e=c.filter(f,e);e=this.length>1?c.unique(e):e;if((this.length>1||gb.test(f))&&fb.test(a))e=e.reverse();return this.pushStack(e,a,R.call(arguments).join(","))}});c.extend({filter:function(a,b,d){if(d)a=":not("+a+")";return c.find.matches(a,b)},dir:function(a,b,d){var f=[];for(a=a[b];a&&a.nodeType!==9&&(d===w||a.nodeType!==1||!c(a).is(d));){a.nodeType=== 100 | 1&&f.push(a);a=a[b]}return f},nth:function(a,b,d){b=b||1;for(var f=0;a;a=a[d])if(a.nodeType===1&&++f===b)break;return a},sibling:function(a,b){for(var d=[];a;a=a.nextSibling)a.nodeType===1&&a!==b&&d.push(a);return d}});var Ja=/ jQuery\d+="(?:\d+|null)"/g,V=/^\s+/,Ka=/(<([\w:]+)[^>]*?)\/>/g,hb=/^(?:area|br|col|embed|hr|img|input|link|meta|param)$/i,La=/<([\w:]+)/,ib=/"+d+">"},F={option:[1,""],legend:[1,""],thead:[1," ","
"],tr:[2,"","
"],td:[3,""],col:[2,"
"," "],area:[1,""],_default:[0,"",""]};F.optgroup=F.option;F.tbody=F.tfoot=F.colgroup=F.caption=F.thead;F.th=F.td;if(!c.support.htmlSerialize)F._default=[1,"div
"," ",""];c.fn.extend({text:function(a){if(c.isFunction(a))return this.each(function(b){var d= 102 | c(this);d.text(a.call(this,b,d.text()))});if(typeof a!=="object"&&a!==w)return this.empty().append((this[0]&&this[0].ownerDocument||s).createTextNode(a));return c.text(this)},wrapAll:function(a){if(c.isFunction(a))return this.each(function(d){c(this).wrapAll(a.call(this,d))});if(this[0]){var b=c(a,this[0].ownerDocument).eq(0).clone(true);this[0].parentNode&&b.insertBefore(this[0]);b.map(function(){for(var d=this;d.firstChild&&d.firstChild.nodeType===1;)d=d.firstChild;return d}).append(this)}return this}, 103 | wrapInner:function(a){if(c.isFunction(a))return this.each(function(b){c(this).wrapInner(a.call(this,b))});return this.each(function(){var b=c(this),d=b.contents();d.length?d.wrapAll(a):b.append(a)})},wrap:function(a){return this.each(function(){c(this).wrapAll(a)})},unwrap:function(){return this.parent().each(function(){c.nodeName(this,"body")||c(this).replaceWith(this.childNodes)}).end()},append:function(){return this.domManip(arguments,true,function(a){this.nodeType===1&&this.appendChild(a)})}, 104 | prepend:function(){return this.domManip(arguments,true,function(a){this.nodeType===1&&this.insertBefore(a,this.firstChild)})},before:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,false,function(b){this.parentNode.insertBefore(b,this)});else if(arguments.length){var a=c(arguments[0]);a.push.apply(a,this.toArray());return this.pushStack(a,"before",arguments)}},after:function(){if(this[0]&&this[0].parentNode)return this.domManip(arguments,false,function(b){this.parentNode.insertBefore(b, 105 | this.nextSibling)});else if(arguments.length){var a=this.pushStack(this,"after",arguments);a.push.apply(a,c(arguments[0]).toArray());return a}},remove:function(a,b){for(var d=0,f;(f=this[d])!=null;d++)if(!a||c.filter(a,[f]).length){if(!b&&f.nodeType===1){c.cleanData(f.getElementsByTagName("*"));c.cleanData([f])}f.parentNode&&f.parentNode.removeChild(f)}return this},empty:function(){for(var a=0,b;(b=this[a])!=null;a++)for(b.nodeType===1&&c.cleanData(b.getElementsByTagName("*"));b.firstChild;)b.removeChild(b.firstChild); 106 | return this},clone:function(a){var b=this.map(function(){if(!c.support.noCloneEvent&&!c.isXMLDoc(this)){var d=this.outerHTML,f=this.ownerDocument;if(!d){d=f.createElement("div");d.appendChild(this.cloneNode(true));d=d.innerHTML}return c.clean([d.replace(Ja,"").replace(/=([^="'>\s]+\/)>/g,'="$1">').replace(V,"")],f)[0]}else return this.cloneNode(true)});if(a===true){ra(this,b);ra(this.find("*"),b.find("*"))}return b},html:function(a){if(a===w)return this[0]&&this[0].nodeType===1?this[0].innerHTML.replace(Ja, 107 | ""):null;else if(typeof a==="string"&&!ta.test(a)&&(c.support.leadingWhitespace||!V.test(a))&&!F[(La.exec(a)||["",""])[1].toLowerCase()]){a=a.replace(Ka,Ma);try{for(var b=0,d=this.length;b0||e.cacheable||this.length>1?k.cloneNode(true):k)}o.length&&c.each(o,Qa)}return this}});c.fragments={};c.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){c.fn[a]=function(d){var f=[];d=c(d);var e=this.length===1&&this[0].parentNode;if(e&&e.nodeType===11&&e.childNodes.length===1&&d.length===1){d[b](this[0]); 111 | return this}else{e=0;for(var j=d.length;e 0?this.clone(true):this).get();c.fn[b].apply(c(d[e]),i);f=f.concat(i)}return this.pushStack(f,a,d.selector)}}});c.extend({clean:function(a,b,d,f){b=b||s;if(typeof b.createElement==="undefined")b=b.ownerDocument||b[0]&&b[0].ownerDocument||s;for(var e=[],j=0,i;(i=a[j])!=null;j++){if(typeof i==="number")i+="";if(i){if(typeof i==="string"&&!jb.test(i))i=b.createTextNode(i);else if(typeof i==="string"){i=i.replace(Ka,Ma);var o=(La.exec(i)||["", 112 | ""])[1].toLowerCase(),k=F[o]||F._default,n=k[0],r=b.createElement("div");for(r.innerHTML=k[1]+i+k[2];n--;)r=r.lastChild;if(!c.support.tbody){n=ib.test(i);o=o==="table"&&!n?r.firstChild&&r.firstChild.childNodes:k[1]===" "&&!n?r.childNodes:[];for(k=o.length-1;k>=0;--k)c.nodeName(o[k],"tbody")&&!o[k].childNodes.length&&o[k].parentNode.removeChild(o[k])}!c.support.leadingWhitespace&&V.test(i)&&r.insertBefore(b.createTextNode(V.exec(i)[0]),r.firstChild);i=r.childNodes}if(i.nodeType)e.push(i);else e= 113 | c.merge(e,i)}}if(d)for(j=0;e[j];j++)if(f&&c.nodeName(e[j],"script")&&(!e[j].type||e[j].type.toLowerCase()==="text/javascript"))f.push(e[j].parentNode?e[j].parentNode.removeChild(e[j]):e[j]);else{e[j].nodeType===1&&e.splice.apply(e,[j+1,0].concat(c.makeArray(e[j].getElementsByTagName("script"))));d.appendChild(e[j])}return e},cleanData:function(a){for(var b,d,f=c.cache,e=c.event.special,j=c.support.deleteExpando,i=0,o;(o=a[i])!=null;i++)if(d=o[c.expando]){b=f[d];if(b.events)for(var k in b.events)e[k]? 114 | c.event.remove(o,k):Ca(o,k,b.handle);if(j)delete o[c.expando];else o.removeAttribute&&o.removeAttribute(c.expando);delete f[d]}}});var kb=/z-?index|font-?weight|opacity|zoom|line-?height/i,Na=/alpha\([^)]*\)/,Oa=/opacity=([^)]*)/,ha=/float/i,ia=/-([a-z])/ig,lb=/([A-Z])/g,mb=/^-?\d+(?:px)?$/i,nb=/^-?\d/,ob={position:"absolute",visibility:"hidden",display:"block"},pb=["Left","Right"],qb=["Top","Bottom"],rb=s.defaultView&&s.defaultView.getComputedStyle,Pa=c.support.cssFloat?"cssFloat":"styleFloat",ja= 115 | function(a,b){return b.toUpperCase()};c.fn.css=function(a,b){return X(this,a,b,true,function(d,f,e){if(e===w)return c.curCSS(d,f);if(typeof e==="number"&&!kb.test(f))e+="px";c.style(d,f,e)})};c.extend({style:function(a,b,d){if(!a||a.nodeType===3||a.nodeType===8)return w;if((b==="width"||b==="height")&&parseFloat(d)<0)d=w;var f=a.style||a,e=d!==w;if(!c.support.opacity&&b==="opacity"){if(e){f.zoom=1;b=parseInt(d,10)+""==="NaN"?"":"alpha(opacity="+d*100+")";a=f.filter||c.curCSS(a,"filter")||"";f.filter= 116 | Na.test(a)?a.replace(Na,b):b}return f.filter&&f.filter.indexOf("opacity=")>=0?parseFloat(Oa.exec(f.filter)[1])/100+"":""}if(ha.test(b))b=Pa;b=b.replace(ia,ja);if(e)f[b]=d;return f[b]},css:function(a,b,d,f){if(b==="width"||b==="height"){var e,j=b==="width"?pb:qb;function i(){e=b==="width"?a.offsetWidth:a.offsetHeight;f!=="border"&&c.each(j,function(){f||(e-=parseFloat(c.curCSS(a,"padding"+this,true))||0);if(f==="margin")e+=parseFloat(c.curCSS(a,"margin"+this,true))||0;else e-=parseFloat(c.curCSS(a, 117 | "border"+this+"Width",true))||0})}a.offsetWidth!==0?i():c.swap(a,ob,i);return Math.max(0,Math.round(e))}return c.curCSS(a,b,d)},curCSS:function(a,b,d){var f,e=a.style;if(!c.support.opacity&&b==="opacity"&&a.currentStyle){f=Oa.test(a.currentStyle.filter||"")?parseFloat(RegExp.$1)/100+"":"";return f===""?"1":f}if(ha.test(b))b=Pa;if(!d&&e&&e[b])f=e[b];else if(rb){if(ha.test(b))b="float";b=b.replace(lb,"-$1").toLowerCase();e=a.ownerDocument.defaultView;if(!e)return null;if(a=e.getComputedStyle(a,null))f= 118 | a.getPropertyValue(b);if(b==="opacity"&&f==="")f="1"}else if(a.currentStyle){d=b.replace(ia,ja);f=a.currentStyle[b]||a.currentStyle[d];if(!mb.test(f)&&nb.test(f)){b=e.left;var j=a.runtimeStyle.left;a.runtimeStyle.left=a.currentStyle.left;e.left=d==="fontSize"?"1em":f||0;f=e.pixelLeft+"px";e.left=b;a.runtimeStyle.left=j}}return f},swap:function(a,b,d){var f={};for(var e in b){f[e]=a.style[e];a.style[e]=b[e]}d.call(a);for(e in b)a.style[e]=f[e]}});if(c.expr&&c.expr.filters){c.expr.filters.hidden=function(a){var b= 119 | a.offsetWidth,d=a.offsetHeight,f=a.nodeName.toLowerCase()==="tr";return b===0&&d===0&&!f?true:b>0&&d>0&&!f?false:c.curCSS(a,"display")==="none"};c.expr.filters.visible=function(a){return!c.expr.filters.hidden(a)}}var sb=J(),tb=/