├── 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 | NodeChat 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 |
Node Chat
14 | 15 |
16 |

Welcome to Node-Chat

17 |

18 | 19 | 20 |

21 |
22 | 23 |
24 | 25 |
26 |
27 |
28 |
29 | 30 | 32 | 33 |
34 |
35 |

36 | 37 |
38 |
39 | 40 | 46 | 47 |
48 | 49 | 50 | -------------------------------------------------------------------------------- /demo/web/js/client.js: -------------------------------------------------------------------------------- 1 | (function($) { 2 | 3 | var title = document.title, 4 | colors = ["green", "orange", "yellow", "red", "fuschia", "blue"], 5 | channel = nodeChat.connect("/chat"), 6 | log, 7 | message; 8 | 9 | // TODO: handle connectionerror 10 | 11 | $(function() { 12 | log = $("#chat-log"); 13 | message = $("#message"); 14 | 15 | // Add a button that can be easily styled 16 | $("", { 17 | id: "submit", 18 | text: "Send", 19 | href: "#", 20 | click: function(event) { 21 | event.preventDefault(); 22 | $(this).closest("form").submit(); 23 | } 24 | }) 25 | .appendTo("#entry fieldset"); 26 | 27 | // Add a message indicator when a nickname is clicked 28 | $("#users").delegate("li", "click", function() { 29 | message 30 | .val($(this).text() + ": " + message.val()) 31 | .focus(); 32 | }); 33 | }); 34 | 35 | // new message posted to channel 36 | // - add to the chat log 37 | $(channel).bind("msg", function(event, message) { 38 | var time = formatTime(message.timestamp), 39 | row = $("
") 40 | .addClass("chat-msg"); 41 | 42 | $("") 43 | .addClass("chat-time") 44 | .text(time) 45 | .appendTo(row); 46 | 47 | $("") 48 | .addClass("chat-nick") 49 | .text(message.nick) 50 | .appendTo(row); 51 | 52 | $("") 53 | .addClass("chat-text") 54 | .text(message.text) 55 | .appendTo(row); 56 | 57 | row.appendTo(log); 58 | }) 59 | // another user joined the channel 60 | // - add to the chat log 61 | .bind("join", function(event, message) { 62 | var time = formatTime(message.timestamp), 63 | row = $("
") 64 | .addClass("chat-msg chat-system-msg"); 65 | 66 | $("") 67 | .addClass("chat-time") 68 | .text(time) 69 | .appendTo(row); 70 | 71 | $("") 72 | .addClass("chat-nick") 73 | .text(message.nick) 74 | .appendTo(row); 75 | 76 | $("") 77 | .addClass("chat-text") 78 | .text("joined the room") 79 | .appendTo(row); 80 | 81 | row.appendTo(log); 82 | }) 83 | // another user joined the channel 84 | // - add to the user list 85 | .bind("join", function(event, message) { 86 | var added = false, 87 | nick = $("
  • ", { 88 | "class": colors[0], 89 | text: message.nick 90 | }); 91 | colors.push(colors.shift()); 92 | $("#users > li").each(function() { 93 | if (message.nick == this.innerHTML) { 94 | added = true; 95 | return false; 96 | } 97 | if (message.nick < this.innerHTML) { 98 | added = true; 99 | nick.insertBefore(this); 100 | return false; 101 | } 102 | }); 103 | if (!added) { 104 | $("#users").append(nick); 105 | } 106 | }) 107 | // another user left the channel 108 | // - add to the chat log 109 | .bind("part", function(event, message) { 110 | var time = formatTime(message.timestamp), 111 | row = $("
    ") 112 | .addClass("chat-msg chat-system-msg"); 113 | 114 | $("") 115 | .addClass("chat-time") 116 | .text(time) 117 | .appendTo(row); 118 | 119 | $("") 120 | .addClass("chat-nick") 121 | .text(message.nick) 122 | .appendTo(row); 123 | 124 | $("") 125 | .addClass("chat-text") 126 | .text("left the room") 127 | .appendTo(row); 128 | 129 | row.appendTo(log); 130 | }) 131 | // another user left the channel 132 | // - remove from the user list 133 | .bind("part", function(event, message) { 134 | $("#users > li").each(function() { 135 | if (this.innerHTML == message.nick) { 136 | $(this).remove(); 137 | return false; 138 | } 139 | }); 140 | }) 141 | 142 | // Auto scroll list to bottom 143 | .bind("join part msg", function() { 144 | // auto scroll if we're within 50 pixels of the bottom 145 | if (log.scrollTop() + 50 >= log[0].scrollHeight - log.height()) { 146 | window.setTimeout(function() { 147 | log.scrollTop(log[0].scrollHeight); 148 | }, 10); 149 | } 150 | }); 151 | 152 | // handle login (choosing a nick) 153 | $(function() { 154 | function loginError(error) { 155 | login 156 | .addClass("error") 157 | .find("label") 158 | .text(error + " Please choose another:") 159 | .end() 160 | .find("input") 161 | .focus(); 162 | } 163 | 164 | var login = $("#login"); 165 | login.submit(function() { 166 | var nick = $.trim($("#nick").val()); 167 | 168 | // TODO: move the check into nodechat.js 169 | if (!nick.length || !/^[a-zA-Z\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+$/.test(nick)) { 170 | loginError("Invalid Nickname."); 171 | return false; 172 | } 173 | 174 | channel.join(nick, { 175 | success: function() { 176 | $("body") 177 | .removeClass("login") 178 | .addClass("channel"); 179 | message.focus(); 180 | }, 181 | error: function() { 182 | loginError("Nickname in use."); 183 | } 184 | }); 185 | 186 | return false; 187 | }); 188 | login.find("input").focus(); 189 | }); 190 | 191 | // handle sending a message 192 | $(function() { 193 | $("#channel form").submit(function() { 194 | channel.send(message.val()); 195 | message.val("").focus(); 196 | 197 | return false; 198 | }); 199 | }); 200 | 201 | // update the page title to show if there are unread messages 202 | $(function() { 203 | var focused = true, 204 | unread = 0; 205 | 206 | $(window) 207 | .blur(function() { 208 | focused = false; 209 | }) 210 | .focus(function() { 211 | focused = true; 212 | unread = 0; 213 | document.title = title; 214 | }); 215 | 216 | $(channel).bind("msg", function(event, message) { 217 | if (!focused) { 218 | unread++; 219 | document.title = "(" + unread + ") " + title; 220 | } 221 | }); 222 | }); 223 | 224 | // notify the chat server that we're leaving if we close the window 225 | $(window).unload(function() { 226 | channel.part(); 227 | }); 228 | 229 | function formatTime(timestamp) { 230 | var date = new Date(timestamp), 231 | hours = date.getHours(), 232 | minutes = date.getMinutes(), 233 | ampm = "AM"; 234 | 235 | if (hours > 12) { 236 | hours -= 12; 237 | ampm = "PM"; 238 | } 239 | 240 | if (minutes < 10) { 241 | minutes = "0" + minutes; 242 | } 243 | 244 | return hours + ":" + minutes + " " + ampm; 245 | } 246 | 247 | })(jQuery); 248 | -------------------------------------------------------------------------------- /demo/web/js/jquery-1.4.2.js: -------------------------------------------------------------------------------- 1 | /*! 2 | * jQuery JavaScript Library v1.4.2 3 | * http://jquery.com/ 4 | * 5 | * Copyright 2010, John Resig 6 | * Dual licensed under the MIT or GPL Version 2 licenses. 7 | * http://jquery.org/license 8 | * 9 | * Includes Sizzle.js 10 | * http://sizzlejs.com/ 11 | * Copyright 2010, The Dojo Foundation 12 | * Released under the MIT, BSD, and GPL Licenses. 13 | * 14 | * Date: Sat Feb 13 22:33:48 2010 -0500 15 | */ 16 | (function(A,w){function ma(){if(!c.isReady){try{s.documentElement.doScroll("left")}catch(a){setTimeout(ma,1);return}c.ready()}}function Qa(a,b){b.src?c.ajax({url:b.src,async:false,dataType:"script"}):c.globalEval(b.text||b.textContent||b.innerHTML||"");b.parentNode&&b.parentNode.removeChild(b)}function X(a,b,d,f,e,j){var i=a.length;if(typeof b==="object"){for(var o in b)X(a,o,b[o],f,e,d);return a}if(d!==w){f=!j&&f&&c.isFunction(d);for(o=0;o)[^>]*$|^#([\w-]+)$/,Ua=/^.[^:#\[\.,]*$/,Va=/\S/, 21 | Wa=/^(\s|\u00A0)+|(\s|\u00A0)+$/g,Xa=/^<(\w+)\s*\/?>(?:<\/\1>)?$/,P=navigator.userAgent,xa=false,Q=[],L,$=Object.prototype.toString,aa=Object.prototype.hasOwnProperty,ba=Array.prototype.push,R=Array.prototype.slice,ya=Array.prototype.indexOf;c.fn=c.prototype={init:function(a,b){var d,f;if(!a)return this;if(a.nodeType){this.context=this[0]=a;this.length=1;return this}if(a==="body"&&!b){this.context=s;this[0]=s.body;this.selector="body";this.length=1;return this}if(typeof a==="string")if((d=Ta.exec(a))&& 22 | (d[1]||!b))if(d[1]){f=b?b.ownerDocument||b:s;if(a=Xa.exec(a))if(c.isPlainObject(b)){a=[s.createElement(a[1])];c.fn.attr.call(a,b,true)}else a=[f.createElement(a[1])];else{a=sa([d[1]],[f]);a=(a.cacheable?a.fragment.cloneNode(true):a.fragment).childNodes}return c.merge(this,a)}else{if(b=s.getElementById(d[2])){if(b.id!==d[2])return T.find(a);this.length=1;this[0]=b}this.context=s;this.selector=a;return this}else if(!b&&/^\w+$/.test(a)){this.selector=a;this.context=s;a=s.getElementsByTagName(a);return c.merge(this, 23 | a)}else return!b||b.jquery?(b||T).find(a):c(b).find(a);else if(c.isFunction(a))return T.ready(a);if(a.selector!==w){this.selector=a.selector;this.context=a.context}return c.makeArray(a,this)},selector:"",jquery:"1.4.2",length:0,size:function(){return this.length},toArray:function(){return R.call(this,0)},get:function(a){return a==null?this.toArray():a<0?this.slice(a)[0]:this[a]},pushStack:function(a,b,d){var f=c();c.isArray(a)?ba.apply(f,a):c.merge(f,a);f.prevObject=this;f.context=this.context;if(b=== 24 | "find")f.selector=this.selector+(this.selector?" ":"")+d;else if(b)f.selector=this.selector+"."+b+"("+d+")";return f},each:function(a,b){return c.each(this,a,b)},ready:function(a){c.bindReady();if(c.isReady)a.call(s,c);else Q&&Q.push(a);return this},eq:function(a){return a===-1?this.slice(a):this.slice(a,+a+1)},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},slice:function(){return this.pushStack(R.apply(this,arguments),"slice",R.call(arguments).join(","))},map:function(a){return this.pushStack(c.map(this, 25 | function(b,d){return a.call(b,d,b)}))},end:function(){return this.prevObject||c(null)},push:ba,sort:[].sort,splice:[].splice};c.fn.init.prototype=c.fn;c.extend=c.fn.extend=function(){var a=arguments[0]||{},b=1,d=arguments.length,f=false,e,j,i,o;if(typeof a==="boolean"){f=a;a=arguments[1]||{};b=2}if(typeof a!=="object"&&!c.isFunction(a))a={};if(d===b){a=this;--b}for(;b
    a"; 34 | var e=d.getElementsByTagName("*"),j=d.getElementsByTagName("a")[0];if(!(!e||!e.length||!j)){c.support={leadingWhitespace:d.firstChild.nodeType===3,tbody:!d.getElementsByTagName("tbody").length,htmlSerialize:!!d.getElementsByTagName("link").length,style:/red/.test(j.getAttribute("style")),hrefNormalized:j.getAttribute("href")==="/a",opacity:/^0.55$/.test(j.style.opacity),cssFloat:!!j.style.cssFloat,checkOn:d.getElementsByTagName("input")[0].value==="on",optSelected:s.createElement("select").appendChild(s.createElement("option")).selected, 35 | parentNode:d.removeChild(d.appendChild(s.createElement("div"))).parentNode===null,deleteExpando:true,checkClone:false,scriptEval:false,noCloneEvent:true,boxModel:null};b.type="text/javascript";try{b.appendChild(s.createTextNode("window."+f+"=1;"))}catch(i){}a.insertBefore(b,a.firstChild);if(A[f]){c.support.scriptEval=true;delete A[f]}try{delete b.test}catch(o){c.support.deleteExpando=false}a.removeChild(b);if(d.attachEvent&&d.fireEvent){d.attachEvent("onclick",function k(){c.support.noCloneEvent= 36 | false;d.detachEvent("onclick",k)});d.cloneNode(true).fireEvent("onclick")}d=s.createElement("div");d.innerHTML="";a=s.createDocumentFragment();a.appendChild(d.firstChild);c.support.checkClone=a.cloneNode(true).cloneNode(true).lastChild.checked;c(function(){var k=s.createElement("div");k.style.width=k.style.paddingLeft="1px";s.body.appendChild(k);c.boxModel=c.support.boxModel=k.offsetWidth===2;s.body.removeChild(k).style.display="none"});a=function(k){var n= 37 | s.createElement("div");k="on"+k;var r=k in n;if(!r){n.setAttribute(k,"return;");r=typeof n[k]==="function"}return r};c.support.submitBubbles=a("submit");c.support.changeBubbles=a("change");a=b=d=e=j=null}})();c.props={"for":"htmlFor","class":"className",readonly:"readOnly",maxlength:"maxLength",cellspacing:"cellSpacing",rowspan:"rowSpan",colspan:"colSpan",tabindex:"tabIndex",usemap:"useMap",frameborder:"frameBorder"};var G="jQuery"+J(),Ya=0,za={};c.extend({cache:{},expando:G,noData:{embed:true,object:true, 38 | applet:true},data:function(a,b,d){if(!(a.nodeName&&c.noData[a.nodeName.toLowerCase()])){a=a==A?za:a;var f=a[G],e=c.cache;if(!f&&typeof b==="string"&&d===w)return null;f||(f=++Ya);if(typeof b==="object"){a[G]=f;e[f]=c.extend(true,{},b)}else if(!e[f]){a[G]=f;e[f]={}}a=e[f];if(d!==w)a[b]=d;return typeof b==="string"?a[b]:a}},removeData:function(a,b){if(!(a.nodeName&&c.noData[a.nodeName.toLowerCase()])){a=a==A?za:a;var d=a[G],f=c.cache,e=f[d];if(b){if(e){delete e[b];c.isEmptyObject(e)&&c.removeData(a)}}else{if(c.support.deleteExpando)delete a[c.expando]; 39 | else a.removeAttribute&&a.removeAttribute(c.expando);delete f[d]}}}});c.fn.extend({data:function(a,b){if(typeof a==="undefined"&&this.length)return c.data(this[0]);else if(typeof a==="object")return this.each(function(){c.data(this,a)});var d=a.split(".");d[1]=d[1]?"."+d[1]:"";if(b===w){var f=this.triggerHandler("getData"+d[1]+"!",[d[0]]);if(f===w&&this.length)f=c.data(this[0],a);return f===w&&d[1]?this.data(d[0]):f}else return this.trigger("setData"+d[1]+"!",[d[0],b]).each(function(){c.data(this, 40 | a,b)})},removeData:function(a){return this.each(function(){c.removeData(this,a)})}});c.extend({queue:function(a,b,d){if(a){b=(b||"fx")+"queue";var f=c.data(a,b);if(!d)return f||[];if(!f||c.isArray(d))f=c.data(a,b,c.makeArray(d));else f.push(d);return f}},dequeue:function(a,b){b=b||"fx";var d=c.queue(a,b),f=d.shift();if(f==="inprogress")f=d.shift();if(f){b==="fx"&&d.unshift("inprogress");f.call(a,function(){c.dequeue(a,b)})}}});c.fn.extend({queue:function(a,b){if(typeof a!=="string"){b=a;a="fx"}if(b=== 41 | w)return c.queue(this[0],a);return this.each(function(){var d=c.queue(this,a,b);a==="fx"&&d[0]!=="inprogress"&&c.dequeue(this,a)})},dequeue:function(a){return this.each(function(){c.dequeue(this,a)})},delay:function(a,b){a=c.fx?c.fx.speeds[a]||a:a;b=b||"fx";return this.queue(b,function(){var d=this;setTimeout(function(){c.dequeue(d,b)},a)})},clearQueue:function(a){return this.queue(a||"fx",[])}});var Aa=/[\n\t]/g,ca=/\s+/,Za=/\r/g,$a=/href|src|style/,ab=/(button|input)/i,bb=/(button|input|object|select|textarea)/i, 42 | cb=/^(a|area)$/i,Ba=/radio|checkbox/;c.fn.extend({attr:function(a,b){return X(this,a,b,true,c.attr)},removeAttr:function(a){return this.each(function(){c.attr(this,a,"");this.nodeType===1&&this.removeAttribute(a)})},addClass:function(a){if(c.isFunction(a))return this.each(function(n){var r=c(this);r.addClass(a.call(this,n,r.attr("class")))});if(a&&typeof a==="string")for(var b=(a||"").split(ca),d=0,f=this.length;d-1)return true;return false},val:function(a){if(a===w){var b=this[0];if(b){if(c.nodeName(b,"option"))return(b.attributes.value||{}).specified?b.value:b.text;if(c.nodeName(b,"select")){var d=b.selectedIndex,f=[],e=b.options;b=b.type==="select-one";if(d<0)return null;var j=b?d:0;for(d=b?d+1:e.length;j=0;else if(c.nodeName(this,"select")){var u=c.makeArray(r);c("option",this).each(function(){this.selected= 47 | c.inArray(c(this).val(),u)>=0});if(!u.length)this.selectedIndex=-1}else this.value=r}})}});c.extend({attrFn:{val:true,css:true,html:true,text:true,data:true,width:true,height:true,offset:true},attr:function(a,b,d,f){if(!a||a.nodeType===3||a.nodeType===8)return w;if(f&&b in c.attrFn)return c(a)[b](d);f=a.nodeType!==1||!c.isXMLDoc(a);var e=d!==w;b=f&&c.props[b]||b;if(a.nodeType===1){var j=$a.test(b);if(b in a&&f&&!j){if(e){b==="type"&&ab.test(a.nodeName)&&a.parentNode&&c.error("type property can't be changed"); 48 | a[b]=d}if(c.nodeName(a,"form")&&a.getAttributeNode(b))return a.getAttributeNode(b).nodeValue;if(b==="tabIndex")return(b=a.getAttributeNode("tabIndex"))&&b.specified?b.value:bb.test(a.nodeName)||cb.test(a.nodeName)&&a.href?0:w;return a[b]}if(!c.support.style&&f&&b==="style"){if(e)a.style.cssText=""+d;return a.style.cssText}e&&a.setAttribute(b,""+d);a=!c.support.hrefNormalized&&f&&j?a.getAttribute(b,2):a.getAttribute(b);return a===null?w:a}return c.style(a,b,d)}});var O=/\.(.*)$/,db=function(a){return a.replace(/[^\w\s\.\|`]/g, 49 | function(b){return"\\"+b})};c.event={add:function(a,b,d,f){if(!(a.nodeType===3||a.nodeType===8)){if(a.setInterval&&a!==A&&!a.frameElement)a=A;var e,j;if(d.handler){e=d;d=e.handler}if(!d.guid)d.guid=c.guid++;if(j=c.data(a)){var i=j.events=j.events||{},o=j.handle;if(!o)j.handle=o=function(){return typeof c!=="undefined"&&!c.event.triggered?c.event.handle.apply(o.elem,arguments):w};o.elem=a;b=b.split(" ");for(var k,n=0,r;k=b[n++];){j=e?c.extend({},e):{handler:d,data:f};if(k.indexOf(".")>-1){r=k.split("."); 50 | k=r.shift();j.namespace=r.slice(0).sort().join(".")}else{r=[];j.namespace=""}j.type=k;j.guid=d.guid;var u=i[k],z=c.event.special[k]||{};if(!u){u=i[k]=[];if(!z.setup||z.setup.call(a,f,r,o)===false)if(a.addEventListener)a.addEventListener(k,o,false);else a.attachEvent&&a.attachEvent("on"+k,o)}if(z.add){z.add.call(a,j);if(!j.handler.guid)j.handler.guid=d.guid}u.push(j);c.event.global[k]=true}a=null}}},global:{},remove:function(a,b,d,f){if(!(a.nodeType===3||a.nodeType===8)){var e,j=0,i,o,k,n,r,u,z=c.data(a), 51 | C=z&&z.events;if(z&&C){if(b&&b.type){d=b.handler;b=b.type}if(!b||typeof b==="string"&&b.charAt(0)==="."){b=b||"";for(e in C)c.event.remove(a,e+b)}else{for(b=b.split(" ");e=b[j++];){n=e;i=e.indexOf(".")<0;o=[];if(!i){o=e.split(".");e=o.shift();k=new RegExp("(^|\\.)"+c.map(o.slice(0).sort(),db).join("\\.(?:.*\\.)?")+"(\\.|$)")}if(r=C[e])if(d){n=c.event.special[e]||{};for(B=f||0;B=0){a.type= 53 | e=e.slice(0,-1);a.exclusive=true}if(!d){a.stopPropagation();c.event.global[e]&&c.each(c.cache,function(){this.events&&this.events[e]&&c.event.trigger(a,b,this.handle.elem)})}if(!d||d.nodeType===3||d.nodeType===8)return w;a.result=w;a.target=d;b=c.makeArray(b);b.unshift(a)}a.currentTarget=d;(f=c.data(d,"handle"))&&f.apply(d,b);f=d.parentNode||d.ownerDocument;try{if(!(d&&d.nodeName&&c.noData[d.nodeName.toLowerCase()]))if(d["on"+e]&&d["on"+e].apply(d,b)===false)a.result=false}catch(j){}if(!a.isPropagationStopped()&& 54 | f)c.event.trigger(a,b,f,true);else if(!a.isDefaultPrevented()){f=a.target;var i,o=c.nodeName(f,"a")&&e==="click",k=c.event.special[e]||{};if((!k._default||k._default.call(d,a)===false)&&!o&&!(f&&f.nodeName&&c.noData[f.nodeName.toLowerCase()])){try{if(f[e]){if(i=f["on"+e])f["on"+e]=null;c.event.triggered=true;f[e]()}}catch(n){}if(i)f["on"+e]=i;c.event.triggered=false}}},handle:function(a){var b,d,f,e;a=arguments[0]=c.event.fix(a||A.event);a.currentTarget=this;b=a.type.indexOf(".")<0&&!a.exclusive; 55 | if(!b){d=a.type.split(".");a.type=d.shift();f=new RegExp("(^|\\.)"+d.slice(0).sort().join("\\.(?:.*\\.)?")+"(\\.|$)")}e=c.data(this,"events");d=e[a.type];if(e&&d){d=d.slice(0);e=0;for(var j=d.length;e-1?c.map(a.options,function(f){return f.selected}).join("-"):"";else if(a.nodeName.toLowerCase()==="select")d=a.selectedIndex;return d},fa=function(a,b){var d=a.target,f,e;if(!(!da.test(d.nodeName)||d.readOnly)){f=c.data(d,"_change_data");e=Fa(d);if(a.type!=="focusout"||d.type!=="radio")c.data(d,"_change_data", 63 | e);if(!(f===w||e===f))if(f!=null||e){a.type="change";return c.event.trigger(a,b,d)}}};c.event.special.change={filters:{focusout:fa,click:function(a){var b=a.target,d=b.type;if(d==="radio"||d==="checkbox"||b.nodeName.toLowerCase()==="select")return fa.call(this,a)},keydown:function(a){var b=a.target,d=b.type;if(a.keyCode===13&&b.nodeName.toLowerCase()!=="textarea"||a.keyCode===32&&(d==="checkbox"||d==="radio")||d==="select-multiple")return fa.call(this,a)},beforeactivate:function(a){a=a.target;c.data(a, 64 | "_change_data",Fa(a))}},setup:function(){if(this.type==="file")return false;for(var a in ea)c.event.add(this,a+".specialChange",ea[a]);return da.test(this.nodeName)},teardown:function(){c.event.remove(this,".specialChange");return da.test(this.nodeName)}};ea=c.event.special.change.filters}s.addEventListener&&c.each({focus:"focusin",blur:"focusout"},function(a,b){function d(f){f=c.event.fix(f);f.type=b;return c.event.handle.call(this,f)}c.event.special[b]={setup:function(){this.addEventListener(a, 65 | d,true)},teardown:function(){this.removeEventListener(a,d,true)}}});c.each(["bind","one"],function(a,b){c.fn[b]=function(d,f,e){if(typeof d==="object"){for(var j in d)this[b](j,f,d[j],e);return this}if(c.isFunction(f)){e=f;f=w}var i=b==="one"?c.proxy(e,function(k){c(this).unbind(k,i);return e.apply(this,arguments)}):e;if(d==="unload"&&b!=="one")this.one(d,f,e);else{j=0;for(var o=this.length;j0){y=t;break}}t=t[g]}m[q]=y}}}var f=/((?:\((?:\([^()]+\)|[^()]+)+\)|\[(?:\[[^[\]]*\]|['"][^'"]*['"]|[^[\]'"]+)+\]|\\.|[^ >+~,(\[\\]+)+|[>+~])(\s*,\s*)?((?:.|\r|\n)*)/g, 71 | e=0,j=Object.prototype.toString,i=false,o=true;[0,0].sort(function(){o=false;return 0});var k=function(g,h,l,m){l=l||[];var q=h=h||s;if(h.nodeType!==1&&h.nodeType!==9)return[];if(!g||typeof g!=="string")return l;for(var p=[],v,t,y,S,H=true,M=x(h),I=g;(f.exec(""),v=f.exec(I))!==null;){I=v[3];p.push(v[1]);if(v[2]){S=v[3];break}}if(p.length>1&&r.exec(g))if(p.length===2&&n.relative[p[0]])t=ga(p[0]+p[1],h);else for(t=n.relative[p[0]]?[h]:k(p.shift(),h);p.length;){g=p.shift();if(n.relative[g])g+=p.shift(); 72 | t=ga(g,t)}else{if(!m&&p.length>1&&h.nodeType===9&&!M&&n.match.ID.test(p[0])&&!n.match.ID.test(p[p.length-1])){v=k.find(p.shift(),h,M);h=v.expr?k.filter(v.expr,v.set)[0]:v.set[0]}if(h){v=m?{expr:p.pop(),set:z(m)}:k.find(p.pop(),p.length===1&&(p[0]==="~"||p[0]==="+")&&h.parentNode?h.parentNode:h,M);t=v.expr?k.filter(v.expr,v.set):v.set;if(p.length>0)y=z(t);else H=false;for(;p.length;){var D=p.pop();v=D;if(n.relative[D])v=p.pop();else D="";if(v==null)v=h;n.relative[D](y,v,M)}}else y=[]}y||(y=t);y||k.error(D|| 73 | g);if(j.call(y)==="[object Array]")if(H)if(h&&h.nodeType===1)for(g=0;y[g]!=null;g++){if(y[g]&&(y[g]===true||y[g].nodeType===1&&E(h,y[g])))l.push(t[g])}else for(g=0;y[g]!=null;g++)y[g]&&y[g].nodeType===1&&l.push(t[g]);else l.push.apply(l,y);else z(y,l);if(S){k(S,q,l,m);k.uniqueSort(l)}return l};k.uniqueSort=function(g){if(B){i=o;g.sort(B);if(i)for(var h=1;h":function(g,h){var l=typeof h==="string";if(l&&!/\W/.test(h)){h=h.toLowerCase();for(var m=0,q=g.length;m=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;f0)for(var j=d;j0},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=/"},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;e0?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=//gi,ub=/select|textarea/i,vb=/color|date|datetime|email|hidden|month|number|password|range|search|tel|text|time|url|week/i,N=/=\?(&|$)/,ka=/\?/,wb=/(\?|&)_=.*?(&|$)/,xb=/^(\w+:)?\/\/([^\/?#]+)/,yb=/%20/g,zb=c.fn.load;c.fn.extend({load:function(a,b,d){if(typeof a!== 120 | "string")return zb.call(this,a);else if(!this.length)return this;var f=a.indexOf(" ");if(f>=0){var e=a.slice(f,a.length);a=a.slice(0,f)}f="GET";if(b)if(c.isFunction(b)){d=b;b=null}else if(typeof b==="object"){b=c.param(b,c.ajaxSettings.traditional);f="POST"}var j=this;c.ajax({url:a,type:f,dataType:"html",data:b,complete:function(i,o){if(o==="success"||o==="notmodified")j.html(e?c("
    ").append(i.responseText.replace(tb,"")).find(e):i.responseText);d&&j.each(d,[i.responseText,o,i])}});return this}, 121 | serialize:function(){return c.param(this.serializeArray())},serializeArray:function(){return this.map(function(){return this.elements?c.makeArray(this.elements):this}).filter(function(){return this.name&&!this.disabled&&(this.checked||ub.test(this.nodeName)||vb.test(this.type))}).map(function(a,b){a=c(this).val();return a==null?null:c.isArray(a)?c.map(a,function(d){return{name:b.name,value:d}}):{name:b.name,value:a}}).get()}});c.each("ajaxStart ajaxStop ajaxComplete ajaxError ajaxSuccess ajaxSend".split(" "), 122 | function(a,b){c.fn[b]=function(d){return this.bind(b,d)}});c.extend({get:function(a,b,d,f){if(c.isFunction(b)){f=f||d;d=b;b=null}return c.ajax({type:"GET",url:a,data:b,success:d,dataType:f})},getScript:function(a,b){return c.get(a,null,b,"script")},getJSON:function(a,b,d){return c.get(a,b,d,"json")},post:function(a,b,d,f){if(c.isFunction(b)){f=f||d;d=b;b={}}return c.ajax({type:"POST",url:a,data:b,success:d,dataType:f})},ajaxSetup:function(a){c.extend(c.ajaxSettings,a)},ajaxSettings:{url:location.href, 123 | global:true,type:"GET",contentType:"application/x-www-form-urlencoded",processData:true,async:true,xhr:A.XMLHttpRequest&&(A.location.protocol!=="file:"||!A.ActiveXObject)?function(){return new A.XMLHttpRequest}:function(){try{return new A.ActiveXObject("Microsoft.XMLHTTP")}catch(a){}},accepts:{xml:"application/xml, text/xml",html:"text/html",script:"text/javascript, application/javascript",json:"application/json, text/javascript",text:"text/plain",_default:"*/*"}},lastModified:{},etag:{},ajax:function(a){function b(){e.success&& 124 | e.success.call(k,o,i,x);e.global&&f("ajaxSuccess",[x,e])}function d(){e.complete&&e.complete.call(k,x,i);e.global&&f("ajaxComplete",[x,e]);e.global&&!--c.active&&c.event.trigger("ajaxStop")}function f(q,p){(e.context?c(e.context):c.event).trigger(q,p)}var e=c.extend(true,{},c.ajaxSettings,a),j,i,o,k=a&&a.context||e,n=e.type.toUpperCase();if(e.data&&e.processData&&typeof e.data!=="string")e.data=c.param(e.data,e.traditional);if(e.dataType==="jsonp"){if(n==="GET")N.test(e.url)||(e.url+=(ka.test(e.url)? 125 | "&":"?")+(e.jsonp||"callback")+"=?");else if(!e.data||!N.test(e.data))e.data=(e.data?e.data+"&":"")+(e.jsonp||"callback")+"=?";e.dataType="json"}if(e.dataType==="json"&&(e.data&&N.test(e.data)||N.test(e.url))){j=e.jsonpCallback||"jsonp"+sb++;if(e.data)e.data=(e.data+"").replace(N,"="+j+"$1");e.url=e.url.replace(N,"="+j+"$1");e.dataType="script";A[j]=A[j]||function(q){o=q;b();d();A[j]=w;try{delete A[j]}catch(p){}z&&z.removeChild(C)}}if(e.dataType==="script"&&e.cache===null)e.cache=false;if(e.cache=== 126 | false&&n==="GET"){var r=J(),u=e.url.replace(wb,"$1_="+r+"$2");e.url=u+(u===e.url?(ka.test(e.url)?"&":"?")+"_="+r:"")}if(e.data&&n==="GET")e.url+=(ka.test(e.url)?"&":"?")+e.data;e.global&&!c.active++&&c.event.trigger("ajaxStart");r=(r=xb.exec(e.url))&&(r[1]&&r[1]!==location.protocol||r[2]!==location.host);if(e.dataType==="script"&&n==="GET"&&r){var z=s.getElementsByTagName("head")[0]||s.documentElement,C=s.createElement("script");C.src=e.url;if(e.scriptCharset)C.charset=e.scriptCharset;if(!j){var B= 127 | false;C.onload=C.onreadystatechange=function(){if(!B&&(!this.readyState||this.readyState==="loaded"||this.readyState==="complete")){B=true;b();d();C.onload=C.onreadystatechange=null;z&&C.parentNode&&z.removeChild(C)}}}z.insertBefore(C,z.firstChild);return w}var E=false,x=e.xhr();if(x){e.username?x.open(n,e.url,e.async,e.username,e.password):x.open(n,e.url,e.async);try{if(e.data||a&&a.contentType)x.setRequestHeader("Content-Type",e.contentType);if(e.ifModified){c.lastModified[e.url]&&x.setRequestHeader("If-Modified-Since", 128 | c.lastModified[e.url]);c.etag[e.url]&&x.setRequestHeader("If-None-Match",c.etag[e.url])}r||x.setRequestHeader("X-Requested-With","XMLHttpRequest");x.setRequestHeader("Accept",e.dataType&&e.accepts[e.dataType]?e.accepts[e.dataType]+", */*":e.accepts._default)}catch(ga){}if(e.beforeSend&&e.beforeSend.call(k,x,e)===false){e.global&&!--c.active&&c.event.trigger("ajaxStop");x.abort();return false}e.global&&f("ajaxSend",[x,e]);var g=x.onreadystatechange=function(q){if(!x||x.readyState===0||q==="abort"){E|| 129 | d();E=true;if(x)x.onreadystatechange=c.noop}else if(!E&&x&&(x.readyState===4||q==="timeout")){E=true;x.onreadystatechange=c.noop;i=q==="timeout"?"timeout":!c.httpSuccess(x)?"error":e.ifModified&&c.httpNotModified(x,e.url)?"notmodified":"success";var p;if(i==="success")try{o=c.httpData(x,e.dataType,e)}catch(v){i="parsererror";p=v}if(i==="success"||i==="notmodified")j||b();else c.handleError(e,x,i,p);d();q==="timeout"&&x.abort();if(e.async)x=null}};try{var h=x.abort;x.abort=function(){x&&h.call(x); 130 | g("abort")}}catch(l){}e.async&&e.timeout>0&&setTimeout(function(){x&&!E&&g("timeout")},e.timeout);try{x.send(n==="POST"||n==="PUT"||n==="DELETE"?e.data:null)}catch(m){c.handleError(e,x,null,m);d()}e.async||g();return x}},handleError:function(a,b,d,f){if(a.error)a.error.call(a.context||a,b,d,f);if(a.global)(a.context?c(a.context):c.event).trigger("ajaxError",[b,a,f])},active:0,httpSuccess:function(a){try{return!a.status&&location.protocol==="file:"||a.status>=200&&a.status<300||a.status===304||a.status=== 131 | 1223||a.status===0}catch(b){}return false},httpNotModified:function(a,b){var d=a.getResponseHeader("Last-Modified"),f=a.getResponseHeader("Etag");if(d)c.lastModified[b]=d;if(f)c.etag[b]=f;return a.status===304||a.status===0},httpData:function(a,b,d){var f=a.getResponseHeader("content-type")||"",e=b==="xml"||!b&&f.indexOf("xml")>=0;a=e?a.responseXML:a.responseText;e&&a.documentElement.nodeName==="parsererror"&&c.error("parsererror");if(d&&d.dataFilter)a=d.dataFilter(a,b);if(typeof a==="string")if(b=== 132 | "json"||!b&&f.indexOf("json")>=0)a=c.parseJSON(a);else if(b==="script"||!b&&f.indexOf("javascript")>=0)c.globalEval(a);return a},param:function(a,b){function d(i,o){if(c.isArray(o))c.each(o,function(k,n){b||/\[\]$/.test(i)?f(i,n):d(i+"["+(typeof n==="object"||c.isArray(n)?k:"")+"]",n)});else!b&&o!=null&&typeof o==="object"?c.each(o,function(k,n){d(i+"["+k+"]",n)}):f(i,o)}function f(i,o){o=c.isFunction(o)?o():o;e[e.length]=encodeURIComponent(i)+"="+encodeURIComponent(o)}var e=[];if(b===w)b=c.ajaxSettings.traditional; 133 | if(c.isArray(a)||a.jquery)c.each(a,function(){f(this.name,this.value)});else for(var j in a)d(j,a[j]);return e.join("&").replace(yb,"+")}});var la={},Ab=/toggle|show|hide/,Bb=/^([+-]=)?([\d+-.]+)(.*)$/,W,va=[["height","marginTop","marginBottom","paddingTop","paddingBottom"],["width","marginLeft","marginRight","paddingLeft","paddingRight"],["opacity"]];c.fn.extend({show:function(a,b){if(a||a===0)return this.animate(K("show",3),a,b);else{a=0;for(b=this.length;a").appendTo("body");f=e.css("display");if(f==="none")f="block";e.remove();la[d]=f}c.data(this[a],"olddisplay",f)}}a=0;for(b=this.length;a=0;f--)if(d[f].elem===this){b&&d[f](true);d.splice(f,1)}});b||this.dequeue();return this}});c.each({slideDown:K("show",1),slideUp:K("hide",1),slideToggle:K("toggle",1),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"}},function(a,b){c.fn[a]=function(d,f){return this.animate(b,d,f)}});c.extend({speed:function(a,b,d){var f=a&&typeof a==="object"?a:{complete:d||!d&&b||c.isFunction(a)&&a,duration:a,easing:d&&b||b&&!c.isFunction(b)&&b};f.duration=c.fx.off?0:typeof f.duration=== 139 | "number"?f.duration:c.fx.speeds[f.duration]||c.fx.speeds._default;f.old=f.complete;f.complete=function(){f.queue!==false&&c(this).dequeue();c.isFunction(f.old)&&f.old.call(this)};return f},easing:{linear:function(a,b,d,f){return d+f*a},swing:function(a,b,d,f){return(-Math.cos(a*Math.PI)/2+0.5)*f+d}},timers:[],fx:function(a,b,d){this.options=b;this.elem=a;this.prop=d;if(!b.orig)b.orig={}}});c.fx.prototype={update:function(){this.options.step&&this.options.step.call(this.elem,this.now,this);(c.fx.step[this.prop]|| 140 | c.fx.step._default)(this);if((this.prop==="height"||this.prop==="width")&&this.elem.style)this.elem.style.display="block"},cur:function(a){if(this.elem[this.prop]!=null&&(!this.elem.style||this.elem.style[this.prop]==null))return this.elem[this.prop];return(a=parseFloat(c.css(this.elem,this.prop,a)))&&a>-10000?a:parseFloat(c.curCSS(this.elem,this.prop))||0},custom:function(a,b,d){function f(j){return e.step(j)}this.startTime=J();this.start=a;this.end=b;this.unit=d||this.unit||"px";this.now=this.start; 141 | this.pos=this.state=0;var e=this;f.elem=this.elem;if(f()&&c.timers.push(f)&&!W)W=setInterval(c.fx.tick,13)},show:function(){this.options.orig[this.prop]=c.style(this.elem,this.prop);this.options.show=true;this.custom(this.prop==="width"||this.prop==="height"?1:0,this.cur());c(this.elem).show()},hide:function(){this.options.orig[this.prop]=c.style(this.elem,this.prop);this.options.hide=true;this.custom(this.cur(),0)},step:function(a){var b=J(),d=true;if(a||b>=this.options.duration+this.startTime){this.now= 142 | this.end;this.pos=this.state=1;this.update();this.options.curAnim[this.prop]=true;for(var f in this.options.curAnim)if(this.options.curAnim[f]!==true)d=false;if(d){if(this.options.display!=null){this.elem.style.overflow=this.options.overflow;a=c.data(this.elem,"olddisplay");this.elem.style.display=a?a:this.options.display;if(c.css(this.elem,"display")==="none")this.elem.style.display="block"}this.options.hide&&c(this.elem).hide();if(this.options.hide||this.options.show)for(var e in this.options.curAnim)c.style(this.elem, 143 | e,this.options.orig[e]);this.options.complete.call(this.elem)}return false}else{e=b-this.startTime;this.state=e/this.options.duration;a=this.options.easing||(c.easing.swing?"swing":"linear");this.pos=c.easing[this.options.specialEasing&&this.options.specialEasing[this.prop]||a](this.state,e,0,1,this.options.duration);this.now=this.start+(this.end-this.start)*this.pos;this.update()}return true}};c.extend(c.fx,{tick:function(){for(var a=c.timers,b=0;b
    "; 149 | a.insertBefore(b,a.firstChild);d=b.firstChild;f=d.firstChild;e=d.nextSibling.firstChild.firstChild;this.doesNotAddBorder=f.offsetTop!==5;this.doesAddBorderForTableAndCells=e.offsetTop===5;f.style.position="fixed";f.style.top="20px";this.supportsFixedPosition=f.offsetTop===20||f.offsetTop===15;f.style.position=f.style.top="";d.style.overflow="hidden";d.style.position="relative";this.subtractsBorderForOverflowNotVisible=f.offsetTop===-5;this.doesNotIncludeMarginInBodyOffset=a.offsetTop!==j;a.removeChild(b); 150 | c.offset.initialize=c.noop},bodyOffset:function(a){var b=a.offsetTop,d=a.offsetLeft;c.offset.initialize();if(c.offset.doesNotIncludeMarginInBodyOffset){b+=parseFloat(c.curCSS(a,"marginTop",true))||0;d+=parseFloat(c.curCSS(a,"marginLeft",true))||0}return{top:b,left:d}},setOffset:function(a,b,d){if(/static/.test(c.curCSS(a,"position")))a.style.position="relative";var f=c(a),e=f.offset(),j=parseInt(c.curCSS(a,"top",true),10)||0,i=parseInt(c.curCSS(a,"left",true),10)||0;if(c.isFunction(b))b=b.call(a, 151 | d,e);d={top:b.top-e.top+j,left:b.left-e.left+i};"using"in b?b.using.call(a,d):f.css(d)}};c.fn.extend({position:function(){if(!this[0])return null;var a=this[0],b=this.offsetParent(),d=this.offset(),f=/^body|html$/i.test(b[0].nodeName)?{top:0,left:0}:b.offset();d.top-=parseFloat(c.curCSS(a,"marginTop",true))||0;d.left-=parseFloat(c.curCSS(a,"marginLeft",true))||0;f.top+=parseFloat(c.curCSS(b[0],"borderTopWidth",true))||0;f.left+=parseFloat(c.curCSS(b[0],"borderLeftWidth",true))||0;return{top:d.top- 152 | f.top,left:d.left-f.left}},offsetParent:function(){return this.map(function(){for(var a=this.offsetParent||s.body;a&&!/^body|html$/i.test(a.nodeName)&&c.css(a,"position")==="static";)a=a.offsetParent;return a})}});c.each(["Left","Top"],function(a,b){var d="scroll"+b;c.fn[d]=function(f){var e=this[0],j;if(!e)return null;if(f!==w)return this.each(function(){if(j=wa(this))j.scrollTo(!a?f:c(j).scrollLeft(),a?f:c(j).scrollTop());else this[d]=f});else return(j=wa(e))?"pageXOffset"in j?j[a?"pageYOffset": 153 | "pageXOffset"]:c.support.boxModel&&j.document.documentElement[d]||j.document.body[d]:e[d]}});c.each(["Height","Width"],function(a,b){var d=b.toLowerCase();c.fn["inner"+b]=function(){return this[0]?c.css(this[0],d,false,"padding"):null};c.fn["outer"+b]=function(f){return this[0]?c.css(this[0],d,false,f?"margin":"border"):null};c.fn[d]=function(f){var e=this[0];if(!e)return f==null?null:this;if(c.isFunction(f))return this.each(function(j){var i=c(this);i[d](f.call(this,j,i[d]()))});return"scrollTo"in 154 | e&&e.document?e.document.compatMode==="CSS1Compat"&&e.document.documentElement["client"+b]||e.document.body["client"+b]:e.nodeType===9?Math.max(e.documentElement["client"+b],e.body["scroll"+b],e.documentElement["scroll"+b],e.body["offset"+b],e.documentElement["offset"+b]):f===w?c.css(e,d):this.css(d,typeof f==="string"?f:f+"px")}});A.jQuery=A.$=c})(window); -------------------------------------------------------------------------------- /lib/channel.js: -------------------------------------------------------------------------------- 1 | var EventEmitter = require("events").EventEmitter, 2 | sys = require("sys"), 3 | Session = require("./session").Session; 4 | 5 | function Channel(options) { 6 | EventEmitter.call(this); 7 | 8 | if (!options || !options.basePath) { 9 | return false; 10 | } 11 | 12 | this.basePath = options.basePath; 13 | this.messageBacklog = parseInt(options.messageBacklog) || 200; 14 | this.sessionTimeout = (parseInt(options.sessionTimeout) || 60) * 1000; 15 | 16 | this.nextMessageId = 0; 17 | this.messages = []; 18 | this.callbacks = []; 19 | this.sessions = {}; 20 | 21 | var channel = this; 22 | setInterval(function() { 23 | channel.flushCallbacks(); 24 | channel.expireOldSessions(); 25 | }, 1000); 26 | } 27 | sys.inherits(Channel, EventEmitter); 28 | 29 | extend(Channel.prototype, { 30 | appendMessage: function(nick, type, text) { 31 | var id = ++this.nextMessageId, 32 | message = { 33 | id: id, 34 | nick: nick, 35 | type: type, 36 | text: text, 37 | timestamp: (new Date()).getTime() 38 | }; 39 | this.messages.push(message); 40 | this.emit(type, message); 41 | 42 | while (this.callbacks.length > 0) { 43 | this.callbacks.shift().callback([message]); 44 | } 45 | 46 | while (this.messages.length > this.messageBacklog) { 47 | this.messages.shift(); 48 | } 49 | 50 | return id; 51 | }, 52 | 53 | query: function(since, callback) { 54 | var matching = [], 55 | length = this.messages.length; 56 | for (var i = 0; i < length; i++) { 57 | if (this.messages[i].id > since) { 58 | matching = this.messages.slice(i); 59 | break; 60 | } 61 | } 62 | 63 | if (matching.length) { 64 | callback(matching); 65 | } else { 66 | this.callbacks.push({ 67 | timestamp: new Date(), 68 | callback: callback 69 | }); 70 | } 71 | }, 72 | 73 | flushCallbacks: function() { 74 | var now = new Date(); 75 | while (this.callbacks.length && now - this.callbacks[0].timestamp > this.sessionTimeout * 0.75) { 76 | this.callbacks.shift().callback([]); 77 | } 78 | }, 79 | 80 | createSession: function(nick) { 81 | var session = new Session(nick); 82 | if (!session) { 83 | return; 84 | } 85 | 86 | nick = nick.toLowerCase(); 87 | for (var i in this.sessions) { 88 | if (this.sessions[i].nick && this.sessions[i].nick.toLowerCase() === nick) { 89 | return; 90 | } 91 | } 92 | 93 | this.sessions[session.id] = session; 94 | session.since = this.appendMessage(nick, "join"); 95 | 96 | return session; 97 | }, 98 | 99 | destroySession: function(id) { 100 | if (!id || !this.sessions[id]) { 101 | return false; 102 | } 103 | 104 | var eventId = this.appendMessage(this.sessions[id].nick, "part"); 105 | delete this.sessions[id]; 106 | return eventId; 107 | }, 108 | 109 | expireOldSessions: function() { 110 | var now = new Date(); 111 | for (var session in this.sessions) { 112 | if (now - this.sessions[session].timestamp > this.sessionTimeout) { 113 | this.destroySession(session); 114 | } 115 | } 116 | } 117 | }); 118 | 119 | exports.Channel = Channel; 120 | 121 | 122 | 123 | function extend(obj, props) { 124 | for (var prop in props) { 125 | obj[prop] = props[prop]; 126 | } 127 | } 128 | -------------------------------------------------------------------------------- /lib/router/index.js: -------------------------------------------------------------------------------- 1 | var createServer = require("http").createServer, 2 | readFile = require("fs").readFile, 3 | sys = require("sys"), 4 | url = require("url"), 5 | mime = require("./mime"); 6 | 7 | var router = exports; 8 | var NOT_FOUND = "Not Found\n"; 9 | 10 | function notFound(req, res) { 11 | res.writeHead(404, [ 12 | ["Content-Type", "text/plain"], 13 | ["Content-Length", NOT_FOUND.length] 14 | ]); 15 | res.write(NOT_FOUND); 16 | res.end(); 17 | } 18 | 19 | router.createServer = function() { 20 | var getMap = {}; 21 | 22 | server = createServer(function(req, res) { 23 | if (req.method === "GET" || req.method === "HEAD") { 24 | var handler = getMap[url.parse(req.url).pathname] || notFound; 25 | 26 | res.simpleText = function (code, body) { 27 | res.writeHead(code, [ ["Content-Type", "text/plain"] 28 | , ["Content-Length", body.length] 29 | ]); 30 | res.write(body); 31 | res.end(); 32 | }; 33 | 34 | res.simpleJSON = function (code, obj) { 35 | var body = JSON.stringify(obj); 36 | res.writeHead(code, [ ["Content-Type", "text/json"] 37 | , ["Content-Length", body.length] 38 | ]); 39 | res.write(body); 40 | res.end(); 41 | }; 42 | 43 | handler(req, res); 44 | } 45 | }); 46 | 47 | return { 48 | listen: function(port, host) { 49 | server.listen(port, host); 50 | console.log("Server at http://" + (host || "127.0.0.1") + ":" + port.toString() + "/"); 51 | }, 52 | get: function(path, handler) { 53 | getMap[path] = handler; 54 | } 55 | }; 56 | }; 57 | 58 | function extname (path) { 59 | var index = path.lastIndexOf("."); 60 | return index < 0 ? "" : path.substring(index); 61 | } 62 | 63 | router.staticHandler = function (filename) { 64 | var body, headers; 65 | var content_type = mime.lookupExtension(extname(filename)); 66 | var encoding = (content_type.slice(0,4) === "text" ? "utf8" : "binary"); 67 | 68 | function loadResponseData(callback) { 69 | if (body && headers) { 70 | callback(); 71 | return; 72 | } 73 | 74 | console.log("loading " + filename + "..."); 75 | readFile(filename, encoding, function (err, data) { 76 | if (err) { 77 | console.log("Error loading " + filename); 78 | } else { 79 | body = data; 80 | headers = [ [ "Content-Type" , content_type ] 81 | , [ "Content-Length" , body.length ] 82 | ]; 83 | headers.push(["Cache-Control", "public"]); 84 | 85 | console.log("static file " + filename + " loaded"); 86 | callback(); 87 | } 88 | }); 89 | } 90 | 91 | return function (req, res) { 92 | loadResponseData(function () { 93 | res.writeHead(200, headers); 94 | res.write(body, encoding); 95 | res.end(); 96 | }); 97 | }; 98 | }; 99 | -------------------------------------------------------------------------------- /lib/router/mime.js: -------------------------------------------------------------------------------- 1 | // returns MIME type for extension, or fallback, or octet-steam 2 | exports.lookupExtension = function(ext, fallback) { 3 | return exports.TYPES[ext.toLowerCase()] || fallback || 'application/octet-stream'; 4 | }; 5 | 6 | // List of most common mime-types, stolen from Rack. 7 | exports.TYPES = { ".3gp" : "video/3gpp" 8 | , ".a" : "application/octet-stream" 9 | , ".ai" : "application/postscript" 10 | , ".aif" : "audio/x-aiff" 11 | , ".aiff" : "audio/x-aiff" 12 | , ".asc" : "application/pgp-signature" 13 | , ".asf" : "video/x-ms-asf" 14 | , ".asm" : "text/x-asm" 15 | , ".asx" : "video/x-ms-asf" 16 | , ".atom" : "application/atom+xml" 17 | , ".au" : "audio/basic" 18 | , ".avi" : "video/x-msvideo" 19 | , ".bat" : "application/x-msdownload" 20 | , ".bin" : "application/octet-stream" 21 | , ".bmp" : "image/bmp" 22 | , ".bz2" : "application/x-bzip2" 23 | , ".c" : "text/x-c" 24 | , ".cab" : "application/vnd.ms-cab-compressed" 25 | , ".cc" : "text/x-c" 26 | , ".chm" : "application/vnd.ms-htmlhelp" 27 | , ".class" : "application/octet-stream" 28 | , ".com" : "application/x-msdownload" 29 | , ".conf" : "text/plain" 30 | , ".cpp" : "text/x-c" 31 | , ".crt" : "application/x-x509-ca-cert" 32 | , ".css" : "text/css" 33 | , ".csv" : "text/csv" 34 | , ".cxx" : "text/x-c" 35 | , ".deb" : "application/x-debian-package" 36 | , ".der" : "application/x-x509-ca-cert" 37 | , ".diff" : "text/x-diff" 38 | , ".djv" : "image/vnd.djvu" 39 | , ".djvu" : "image/vnd.djvu" 40 | , ".dll" : "application/x-msdownload" 41 | , ".dmg" : "application/octet-stream" 42 | , ".doc" : "application/msword" 43 | , ".dot" : "application/msword" 44 | , ".dtd" : "application/xml-dtd" 45 | , ".dvi" : "application/x-dvi" 46 | , ".ear" : "application/java-archive" 47 | , ".eml" : "message/rfc822" 48 | , ".eps" : "application/postscript" 49 | , ".exe" : "application/x-msdownload" 50 | , ".f" : "text/x-fortran" 51 | , ".f77" : "text/x-fortran" 52 | , ".f90" : "text/x-fortran" 53 | , ".flv" : "video/x-flv" 54 | , ".for" : "text/x-fortran" 55 | , ".gem" : "application/octet-stream" 56 | , ".gemspec" : "text/x-script.ruby" 57 | , ".gif" : "image/gif" 58 | , ".gz" : "application/x-gzip" 59 | , ".h" : "text/x-c" 60 | , ".hh" : "text/x-c" 61 | , ".htm" : "text/html" 62 | , ".html" : "text/html" 63 | , ".ico" : "image/vnd.microsoft.icon" 64 | , ".ics" : "text/calendar" 65 | , ".ifb" : "text/calendar" 66 | , ".iso" : "application/octet-stream" 67 | , ".jar" : "application/java-archive" 68 | , ".java" : "text/x-java-source" 69 | , ".jnlp" : "application/x-java-jnlp-file" 70 | , ".jpeg" : "image/jpeg" 71 | , ".jpg" : "image/jpeg" 72 | , ".js" : "application/javascript" 73 | , ".json" : "application/json" 74 | , ".log" : "text/plain" 75 | , ".m3u" : "audio/x-mpegurl" 76 | , ".m4v" : "video/mp4" 77 | , ".man" : "text/troff" 78 | , ".mathml" : "application/mathml+xml" 79 | , ".mbox" : "application/mbox" 80 | , ".mdoc" : "text/troff" 81 | , ".me" : "text/troff" 82 | , ".mid" : "audio/midi" 83 | , ".midi" : "audio/midi" 84 | , ".mime" : "message/rfc822" 85 | , ".mml" : "application/mathml+xml" 86 | , ".mng" : "video/x-mng" 87 | , ".mov" : "video/quicktime" 88 | , ".mp3" : "audio/mpeg" 89 | , ".mp4" : "video/mp4" 90 | , ".mp4v" : "video/mp4" 91 | , ".mpeg" : "video/mpeg" 92 | , ".mpg" : "video/mpeg" 93 | , ".ms" : "text/troff" 94 | , ".msi" : "application/x-msdownload" 95 | , ".odp" : "application/vnd.oasis.opendocument.presentation" 96 | , ".ods" : "application/vnd.oasis.opendocument.spreadsheet" 97 | , ".odt" : "application/vnd.oasis.opendocument.text" 98 | , ".ogg" : "application/ogg" 99 | , ".p" : "text/x-pascal" 100 | , ".pas" : "text/x-pascal" 101 | , ".pbm" : "image/x-portable-bitmap" 102 | , ".pdf" : "application/pdf" 103 | , ".pem" : "application/x-x509-ca-cert" 104 | , ".pgm" : "image/x-portable-graymap" 105 | , ".pgp" : "application/pgp-encrypted" 106 | , ".pkg" : "application/octet-stream" 107 | , ".pl" : "text/x-script.perl" 108 | , ".pm" : "text/x-script.perl-module" 109 | , ".png" : "image/png" 110 | , ".pnm" : "image/x-portable-anymap" 111 | , ".ppm" : "image/x-portable-pixmap" 112 | , ".pps" : "application/vnd.ms-powerpoint" 113 | , ".ppt" : "application/vnd.ms-powerpoint" 114 | , ".ps" : "application/postscript" 115 | , ".psd" : "image/vnd.adobe.photoshop" 116 | , ".py" : "text/x-script.python" 117 | , ".qt" : "video/quicktime" 118 | , ".ra" : "audio/x-pn-realaudio" 119 | , ".rake" : "text/x-script.ruby" 120 | , ".ram" : "audio/x-pn-realaudio" 121 | , ".rar" : "application/x-rar-compressed" 122 | , ".rb" : "text/x-script.ruby" 123 | , ".rdf" : "application/rdf+xml" 124 | , ".roff" : "text/troff" 125 | , ".rpm" : "application/x-redhat-package-manager" 126 | , ".rss" : "application/rss+xml" 127 | , ".rtf" : "application/rtf" 128 | , ".ru" : "text/x-script.ruby" 129 | , ".s" : "text/x-asm" 130 | , ".sgm" : "text/sgml" 131 | , ".sgml" : "text/sgml" 132 | , ".sh" : "application/x-sh" 133 | , ".sig" : "application/pgp-signature" 134 | , ".snd" : "audio/basic" 135 | , ".so" : "application/octet-stream" 136 | , ".svg" : "image/svg+xml" 137 | , ".svgz" : "image/svg+xml" 138 | , ".swf" : "application/x-shockwave-flash" 139 | , ".t" : "text/troff" 140 | , ".tar" : "application/x-tar" 141 | , ".tbz" : "application/x-bzip-compressed-tar" 142 | , ".tcl" : "application/x-tcl" 143 | , ".tex" : "application/x-tex" 144 | , ".texi" : "application/x-texinfo" 145 | , ".texinfo" : "application/x-texinfo" 146 | , ".text" : "text/plain" 147 | , ".tif" : "image/tiff" 148 | , ".tiff" : "image/tiff" 149 | , ".torrent" : "application/x-bittorrent" 150 | , ".tr" : "text/troff" 151 | , ".txt" : "text/plain" 152 | , ".vcf" : "text/x-vcard" 153 | , ".vcs" : "text/x-vcalendar" 154 | , ".vrml" : "model/vrml" 155 | , ".war" : "application/java-archive" 156 | , ".wav" : "audio/x-wav" 157 | , ".wma" : "audio/x-ms-wma" 158 | , ".wmv" : "video/x-ms-wmv" 159 | , ".wmx" : "video/x-ms-wmx" 160 | , ".wrl" : "model/vrml" 161 | , ".wsdl" : "application/wsdl+xml" 162 | , ".xbm" : "image/x-xbitmap" 163 | , ".xhtml" : "application/xhtml+xml" 164 | , ".xls" : "application/vnd.ms-excel" 165 | , ".xml" : "application/xml" 166 | , ".xpm" : "image/x-xpixmap" 167 | , ".xsl" : "application/xml" 168 | , ".xslt" : "application/xslt+xml" 169 | , ".yaml" : "text/yaml" 170 | , ".yml" : "text/yaml" 171 | , ".zip" : "application/zip" 172 | }; 173 | -------------------------------------------------------------------------------- /lib/server.js: -------------------------------------------------------------------------------- 1 | var router = require("./router"), 2 | url = require("url"), 3 | qs = require("querystring"), 4 | Channel = require("./channel").Channel; 5 | 6 | function Server() { 7 | this.httpServer = router.createServer(); 8 | this.channels = []; 9 | } 10 | 11 | extend(Server.prototype, { 12 | listen: function(port, host) { 13 | this.httpServer.listen(port, host); 14 | }, 15 | 16 | passThru: function(path, handler) { 17 | this.httpServer.get(path, handler); 18 | }, 19 | 20 | addChannel: function(options) { 21 | var httpServer = this.httpServer, 22 | channel = new Channel(options); 23 | 24 | if (!channel) { 25 | return false; 26 | } 27 | 28 | this.channels.push(channel); 29 | 30 | handlers.forEach(function(handler) { 31 | httpServer.get(channel.basePath + handler.path, 32 | handler.handler.partial(channel)); 33 | }); 34 | 35 | return channel; 36 | } 37 | }); 38 | 39 | exports.createServer = function() { 40 | return new Server(); 41 | }; 42 | 43 | 44 | 45 | var handlers = [ 46 | { path: "/who", handler: function(channel, request, response) { 47 | var nicks = []; 48 | for (var id in channel.sessions) { 49 | nicks.push(channel.sessions[id].nick); 50 | } 51 | response.simpleJSON(200, { nicks: nicks }); 52 | } }, 53 | { path: "/join", handler: function(channel, request, response) { 54 | var nick = qs.parse(url.parse(request.url).query).nick; 55 | 56 | if (!nick) { 57 | response.simpleJSON(400, { error: "bad nick." }); 58 | return; 59 | } 60 | var session = channel.createSession(nick); 61 | if (!session) { 62 | response.simpleJSON(400, { error: "nick in use." }); 63 | return; 64 | } 65 | 66 | response.simpleJSON(200, { id: session.id, nick: nick, since: session.since }); 67 | } }, 68 | { path: "/part", handler: function(channel, request, response) { 69 | var id = qs.parse(url.parse(request.url).query).id; 70 | 71 | var eventId = channel.destroySession(id); 72 | response.simpleJSON(200, { id: eventId }); 73 | } }, 74 | { path: "/recv", handler: function(channel, request, response) { 75 | var query = qs.parse(url.parse(request.url).query), 76 | since = parseInt(query.since, 10), 77 | session = channel.sessions[query.id]; 78 | 79 | if (!session) { 80 | response.simpleJSON(400, { error: "No such session id." }); 81 | return; 82 | } 83 | 84 | if (isNaN(since)) { 85 | response.simpleJSON(400, { error: "Must supply since parameter." }); 86 | return; 87 | } 88 | 89 | session.poke(); 90 | channel.query(since, function(messages) { 91 | session.poke(); 92 | response.simpleJSON(200, { messages: messages }); 93 | }); 94 | } }, 95 | { path: "/send", handler: function(channel, request, response) { 96 | var query = qs.parse(url.parse(request.url).query), 97 | text = query.text, 98 | session = channel.sessions[query.id]; 99 | 100 | if (!session) { 101 | response.simpleJSON(400, { error: "No such session id." }); 102 | return; 103 | } 104 | 105 | if (!text || !text.length) { 106 | response.simpleJSON(400, { error: "Must supply text parameter." }); 107 | return; 108 | } 109 | 110 | session.poke(); 111 | var id = channel.appendMessage(session.nick, "msg", text); 112 | response.simpleJSON(200, { id: id }); 113 | } } 114 | ]; 115 | 116 | 117 | 118 | var slice = [].slice; 119 | Function.prototype.partial = function() { 120 | var fn = this, 121 | args = slice.call(arguments); 122 | 123 | return function() { 124 | return fn.apply(this, args.concat(slice.call(arguments))); 125 | }; 126 | }; 127 | 128 | function extend(obj, props) { 129 | for (var prop in props) { 130 | obj[prop] = props[prop]; 131 | } 132 | } 133 | -------------------------------------------------------------------------------- /lib/session.js: -------------------------------------------------------------------------------- 1 | function Session(nick) { 2 | if (nick.length > 50) { 3 | return; 4 | } 5 | if (!/^[a-zA-Z\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]+$/.test(nick)) { 6 | return; 7 | } 8 | 9 | this.nick = nick; 10 | this.id = Math.floor(Math.random() * 1e10).toString(); 11 | this.timestamp = new Date(); 12 | } 13 | 14 | Session.prototype.poke = function() { 15 | this.timestamp = new Date(); 16 | }; 17 | 18 | exports.Session = Session; 19 | -------------------------------------------------------------------------------- /package-lock.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "chat", 3 | "version": "0.2.1", 4 | "lockfileVersion": 3, 5 | "requires": true, 6 | "packages": { 7 | "": { 8 | "name": "chat", 9 | "version": "0.2.1" 10 | } 11 | } 12 | } 13 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "chat", 3 | "version": "0.2.1", 4 | "description": "A simple, scalable web-based chat server.", 5 | "keywords": [ 6 | "chat" 7 | ], 8 | "homepage": "https://github.com/scottgonzalez/node-chat", 9 | "bugs": "https://github.com/scottgonzalez/node-chat/issues", 10 | "author": { 11 | "name": "Scott González", 12 | "email": "scott.gonzalez@gmail.com", 13 | "url": "http://scottgonzalez.com" 14 | }, 15 | "main": "lib/server.js", 16 | "repository": { 17 | "type": "git", 18 | "url": "git://github.com/scottgonzalez/node-chat.git" 19 | } 20 | } 21 | -------------------------------------------------------------------------------- /readme.md: -------------------------------------------------------------------------------- 1 | NodeChat 2 | ======== 3 | 4 | NodeChat is a simple, scalable web-based chat server built on [Node.js](http://nodejs.org). 5 | 6 | 7 | Server 8 | ------ 9 | 10 | Creating a chat server is super simple. 11 | 12 | // create a server at http://localhost:8001 13 | var chat = require("lib/server"); 14 | var chatServer = chat.createServer(); 15 | chatServer.listen(8001); 16 | 17 | Once you've created a server, you can add channels. 18 | 19 | // create a channel at http://localhost:8001/chat 20 | var channel = chatServer.addChannel({ basePath: "/chat" }); 21 | 22 | Channels have three options: 23 | 24 | * `basePath` (required) - the URL to use for the channel 25 | * `messageBacklog` (default 200) - how many message to keep in memory 26 | * `sessionTimeout` (default 60) - how many seconds to wait before killing inactive sessions 27 | 28 | 29 | Client 30 | ------ 31 | 32 | NodeChat comes with a client library built on [jQuery](http://jquery.com) to make it easy to connect to a NodeChat server. 33 | 34 | // connect to a channel 35 | var channel = nodeChat.connect("http://localhost:8001/chat"); 36 | 37 | // join the channel 38 | channel.join("my-nick"); 39 | 40 | // send a message to the channel 41 | channel.send("hello"); 42 | 43 | // leave the channel 44 | channel.part(); 45 | 46 | 47 | Events 48 | ------ 49 | 50 | The server and client both emit events on the channel objects. 51 | 52 | * `join` - emitted when someone joins the channel. 53 | * `msg` - emitted when someone sends a message to the channel. 54 | * `part` - emitted when someone leaves the channel. 55 | 56 | Each event receives an object with the following properties: 57 | 58 | * `id` - unique id for the event (unique per channel). 59 | * `nick` - the nick of the user who performed the action. 60 | * `type` - the type of action (will be same as the event type). 61 | * `text` - text associated with the action. 62 | * `timestamp` - when the action occurred. 63 | 64 | In addition, the client emits a `connectionerror` event if the connection to the server is lost. 65 | 66 | 67 | Installation & Demo 68 | ------------------- 69 | 70 | To install: 71 | 72 | git clone http://github.com/scottgonzalez/node-chat.git 73 | 74 | To run the demo: 75 | 76 | ./node-chat/demo/chat.js 77 | Then open `http://localhost:8001` in a browser. 78 | 79 | 80 | License 81 | ------- 82 | 83 | Copyright 2013 Scott González. Released under the terms of the MIT license. 84 | -------------------------------------------------------------------------------- /web/nodechat.js: -------------------------------------------------------------------------------- 1 | (function($) { 2 | 3 | var nodeChat = (window.nodeChat = { 4 | connect: function(basePath) { 5 | return new Channel(basePath); 6 | } 7 | }); 8 | 9 | function Channel(basePath) { 10 | this.basePath = basePath; 11 | bindAll(this); 12 | } 13 | 14 | $.extend(Channel.prototype, { 15 | pollingErrors: 0, 16 | lastMessageId: 0, 17 | id: null, 18 | 19 | request: function(url, options) { 20 | var channel = this; 21 | $.ajax($.extend({ 22 | url: this.basePath + url, 23 | cache: false, 24 | dataType: "json" 25 | }, options)); 26 | }, 27 | 28 | poll: function() { 29 | if (this.pollingErrors > 2) { 30 | $(this).triggerHandler("connectionerror"); 31 | return; 32 | } 33 | var channel = this; 34 | this.request("/recv", { 35 | data: { 36 | since: this.lastMessageId, 37 | id: this.id 38 | }, 39 | success: function(data) { 40 | if (data) { 41 | channel.handlePoll(data); 42 | } else { 43 | channel.handlePollError(); 44 | } 45 | }, 46 | error: this.handlePollError 47 | }); 48 | }, 49 | 50 | handlePoll: function(data) { 51 | this.pollingErrors = 0; 52 | var channel = this; 53 | if (data && data.messages) { 54 | $.each(data.messages, function(i, message) { 55 | channel.lastMessageId = Math.max(channel.lastMessageId, message.id); 56 | $(channel).triggerHandler(message.type, message); 57 | }); 58 | } 59 | this.poll(); 60 | }, 61 | 62 | handlePollError: function() { 63 | this.pollingErrors++; 64 | setTimeout(this.poll, 10*1000); 65 | } 66 | }); 67 | 68 | $.extend(Channel.prototype, { 69 | join: function(nick, options) { 70 | var channel = this; 71 | this.request("/join", { 72 | data: { 73 | nick: nick 74 | }, 75 | success: function(data) { 76 | if (!data) { 77 | (options.error || $.noop)(); 78 | return; 79 | } 80 | 81 | channel.id = data.id; 82 | channel.since = data.since; 83 | channel.poll(); 84 | 85 | (options.success || $.noop)(); 86 | }, 87 | error: options.error || $.noop 88 | }); 89 | }, 90 | 91 | part: function() { 92 | if (!this.id) { return; } 93 | this.request("/part", { 94 | data: { id: this.id } 95 | }); 96 | }, 97 | 98 | send: function(msg) { 99 | if (!this.id) { return; } 100 | // TODO: use POST 101 | this.request("/send", { 102 | data: { 103 | id: this.id, 104 | text: msg 105 | } 106 | }); 107 | }, 108 | 109 | who: function() { 110 | if (!this.id) { return; } 111 | this.request("/who", { 112 | success: function(data) { 113 | var users = $("#users"); 114 | $.each(data.nicks, function(i, nick) { 115 | users.append("
  • " + nick + "
  • "); 116 | }); 117 | } 118 | }); 119 | } 120 | }); 121 | 122 | function bind(fn, context) { 123 | return function() { 124 | return fn.apply(context, arguments); 125 | }; 126 | } 127 | function bindAll(obj) { 128 | for (var prop in obj) { 129 | if ($.isFunction(obj[prop])) { 130 | obj[prop] = bind(obj[prop], obj); 131 | } 132 | } 133 | } 134 | 135 | })(jQuery); 136 | --------------------------------------------------------------------------------