omnomirc.js 24 KB

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