omnomirc.js 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062
  1. /*
  2. OmnomIRC COPYRIGHT 2010,2011 Netham45
  3. OmnomIRC3 rewrite COPYRIGHT 2013 Nathaniel 'Eeems' van Diepen
  4. This file is part of OmnomIRC.
  5. OmnomIRC is free software: you can redistribute it and/or modify
  6. it under the terms of the GNU General Public License as published by
  7. the Free Software Foundation, either version 3 of the License, or
  8. (at your option) any later version.
  9. OmnomIRC is distributed in the hope that it will be useful,
  10. but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. GNU General Public License for more details.
  13. You should have received a copy of the GNU General Public License
  14. along with OmnomIRC. If not, see <http://www.gnu.org/licenses/>.
  15. */
  16. (function(window,$,io,undefined){
  17. "use strict";
  18. var $o = window.OmnomIRC = window.$o = function(){
  19. return 'Version: '+version;
  20. },
  21. event = function(msg,type){
  22. if(settings.debug){
  23. type=typeof type == 'undefined'?'event':type;
  24. switch(type){
  25. case 'ready':type='document_ready';break;
  26. }
  27. log('['+type.toUpperCase()+'] '+msg);
  28. }
  29. },
  30. emit = window.emit = function(type,data){
  31. if($o.chat.connected()){
  32. socket.emit.apply(socket,arguments);
  33. }else{
  34. if(tabs.length > 0){
  35. $o.msg('Disconnected, cannot do anything');
  36. }
  37. }
  38. },
  39. noop = function(){},
  40. log = function(){
  41. console.log.apply(console,arguments);
  42. },
  43. exists = function(object){
  44. return typeof object != 'undefined';
  45. },
  46. prevent = function(e){
  47. e.stopImmediatePropagation();
  48. e.stopPropagation();
  49. e.preventDefault();
  50. return false;
  51. },
  52. selectedTab=0,
  53. settings = {
  54. colour: false,
  55. debug: false,
  56. timestamp: 'exact',
  57. server: location.origin,
  58. autoconnect: true,
  59. autojoin: [
  60. '#omnimaga',
  61. '#omnimaga-fr',
  62. '#irp'
  63. ],
  64. scrollspeed: 100,
  65. theme: 'default',
  66. nick: 'User'
  67. },
  68. tabs = [],
  69. properties = {
  70. nick: 'User',
  71. sig: '',
  72. tabs: tabs,
  73. themes: [
  74. 'default'
  75. ]
  76. },
  77. commands = [
  78. { // names
  79. cmd: 'names',
  80. fn: function(args){
  81. emit('names',{
  82. name: $o.ui.tabs.current().name
  83. });
  84. }
  85. },
  86. { // me
  87. cmd: 'me',
  88. help: 'Say something in third person',
  89. fn: function(args){
  90. var i,ret='';
  91. for(i=1;i<args.length;i++){
  92. ret += ' '+args[i];
  93. }
  94. emit('message',{
  95. from: 0,
  96. message: properties.nick+' '+ret,
  97. room: $o.ui.tabs.current().name
  98. });
  99. }
  100. },
  101. { // connect
  102. cmd: 'connect',
  103. fn: function(){
  104. if(!$o.chat.connected()){
  105. $o.chat.connect();
  106. }
  107. }
  108. },
  109. { // disconnect
  110. cmd: 'disconnect',
  111. fn: function(){
  112. $o.disconnect();
  113. }
  114. },
  115. { // nick
  116. cmd: 'nick',
  117. fn: function(args){
  118. $o.set('nick',args[1]);
  119. }
  120. },
  121. { // help
  122. cmd: 'help',
  123. fn: function(args){
  124. if(typeof args[1] == 'undefined'){
  125. var m = 'Commands:',i;
  126. for(i in commands){
  127. m += ' '+commands[i].cmd;
  128. }
  129. $o.msg(m);
  130. }else{
  131. var i,cmd;
  132. for(i in commands){
  133. cmd = commands[i];
  134. if(cmd.cmd == args[1] && typeof cmd.help != 'undefined'){
  135. $o.msg('Command /'+cmd.cmd+': '+cmd.help);
  136. return;
  137. }
  138. }
  139. $o.send('/help');
  140. }
  141. }
  142. },
  143. { // open
  144. cmd: 'open',
  145. fn: function(args){
  146. $o.ui.tabs.add(args[1]);
  147. }
  148. },
  149. { // clear
  150. cmd: 'clear',
  151. fn: function(args){
  152. $o.ui.tabs.current().clear();
  153. }
  154. },
  155. { // close
  156. cmd: 'close',
  157. fn: function(args){
  158. if(args.length > 1){
  159. $o.ui.tabs.remove(args[1]);
  160. }else{
  161. $o.ui.tabs.remove(selectedTab);
  162. }
  163. }
  164. },
  165. { // tabs
  166. cmd: 'tabs',
  167. fn: function(args){
  168. $o.msg('tabs:');
  169. for(var i in tabs){
  170. $o.msg(' ['+i+'] '+tabs[i].name);
  171. }
  172. }
  173. }
  174. ],
  175. handles = [
  176. { // names
  177. on: 'names',
  178. fn: function(data){
  179. var tab = $o.ui.tabs.tab(data.room),
  180. users = tab.users,
  181. i;
  182. tab.users = data.names;
  183. if($o.ui.tabs.idForName(data.room) == selectedTab){
  184. $o.ui.render.users();
  185. }
  186. $(users).each(function(i,v){
  187. if(v != null){
  188. if(tab.users.indexOf(v.trim()) == -1){
  189. emit('echo',{
  190. room: $o.ui.tabs.current().name,
  191. message: v+' left the room',
  192. from: 0
  193. });
  194. }
  195. }
  196. });
  197. }
  198. },
  199. { // authorized
  200. on: 'authorized',
  201. fn: function(data){
  202. properties.nick = data.nick;
  203. for(var i in settings.autojoin){
  204. emit('join',{
  205. name: settings.autojoin[i]
  206. });
  207. }
  208. }
  209. },
  210. { // join
  211. on: 'join',
  212. fn: function(data){
  213. event('joined '+data.name);
  214. var flag = tabs.length == 0;
  215. $o.ui.tabs.add(data.name);
  216. if(flag){
  217. $o.ui.tabs.select(0);
  218. }
  219. }
  220. },
  221. { // reconnect
  222. on: 'reconnect',
  223. fn: function(data){
  224. event('reconnected');
  225. properties.connected = true;
  226. $o.chat.auth();
  227. emit('echo',{
  228. room: $o.ui.tabs.current().name,
  229. from: 0,
  230. message: 'reconnected'
  231. });
  232. }
  233. },
  234. { // connect
  235. on: 'connect',
  236. fn: function(data){
  237. event('connected');
  238. properties.connected = true;
  239. $o.chat.auth();
  240. emit('echo',{
  241. room: $o.ui.tabs.current().name,
  242. from: 0,
  243. message: 'connected'
  244. });
  245. }
  246. },
  247. { // disconnect
  248. on: 'disconnect',
  249. fn: function(data){
  250. event('disconnected');
  251. properties.connected = false;
  252. $o.msg('* disconnected');
  253. }
  254. },
  255. { // message
  256. on: 'message',
  257. fn: function(data){
  258. event('recieved message');
  259. var date = new Date(),
  260. string,
  261. time = date.getTime(),
  262. child,
  263. i,
  264. msg = function(msg){
  265. string = '<span class="cell date_cell">[<abbr class="date date_'+time+'" title="'+date.toISOString()+'"></abbr>]</span>';
  266. child = $('<li>').html(string+'<span class="cell">'+
  267. msg
  268. .htmlentities()
  269. .replace(
  270. /(https?:\/\/(([-\w\.]+)+(:\d+)?(\/([\w/_\.]*(\?\S+)?)?)?))/g,
  271. "<a href=\"$1\" title=\"\">$1</a>"
  272. )
  273. +'</span>');
  274. $o.msg({html:child},data.room);
  275. };
  276. if(data.from != 0){
  277. msg(' <'+data.from+'> '+data.message);
  278. }else{
  279. msg(' * '+data.message);
  280. }
  281. abbrDate('abbr.date_'+time);
  282. if(settings.timestamp == ''){
  283. $('.date_cell').css('visibility','hidden');
  284. }else{
  285. $('.date_cell').css('visibility','visible');
  286. }
  287. }
  288. }
  289. ],
  290. hooks = [
  291. { // load - style
  292. type: 'style',
  293. hook: 'load',
  294. fn: function(){
  295. // STUB
  296. event('testing == '+testing,'debug');
  297. }
  298. }
  299. ],
  300. currentPlugin = 0,
  301. runHook = function(name){
  302. var i,hook,fn,sandbox = {
  303. testing: 'test'
  304. };
  305. for(i in hooks){
  306. hook = hooks[i];
  307. if(hook.hook == name){
  308. fn = (hook.fn+'').replace(/\/\/.+?(?=\n|\r|$)|\/\*[\s\S]+?\*\//g,'').replace(/\"/g,'\\"').replace(/\n/g,'').replace(/\r/g,'');
  309. fn = 'eval("with(this){('+fn+')();}");';
  310. try{
  311. (new Function(fn)).call(sandbox);
  312. }catch(e){
  313. event('Hook failed to run: '+e+"\nFunction that ran: "+fn,'hook_error');
  314. }
  315. }
  316. }
  317. },
  318. version = '3.0',
  319. abbrDate = function(selector){
  320. if(settings.timestamp == 'fuzzy'){
  321. $(selector).timeago();
  322. }else{
  323. $(selector).each(function(){
  324. var timestamp = settings.timestamp,
  325. i,
  326. text='',
  327. date = new Date($(this).attr('title'));
  328. if(timestamp == 'exact'){
  329. timestamp = 'H:m:s t';
  330. }
  331. for(i=0;i<timestamp.length;i++){
  332. switch(timestamp[i]){
  333. case 'H':text+=((date.getHours()+11)%12)+1;break;
  334. case 'h':text+=date.getHours();break;
  335. case 'm':text+=(date.getMinutes()>9?'':'0')+date.getMinutes();break;
  336. case 's':text+=(date.getSeconds()>9?'':'0')+date.getSeconds();break;
  337. case 't':text+=(date.getHours()>11)?'pm':'am';break;
  338. default:text+=timestamp[i];
  339. }
  340. }
  341. $(this).text(text);
  342. }).timeago('dispose');
  343. }
  344. },
  345. socket,$i,$s,$h,$cl,$c,$tl,hht;
  346. runHook('load');
  347. $.extend($o,{
  348. version: function(){
  349. return version;
  350. },
  351. register: {
  352. theme: function(name){
  353. if(-1==$.inArray(properties.themes,name)){
  354. properties.themes.push(name);
  355. return true;
  356. }
  357. return false;
  358. },
  359. command: function(name,fn,help){
  360. if(-1==$.inArray(commands,name)){
  361. var o = {
  362. cmd: name,
  363. fn: fn
  364. };
  365. if(typeof help != 'undefined'){
  366. o.help = help;
  367. }
  368. commands.push(o);
  369. return true;
  370. }
  371. return false;
  372. },
  373. plugin: function(){
  374. // STUB
  375. },
  376. setting: function(name,defaultVal,validate,values){
  377. // STUB
  378. },
  379. hook: function(event,fn){
  380. }
  381. },
  382. hook: function(event,fn){
  383. $o.register.hook(event,fn);
  384. },
  385. ui: {
  386. render: {
  387. settings: function(){
  388. var name,setting,frag = document.createDocumentFragment(),item;
  389. for(name in settings){
  390. setting = $o.get(name,true);
  391. switch(setting.type){
  392. case 'select':
  393. item = $('<select>')
  394. .attr('id','setting_'+name)
  395. .change(function(){
  396. $o.set(this.id.substr(8),$(this).find(':selected').text(),false);
  397. });
  398. for(var i in setting.values){
  399. item.append(
  400. $('<option>')
  401. .text(setting.values[i])
  402. );
  403. }
  404. item.find(':contains('+setting.val+')').attr('selected','selected');
  405. break;
  406. case 'array':
  407. item = $('<input>')
  408. .attr({
  409. type: 'text',
  410. id: 'setting_'+name
  411. })
  412. .val(setting.val)
  413. .change(function(){
  414. $o.set(this.id.substr(8),$(this).val().split(','),false);
  415. });
  416. break;
  417. case 'boolean':
  418. item = $('<input>')
  419. .attr({
  420. type: 'checkbox',
  421. id: 'setting_'+name
  422. })
  423. .change(function(){
  424. $o.set(this.id.substr(8),$(this).is(':checked'),false);
  425. });
  426. if(setting.val){
  427. item.attr('checked','checked');
  428. }
  429. break;
  430. case 'number':
  431. case 'string':default:
  432. item = $('<input>')
  433. .attr({
  434. type: 'text',
  435. id: 'setting_'+name
  436. })
  437. .val(setting.val)
  438. .change(function(){
  439. $o.set(this.id.substr(8),$(this).val(),false);
  440. });
  441. }
  442. $(frag).append(
  443. $('<li>')
  444. .addClass('row')
  445. .append(
  446. $('<span>')
  447. .text(name)
  448. .addClass('cell')
  449. )
  450. .append(
  451. $('<span>')
  452. .append(item)
  453. .addClass('cell')
  454. )
  455. );
  456. }
  457. if(settings.debug){
  458. $(frag).append(
  459. $('<li>')
  460. .addClass('row')
  461. .attr('id','console-log-controls')
  462. .append(
  463. $('<span>')
  464. .text('Debug Log')
  465. .addClass('cell')
  466. )
  467. .append(
  468. $('<span>')
  469. .addClass('cell')
  470. .append(
  471. $('<button>')
  472. .text('Show')
  473. .click(function(){
  474. $(this).text($('#console-log').is(':visible')?'Show':'Hide');
  475. $('#console-log').toggle();
  476. $('#console-log-clear').toggle();
  477. $('#content').toggle();
  478. })
  479. )
  480. .append(
  481. $('<button>')
  482. .text('Clear')
  483. .attr('id','console-log-clear')
  484. .hide()
  485. .click(function(){
  486. $('#console-log-pre').html('');
  487. })
  488. )
  489. )
  490. );
  491. }else{
  492. $('#console-log-pre').html('');
  493. $('#console-log').hide();
  494. $('#content').show();
  495. }
  496. $('#settings-list').html(frag);
  497. },
  498. users: function(){
  499. event('Rendering userlist');
  500. var $ul = $('#user-list').html(''),
  501. i,
  502. names = $o.ui.tabs.current().users;
  503. for(i in names){
  504. $ul.append(
  505. $('<li>').text(names[i])
  506. );
  507. }
  508. },
  509. tab: function(){
  510. $cl.html($($o.ui.tabs.current().body).clone());
  511. },
  512. tablist: function(){
  513. $tl.html('');
  514. var i,tab;
  515. for(i in tabs){
  516. tab = $o.ui.tabs.obj(i);
  517. if(i==selectedTab){
  518. tab.addClass('clicked');
  519. $('#title').text(tabs[i].name);
  520. $('#topic').text(tabs[i].topic);
  521. }
  522. $tl.append(tab);
  523. }
  524. if($tl.get(0).scrollHeight-20 != $tl.scrollTop()){
  525. $('#tabs-scroll-right').removeClass('disabled');
  526. }
  527. if($tl.scrollTop() != 0){
  528. $('#tabs-scroll-left').removeClass('disabled');
  529. }
  530. }
  531. },
  532. tabs: {
  533. add: function(name){
  534. event('Tab added: '+name);
  535. if(!(function(){
  536. for(var i in tabs){
  537. if(name==tabs[i].name){
  538. return true;
  539. }
  540. }
  541. return false;
  542. })()){
  543. var scroll = $.localStorage('tabs'),
  544. i,
  545. frag = document.createDocumentFragment(),
  546. id = tabs.length;
  547. for(i in scroll){
  548. if(scroll[i].name == name){
  549. scroll[i].body = $(scroll[i].body).slice(-10);
  550. $(frag)
  551. .append(scroll[i].body)
  552. .append(
  553. $('<li>').html('<span class="to_remove">-- loaded old scrollback for '+scroll[i].date+' --</span>')
  554. )
  555. .children()
  556. .children('.remove')
  557. .remove();
  558. $(frag)
  559. .children()
  560. .children('.to_remove')
  561. .removeClass('to_remove')
  562. .addClass('remove');
  563. event('loading old tab scrollback for '+name+' last saved +'+scroll[i].date);
  564. }
  565. }
  566. tabs.push({
  567. name: name,
  568. body: frag,
  569. date: new Date(),
  570. send: function(msg){
  571. $o.chat.send(msg,$o.ui.tabs.tab(id).name);
  572. },
  573. close: function(){
  574. $o.ui.tabs.remove(id);
  575. },
  576. users: [],
  577. names: function(){
  578. emit('names',{
  579. name: $o.ui.tabs.tab(id).name
  580. });
  581. },
  582. select: function(){
  583. $o.ui.tabs.select(id);
  584. },
  585. clear: function(){
  586. $cl.html('');
  587. $o.ui.tabs.tab(id).body = document.createDocumentFragment();
  588. emit('echo',{
  589. room: $o.ui.tabs.tab(id).name,
  590. message: 'messages cleared',
  591. from: 0
  592. });
  593. }
  594. });
  595. $tl.append($o.ui.tabs.obj(id));
  596. $o.ui.render.tablist();
  597. $o.ui.render.users();
  598. }else{
  599. event('Attempted to add an existing tab');
  600. }
  601. },
  602. remove: function(name){
  603. if(typeof name == 'number'){
  604. name = tabs[name].name;
  605. }
  606. for(var id=0;id<tabs.length;id++){
  607. if($o.ui.tabs.tab(id).name == name){
  608. event('Tab removed: '+$o.ui.tabs.tab(id).name);
  609. emit('part',{
  610. name: $o.ui.tabs.tab(id).name
  611. });
  612. tabs.splice(id,1);
  613. if(selectedTab==id&&selectedTab>0){
  614. selectedTab--;
  615. }
  616. break;
  617. }
  618. }
  619. $o.ui.render.tablist();
  620. $cl.html($o.ui.tabs.current().body);
  621. $o.ui.render.users();
  622. },
  623. selected: function(){
  624. return selectedTab;
  625. },
  626. idForName: function(name){
  627. for(var i in tabs){
  628. if(tabs[i].name == name){
  629. return i;
  630. }
  631. }
  632. return false;
  633. },
  634. tab: function(id){
  635. if(typeof id == 'string' && !id.isNumber()){
  636. id = $o.ui.tabs.idForName(id);
  637. if(!id) return false;
  638. }
  639. return typeof tabs[id] == 'undefined'?false:tabs[id];
  640. },
  641. dom: function(id){
  642. if(typeof id == 'string' && !id.isNumber()){
  643. id = $o.ui.tabs.idForName(id);
  644. if(!id) return false;
  645. }
  646. return typeof tabs[id] == 'undefined'?false:tabs[id].body;
  647. },
  648. obj: function(id){
  649. if(typeof id !== 'undefined'){
  650. if(typeof id == 'string' && !id.isNumber()){
  651. id = $o.ui.tabs.idForName(id);
  652. if(!id) return;
  653. }
  654. return $('<div>')
  655. .addClass('tab')
  656. .text($o.ui.tabs.tab(id).name)
  657. .mouseup(function(e){
  658. switch(e.which){
  659. case 1: // RMB
  660. if($(this).data('id')!=selectedTab){
  661. $o.ui.tabs.select($(this).data('id'));
  662. return prevent(e);
  663. }
  664. break;
  665. case 2: // MMB
  666. $(this).children('span.close-button').click();
  667. return prevent(e);
  668. break;
  669. case 3: // LMB
  670. return prevent(e);
  671. break;
  672. default:
  673. return prevent(e);
  674. }
  675. })
  676. .append(
  677. $('<span>')
  678. .addClass('close-button')
  679. .click(function(){
  680. $o.ui.tabs.remove(id);
  681. return false;
  682. })
  683. .css({
  684. 'position': 'absolute',
  685. 'background-color': 'inherit',
  686. 'top': 0,
  687. 'right': 0
  688. })
  689. .html('&times;')
  690. )
  691. .data('id',id);
  692. }
  693. },
  694. select: function(id){
  695. if(typeof id == 'string' && !id.isNumber()){
  696. id = $o.ui.tabs.idForName(id);
  697. if(!id) return false;
  698. }
  699. event(id+' '+$o.ui.tabs.tab(id).name,'tab_select');
  700. if(id<tabs.length&&id>=0){
  701. selectedTab=id;
  702. }
  703. $tl.children('.clicked').removeClass('clicked');
  704. $($tl.children().get(id)).addClass('clicked');
  705. $('#title').text($o.ui.tabs.tab(id).name);
  706. $('#topic').text($o.ui.tabs.tab(id).topic);
  707. $cl.html($($o.ui.tabs.tab(id).body).clone());
  708. abbrDate('abbr.date');
  709. $o.ui.render.users();
  710. setTimeout(function scrollContent(){
  711. if($c.scrollTop() < $c[0].scrollHeight){
  712. $c.scrollTop($c.scrollTop()+1);
  713. setTimeout(scrollContent,settings.scrollspeed);
  714. }else{
  715. event('scrolling stopped');
  716. }
  717. },settings.scrollspeed);
  718. },
  719. current: function(){
  720. if(tabs.length > 0 && tabs.length > selectedTab){
  721. return tabs[selectedTab];
  722. }else{
  723. return {
  724. name: '',
  725. body: document.createDocumentFragment(),
  726. date: new Date(),
  727. send: noop,
  728. close: noop,
  729. users: [],
  730. names: noop,
  731. select: noop,
  732. clear: noop
  733. }
  734. }
  735. }
  736. },
  737. },
  738. chat: {
  739. connect: function(server){
  740. if($o.chat.connected()){
  741. $o.disconnect();
  742. }
  743. if(typeof server == 'undefined'){
  744. server = settings.server;
  745. }
  746. socket = window.socket = io.connect(server);
  747. for(var i in handles){
  748. socket.on(handles[i].on,handles[i].fn);
  749. }
  750. $o.chat.auth();
  751. },
  752. disconnect: function(){
  753. if($o.chat.connected()){
  754. socket.disconnect();
  755. }
  756. },
  757. connected: function(){
  758. return typeof socket == 'undefined'?false:properties.connected;
  759. },
  760. send: function(msg,room){
  761. if(typeof room == 'undefined'){
  762. room = $o.ui.tabs.current().name;
  763. }
  764. if(msg !== ''){
  765. if(msg[0] == '/' && msg[1] != '/'){
  766. var args = msg.split(' '),
  767. cmd = args[0].substr(1),
  768. i;
  769. event(msg,'command');
  770. for(i in commands){
  771. if(commands[i].cmd == cmd){
  772. commands[i].fn(args);
  773. return;
  774. }
  775. }
  776. $o.msg(cmd+' is not a valid command.');
  777. }else{
  778. event(msg,'send');
  779. emit('message',{
  780. message: msg,
  781. room: room,
  782. from: properties.nick
  783. });
  784. }
  785. }
  786. },
  787. auth: function(){
  788. if(settings.nick == ''){
  789. $o.set('nick','User');
  790. return;
  791. }
  792. emit('auth',{
  793. nick: settings.nick
  794. // TODO - send authorization info
  795. });
  796. }
  797. },
  798. get: function(name,formatted){
  799. if(typeof formatted == 'undefined'){
  800. return exists(settings[name])?settings[name]:false;
  801. }else{
  802. var val = $o.get(name),
  803. type,
  804. values = false;
  805. switch(name){
  806. case 'theme':
  807. type = 'select';
  808. values = properties.themes;
  809. break;
  810. case 'autojoin':type = 'array';break;
  811. case 'timestamp':type = 'string';break;
  812. default:
  813. type = typeof val;
  814. }
  815. return {
  816. type: type,
  817. val: val,
  818. values: values,
  819. name: name
  820. };
  821. }
  822. },
  823. set: function(name,value,render){
  824. if(exists(settings[name])){
  825. settings[name] = value;
  826. $.localStorage('settings',JSON.stringify(settings));
  827. switch(name){
  828. case 'timestamp':
  829. abbrDate('abbr.date');
  830. if(settings.timestamp == ''){
  831. $('.date_cell').css('visibility','hidden');
  832. }else{
  833. $('.date_cell').css('visibility','visible');
  834. }
  835. break;
  836. case 'nick':
  837. $o.chat.auth();
  838. break;
  839. case 'debug':
  840. if(typeof render != 'undefined'){
  841. $o.ui.render.settings();
  842. }
  843. break;
  844. }
  845. if(typeof render == 'undefined'){
  846. $o.ui.render.settings();
  847. }
  848. return true;
  849. }else{
  850. return false;
  851. }
  852. },
  853. prop: function(name){
  854. return exists(properties[name])?properties[name]:null;
  855. },
  856. send: function(msg){
  857. $o.chat.send(msg);
  858. },
  859. msg: function(msg,tabName){
  860. var frag;
  861. if(typeof tabName == 'undefined' || tabName == $o.ui.tabs.current().name || typeof $o.ui.tabs.tab(tabName).body == 'undefined'){
  862. frag = document.createDocumentFragment();
  863. }else{
  864. frag = $o.ui.tabs.tab(tabName).body;
  865. }
  866. try{
  867. switch(typeof msg){
  868. case 'string':
  869. $(frag).append($('<li>').html(msg.htmlentities()));
  870. break;
  871. case 'object':
  872. if(typeof msg.html == 'undefined'){
  873. $(frag).append($('<li>').html('&lt;'+msg.user+'&gt;&nbsp;'+msg.text.htmlentities()));
  874. }else{
  875. $(frag).append(msg.html);
  876. }
  877. break;
  878. }
  879. }catch(e){event('Failed to add message','error')}
  880. if(tabs.length > 0){
  881. $($o.ui.tabs.tab(tabName).body || $o.ui.tabs.current().body).append(frag);
  882. }
  883. var scroll = [],i,html;
  884. for(i in tabs){
  885. html = '';
  886. $(tabs[i].body).children().each(function(){
  887. html += this.outerHTML;
  888. });
  889. scroll.push({
  890. name: tabs[i].name,
  891. body: html,
  892. date: new Date().toString()
  893. });
  894. }
  895. $.localStorage('tabs',scroll);
  896. if(typeof tabName == 'undefined' || tabName == $o.ui.tabs.current().name){
  897. $o.ui.tabs.select(selectedTab);
  898. }
  899. },
  900. event: function(event_name,message){
  901. event(message,event_name);
  902. }
  903. });
  904. String.prototype.htmlentities = function(){
  905. return this
  906. .replace(/&/g, '&amp;')
  907. .replace(/</g, '&lt;')
  908. .replace(/>/g, '&gt;')
  909. .replace(/\n/g,'<br/>')
  910. .replace(/\t/g, '&nbsp;&nbsp;&nbsp;&nbsp;')
  911. .replace(/\s/g, '&nbsp;')
  912. .replace(/"/g, '&quot;');
  913. };
  914. $(document).ready(function(){
  915. $.extend(settings,$.parseJSON($.localStorage('settings')));
  916. $.localStorage('settings',JSON.stringify(settings));
  917. $i = $('#input');
  918. $s = $('#send');
  919. $cl = $('#content-list');
  920. $c = $('#content');
  921. $tl = $('#tabs-list');
  922. $h = $('#head');
  923. $s.click(function(){
  924. if(!$s.hasClass('clicked')){
  925. $s.addClass('clicked');
  926. setTimeout(function(){
  927. $s.removeClass('clicked');
  928. },500);
  929. }
  930. $o.send($i.val());
  931. $i.val('');
  932. });
  933. $i.keypress(function(e){
  934. if(e.keyCode == 13){
  935. if(!$s.hasClass('clicked')){
  936. $s.addClass('clicked');
  937. setTimeout(function(){
  938. $s.removeClass('clicked');
  939. },500);
  940. }
  941. $o.send($i.val());
  942. $i.val('');
  943. }
  944. });
  945. $('#settings, #users').click(function(){
  946. $(this).addClass('open');
  947. $(this).children('.close-button').show();
  948. }).hover(function(){
  949. $(this).addClass('hovered');
  950. },function(){
  951. $(this).removeClass('hovered');
  952. }).children('.close-button').click(function(){
  953. $(this).parent().removeClass('open');
  954. $(this).hide();
  955. return false;
  956. }).hide();
  957. $('#users').hoverIntent({
  958. out: function(){
  959. $(this).removeClass('open');
  960. $(this).children('.close-button').hide();
  961. },
  962. timeout: 1000
  963. });
  964. $('#content').click(function(){
  965. $('#settings, #users, #head').removeClass('hovered').removeClass('open');
  966. $('#settings, #users').children('.close-button').hide()
  967. });
  968. $('.unselectable').attr('unselectable','on');
  969. $.contextMenu({
  970. selector: 'div.tab',
  971. items: {
  972. add: {
  973. name: 'New Tab',
  974. icon: 'add',
  975. callback: function(){
  976. $(this).contextMenu('hide');
  977. $o.ui.tabs.add(prompt('Channel'));
  978. }
  979. },
  980. s1: '',
  981. close: {
  982. name: 'Close',
  983. icon: 'delete',
  984. callback: function(){
  985. $(this).contextMenu('hide');
  986. $o.ui.tabs.remove($(this).data('id'));
  987. }
  988. }
  989. },
  990. zIndex: 99999,
  991. trigger: 'right'
  992. });
  993. $.contextMenu({
  994. selector: '#tabs-list',
  995. items: {
  996. add: {
  997. name: 'New Tab',
  998. icon: 'add',
  999. callback: function(){
  1000. $(this).contextMenu('hide');
  1001. $o.ui.tabs.add(prompt('channel'));
  1002. }
  1003. }
  1004. },
  1005. zIndex: 99999,
  1006. trigger: 'right'
  1007. });
  1008. $('#tabs-scroll-right').click(function(){
  1009. event('scroll right');
  1010. $tl.scrollTop(($tl.scrollTop()||0)+20);
  1011. if($tl.get(0).scrollHeight-20 == $tl.scrollTop()){
  1012. $('#tabs-scroll-right').addClass('disabled');
  1013. }
  1014. $('#tabs-scroll-left').removeClass('disabled');
  1015. });
  1016. $('#tabs-scroll-left').click(function(){
  1017. event('scroll left');
  1018. $tl.scrollTop(($tl.scrollTop()||0)-20);
  1019. if($tl.scrollTop() == 0){
  1020. $('#tabs-scroll-left').addClass('disabled');
  1021. }
  1022. $('#tabs-scroll-right').removeClass('disabled');
  1023. });
  1024. (function scrollup(){
  1025. $('#tabs-scroll-left').click();
  1026. if($tl.scrollTop() != 0){
  1027. setTimeout(scrollup,10);
  1028. }
  1029. })();
  1030. event('Date '+new Date,'ready');
  1031. $h.addClass('hovered');
  1032. setTimeout(function(){
  1033. $h.removeClass('hovered');
  1034. },1000);
  1035. $o.ui.render.settings();
  1036. if(settings.autoconnect){
  1037. $o.chat.connect();
  1038. }
  1039. });
  1040. window.io = null;
  1041. })(window,jQuery,io);
  1042. if (!Date.prototype.toISOString) {
  1043. Date.prototype.toISOString = function() {
  1044. function pad(n) { return n < 10 ? '0' + n : n }
  1045. return this.getUTCFullYear() + '-'
  1046. + pad(this.getUTCMonth() + 1) + '-'
  1047. + pad(this.getUTCDate()) + 'T'
  1048. + pad(this.getUTCHours()) + ':'
  1049. + pad(this.getUTCMinutes()) + ':'
  1050. + pad(this.getUTCSeconds()) + 'Z';
  1051. };
  1052. }
  1053. if(!String.prototype.isNumber){
  1054. String.prototype.isNumber = function(){return /^\d+$/.test(this);}
  1055. }