omnomirc.js 24 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010
  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. }
  297. }
  298. ],
  299. currentPlugin = 0,
  300. runHook = function(name){
  301. var i,hook,sanbox = {
  302. // Sandbox
  303. };
  304. for(i in hooks){
  305. hook = hooks[i];
  306. if(hook.hook == name){
  307. (new Function('with(sandbox){hook.fn();}'))();
  308. }
  309. }
  310. },
  311. version = '3.0',
  312. abbrDate = function(selector){
  313. if(settings.timestamp == 'fuzzy'){
  314. $(selector).timeago();
  315. }else{
  316. $(selector).each(function(){
  317. var timestamp = settings.timestamp,
  318. i,
  319. text='',
  320. date = new Date($(this).attr('title'));
  321. if(timestamp == 'exact'){
  322. timestamp = 'H:m:s t';
  323. }
  324. for(i=0;i<timestamp.length;i++){
  325. switch(timestamp[i]){
  326. case 'H':text+=((date.getHours()+11)%12)+1;break;
  327. case 'h':text+=date.getHours();break;
  328. case 'm':text+=(date.getMinutes()>9?'':'0')+date.getMinutes();break;
  329. case 's':text+=(date.getSeconds()>9?'':'0')+date.getSeconds();break;
  330. case 't':text+=(date.getHours()>11)?'pm':'am';break;
  331. default:text+=timestamp[i];
  332. }
  333. }
  334. $(this).text(text);
  335. }).timeago('dispose');
  336. }
  337. },
  338. socket,$i,$s,$h,$cl,$c,$tl,hht;
  339. $.extend($o,{
  340. version: function(){
  341. return version;
  342. },
  343. register: {
  344. theme: function(name){
  345. if(-1==$.inArray(properties.themes,name)){
  346. properties.themes.push(name);
  347. return true;
  348. }
  349. return false;
  350. },
  351. command: function(name,fn,help){
  352. if(-1==$.inArray(commands,name)){
  353. var o = {
  354. cmd: name,
  355. fn: fn
  356. };
  357. if(typeof help != 'undefined'){
  358. o.help = help;
  359. }
  360. commands.push(o);
  361. return true;
  362. }
  363. return false;
  364. },
  365. plugin: function(){
  366. // STUB
  367. },
  368. setting: function(name,defaultVal,validate,values){
  369. // STUB
  370. },
  371. hook: function(event,fn){
  372. }
  373. },
  374. hook: function(event,fn){
  375. $o.register.hook(event,fn);
  376. },
  377. ui: {
  378. render: {
  379. settings: function(){
  380. var name,setting,frag = document.createDocumentFragment(),item;
  381. for(name in settings){
  382. setting = $o.get(name,true);
  383. switch(setting.type){
  384. case 'select':
  385. item = $('<select>')
  386. .attr('id','setting_'+name)
  387. .change(function(){
  388. $o.set(this.id.substr(8),$(this).find(':selected').text(),false);
  389. });
  390. for(var i in setting.values){
  391. item.append(
  392. $('<option>')
  393. .text(setting.values[i])
  394. );
  395. }
  396. item.find(':contains('+setting.val+')').attr('selected','selected');
  397. break;
  398. case 'array':
  399. item = $('<input>')
  400. .attr({
  401. type: 'text',
  402. id: 'setting_'+name
  403. })
  404. .val(setting.val)
  405. .change(function(){
  406. $o.set(this.id.substr(8),$(this).val().split(','),false);
  407. });
  408. break;
  409. case 'boolean':
  410. item = $('<input>')
  411. .attr({
  412. type: 'checkbox',
  413. id: 'setting_'+name
  414. })
  415. .change(function(){
  416. $o.set(this.id.substr(8),$(this).is(':checked'),false);
  417. });
  418. if(setting.val){
  419. item.attr('checked','checked');
  420. }
  421. break;
  422. case 'number':
  423. case 'string':default:
  424. item = $('<input>')
  425. .attr({
  426. type: 'text',
  427. id: 'setting_'+name
  428. })
  429. .val(setting.val)
  430. .change(function(){
  431. $o.set(this.id.substr(8),$(this).val(),false);
  432. });
  433. }
  434. $(frag).append(
  435. $('<li>')
  436. .addClass('row')
  437. .append(
  438. $('<span>')
  439. .text(name)
  440. .addClass('cell')
  441. )
  442. .append(
  443. $('<span>')
  444. .append(item)
  445. .addClass('cell')
  446. )
  447. );
  448. }
  449. $('#settings-list').html(frag);
  450. },
  451. users: function(){
  452. event('Rendering userlist');
  453. var $ul = $('#user-list').html(''),
  454. i,
  455. names = $o.ui.tabs.current().users;
  456. for(i in names){
  457. $ul.append(
  458. $('<li>').text(names[i])
  459. );
  460. }
  461. },
  462. tab: function(){
  463. $cl.html($($o.ui.tabs.current().body).clone());
  464. },
  465. tablist: function(){
  466. $tl.html('');
  467. var i,tab;
  468. for(i in tabs){
  469. tab = $o.ui.tabs.obj(i);
  470. if(i==selectedTab){
  471. tab.addClass('clicked');
  472. $('#title').text(tabs[i].name);
  473. $('#topic').text(tabs[i].topic);
  474. }
  475. $tl.append(tab);
  476. }
  477. if($tl.get(0).scrollHeight-20 != $tl.scrollTop()){
  478. $('#tabs-scroll-right').removeClass('disabled');
  479. }
  480. if($tl.scrollTop() != 0){
  481. $('#tabs-scroll-left').removeClass('disabled');
  482. }
  483. }
  484. },
  485. tabs: {
  486. add: function(name){
  487. event('Tab added: '+name);
  488. if(!(function(){
  489. for(var i in tabs){
  490. if(name==tabs[i].name){
  491. return true;
  492. }
  493. }
  494. return false;
  495. })()){
  496. var scroll = $.localStorage('tabs'),
  497. i,
  498. frag = document.createDocumentFragment(),
  499. id = tabs.length;
  500. for(i in scroll){
  501. if(scroll[i].name == name){
  502. scroll[i].body = $(scroll[i].body).slice(-10);
  503. $(frag)
  504. .append(scroll[i].body)
  505. .append(
  506. $('<li>').html('<span class="to_remove">-- loaded old scrollback for '+scroll[i].date+' --</span>')
  507. )
  508. .children()
  509. .children('.remove')
  510. .remove();
  511. $(frag)
  512. .children()
  513. .children('.to_remove')
  514. .removeClass('to_remove')
  515. .addClass('remove');
  516. event('loading old tab scrollback for '+name+' last saved +'+scroll[i].date);
  517. }
  518. }
  519. tabs.push({
  520. name: name,
  521. body: frag,
  522. date: new Date(),
  523. send: function(msg){
  524. $o.chat.send(msg,$o.ui.tabs.tab(id).name);
  525. },
  526. close: function(){
  527. $o.ui.tabs.remove(id);
  528. },
  529. users: [],
  530. names: function(){
  531. emit('names',{
  532. name: $o.ui.tabs.tab(id).name
  533. });
  534. },
  535. select: function(){
  536. $o.ui.tabs.select(id);
  537. },
  538. clear: function(){
  539. $cl.html('');
  540. $o.ui.tabs.tab(id).body = document.createDocumentFragment();
  541. emit('echo',{
  542. room: $o.ui.tabs.tab(id).name,
  543. message: 'messages cleared',
  544. from: 0
  545. });
  546. }
  547. });
  548. $tl.append($o.ui.tabs.obj(id));
  549. $o.ui.render.tablist();
  550. $o.ui.render.users();
  551. }else{
  552. event('Attempted to add an existing tab');
  553. }
  554. },
  555. remove: function(name){
  556. if(typeof name == 'number'){
  557. name = tabs[name].name;
  558. }
  559. for(var id=0;id<tabs.length;id++){
  560. if($o.ui.tabs.tab(id).name == name){
  561. event('Tab removed: '+$o.ui.tabs.tab(id).name);
  562. emit('part',{
  563. name: $o.ui.tabs.tab(id).name
  564. });
  565. tabs.splice(id,1);
  566. if(selectedTab==id&&selectedTab>0){
  567. selectedTab--;
  568. }
  569. break;
  570. }
  571. }
  572. $o.ui.render.tablist();
  573. $cl.html($o.ui.tabs.current().body);
  574. $o.ui.render.users();
  575. },
  576. selected: function(){
  577. return selectedTab;
  578. },
  579. idForName: function(name){
  580. for(var i in tabs){
  581. if(tabs[i].name == name){
  582. return i;
  583. }
  584. }
  585. return false;
  586. },
  587. tab: function(id){
  588. if(typeof id == 'string' && !id.isNumber()){
  589. id = $o.ui.tabs.idForName(id);
  590. if(!id) return false;
  591. }
  592. return typeof tabs[id] == 'undefined'?false:tabs[id];
  593. },
  594. dom: function(id){
  595. if(typeof id == 'string' && !id.isNumber()){
  596. id = $o.ui.tabs.idForName(id);
  597. if(!id) return false;
  598. }
  599. return typeof tabs[id] == 'undefined'?false:tabs[id].body;
  600. },
  601. obj: function(id){
  602. if(typeof id !== 'undefined'){
  603. if(typeof id == 'string' && !id.isNumber()){
  604. id = $o.ui.tabs.idForName(id);
  605. if(!id) return;
  606. }
  607. return $('<div>')
  608. .addClass('tab')
  609. .text($o.ui.tabs.tab(id).name)
  610. .mouseup(function(e){
  611. switch(e.which){
  612. case 1: // RMB
  613. if($(this).data('id')!=selectedTab){
  614. $o.ui.tabs.select($(this).data('id'));
  615. return prevent(e);
  616. }
  617. break;
  618. case 2: // MMB
  619. $(this).children('span.close-button').click();
  620. return prevent(e);
  621. break;
  622. case 3: // LMB
  623. return prevent(e);
  624. break;
  625. default:
  626. return prevent(e);
  627. }
  628. })
  629. .append(
  630. $('<span>')
  631. .addClass('close-button')
  632. .click(function(){
  633. $o.ui.tabs.remove(id);
  634. return false;
  635. })
  636. .css({
  637. 'position': 'absolute',
  638. 'background-color': 'inherit',
  639. 'top': 0,
  640. 'right': 0
  641. })
  642. .html('&times;')
  643. )
  644. .data('id',id);
  645. }
  646. },
  647. select: function(id){
  648. if(typeof id == 'string' && !id.isNumber()){
  649. id = $o.ui.tabs.idForName(id);
  650. if(!id) return false;
  651. }
  652. event(id+' '+$o.ui.tabs.tab(id).name,'tab_select');
  653. if(id<tabs.length&&id>=0){
  654. selectedTab=id;
  655. }
  656. $tl.children('.clicked').removeClass('clicked');
  657. $($tl.children().get(id)).addClass('clicked');
  658. $('#title').text($o.ui.tabs.tab(id).name);
  659. $('#topic').text($o.ui.tabs.tab(id).topic);
  660. $cl.html($($o.ui.tabs.tab(id).body).clone());
  661. abbrDate('abbr.date');
  662. $o.ui.render.users();
  663. setTimeout(function scrollContent(){
  664. if($c.scrollTop() < $c[0].scrollHeight){
  665. $c.scrollTop($c.scrollTop()+1);
  666. setTimeout(scrollContent,settings.scrollspeed);
  667. }else{
  668. event('scrolling stopped');
  669. }
  670. },settings.scrollspeed);
  671. },
  672. current: function(){
  673. if(tabs.length > 0 && tabs.length > selectedTab){
  674. return tabs[selectedTab];
  675. }else{
  676. return {
  677. name: '',
  678. body: document.createDocumentFragment(),
  679. date: new Date(),
  680. send: noop,
  681. close: noop,
  682. users: [],
  683. names: noop,
  684. select: noop,
  685. clear: noop
  686. }
  687. }
  688. }
  689. },
  690. },
  691. chat: {
  692. connect: function(server){
  693. if($o.chat.connected()){
  694. $o.disconnect();
  695. }
  696. if(typeof server == 'undefined'){
  697. server = settings.server;
  698. }
  699. socket = window.socket = io.connect(server);
  700. for(var i in handles){
  701. socket.on(handles[i].on,handles[i].fn);
  702. }
  703. $o.chat.auth();
  704. },
  705. disconnect: function(){
  706. if($o.chat.connected()){
  707. socket.disconnect();
  708. }
  709. },
  710. connected: function(){
  711. return typeof socket == 'undefined'?false:properties.connected;
  712. },
  713. send: function(msg,room){
  714. if(typeof room == 'undefined'){
  715. room = $o.ui.tabs.current().name;
  716. }
  717. if(msg !== ''){
  718. if(msg[0] == '/' && msg[1] != '/'){
  719. var args = msg.split(' '),
  720. cmd = args[0].substr(1),
  721. i;
  722. event(msg,'command');
  723. for(i in commands){
  724. if(commands[i].cmd == cmd){
  725. commands[i].fn(args);
  726. return;
  727. }
  728. }
  729. $o.msg(cmd+' is not a valid command.');
  730. }else{
  731. event(msg,'send');
  732. emit('message',{
  733. message: msg,
  734. room: room,
  735. from: properties.nick
  736. });
  737. }
  738. }
  739. },
  740. auth: function(){
  741. if(settings.nick == ''){
  742. $o.set('nick','User');
  743. return;
  744. }
  745. emit('auth',{
  746. nick: settings.nick
  747. // TODO - send authorization info
  748. });
  749. }
  750. },
  751. get: function(name,formatted){
  752. if(typeof formatted == 'undefined'){
  753. return exists(settings[name])?settings[name]:false;
  754. }else{
  755. var val = $o.get(name),
  756. type,
  757. values = false;
  758. switch(name){
  759. case 'theme':
  760. type = 'select';
  761. values = properties.themes;
  762. break;
  763. case 'autojoin':type = 'array';break;
  764. case 'timestamp':type = 'string';break;
  765. default:
  766. type = typeof val;
  767. }
  768. return {
  769. type: type,
  770. val: val,
  771. values: values,
  772. name: name
  773. };
  774. }
  775. },
  776. set: function(name,value,render){
  777. if(exists(settings[name])){
  778. settings[name] = value;
  779. $.localStorage('settings',JSON.stringify(settings));
  780. switch(name){
  781. case 'timestamp':
  782. abbrDate('abbr.date');
  783. if(settings.timestamp == ''){
  784. $('.date_cell').css('visibility','hidden');
  785. }else{
  786. $('.date_cell').css('visibility','visible');
  787. }
  788. break;
  789. case 'nick':
  790. $o.chat.auth();
  791. break;
  792. }
  793. if(typeof render == 'undefined'){
  794. $o.ui.render.settings();
  795. }
  796. return true;
  797. }else{
  798. return false;
  799. }
  800. },
  801. prop: function(name){
  802. return exists(properties[name])?properties[name]:null;
  803. },
  804. send: function(msg){
  805. $o.chat.send(msg);
  806. },
  807. msg: function(msg,tabName){
  808. var frag;
  809. if(typeof tabName == 'undefined' || tabName == $o.ui.tabs.current().name || typeof $o.ui.tabs.tab(tabName).body == 'undefined'){
  810. frag = document.createDocumentFragment();
  811. }else{
  812. frag = $o.ui.tabs.tab(tabName).body;
  813. }
  814. try{
  815. switch(typeof msg){
  816. case 'string':
  817. $(frag).append($('<li>').html(msg.htmlentities()));
  818. break;
  819. case 'object':
  820. if(typeof msg.html == 'undefined'){
  821. $(frag).append($('<li>').html('&lt;'+msg.user+'&gt;&nbsp;'+msg.text.htmlentities()));
  822. }else{
  823. $(frag).append(msg.html);
  824. }
  825. break;
  826. }
  827. }catch(e){event('Failed to add message','error')}
  828. if(tabs.length > 0){
  829. $($o.ui.tabs.tab(tabName).body || $o.ui.tabs.current().body).append(frag);
  830. }
  831. var scroll = [],i,html;
  832. for(i in tabs){
  833. html = '';
  834. $(tabs[i].body).children().each(function(){
  835. html += this.outerHTML;
  836. });
  837. scroll.push({
  838. name: tabs[i].name,
  839. body: html,
  840. date: new Date().toString()
  841. });
  842. }
  843. $.localStorage('tabs',scroll);
  844. if(typeof tabName == 'undefined' || tabName == $o.ui.tabs.current().name){
  845. $o.ui.tabs.select(selectedTab);
  846. }
  847. },
  848. event: function(event_name,message){
  849. event(message,event_name);
  850. }
  851. });
  852. String.prototype.htmlentities = function(){
  853. return this
  854. .replace(/&/g, '&amp;')
  855. .replace(/</g, '&lt;')
  856. .replace(/>/g, '&gt;')
  857. .replace(/\n/g,'<br/>')
  858. .replace(/\t/g, '&nbsp;&nbsp;&nbsp;&nbsp;')
  859. .replace(/\s/g, '&nbsp;')
  860. .replace(/"/g, '&quot;');
  861. };
  862. $(document).ready(function(){
  863. $.extend(settings,$.parseJSON($.localStorage('settings')));
  864. $.localStorage('settings',JSON.stringify(settings));
  865. $i = $('#input');
  866. $s = $('#send');
  867. $cl = $('#content-list');
  868. $c = $('#content');
  869. $tl = $('#tabs-list');
  870. $h = $('#head');
  871. $s.click(function(){
  872. if(!$s.hasClass('clicked')){
  873. $s.addClass('clicked');
  874. setTimeout(function(){
  875. $s.removeClass('clicked');
  876. },500);
  877. }
  878. $o.send($i.val());
  879. $i.val('');
  880. });
  881. $i.keypress(function(e){
  882. if(e.keyCode == 13){
  883. if(!$s.hasClass('clicked')){
  884. $s.addClass('clicked');
  885. setTimeout(function(){
  886. $s.removeClass('clicked');
  887. },500);
  888. }
  889. $o.send($i.val());
  890. $i.val('');
  891. }
  892. });
  893. $('#settings, #users').click(function(){
  894. $(this).addClass('open');
  895. $(this).children('.close-button').show();
  896. }).hover(function(){
  897. $(this).addClass('hovered');
  898. },function(){
  899. $(this).removeClass('hovered');
  900. }).children('.close-button').click(function(){
  901. $(this).parent().removeClass('open');
  902. $(this).hide();
  903. return false;
  904. }).hide();
  905. $('#users').hoverIntent({
  906. out: function(){
  907. $(this).removeClass('open');
  908. $(this).children('.close-button').hide();
  909. },
  910. timeout: 1000
  911. });
  912. $('#content').click(function(){
  913. $('#settings, #users, #head').removeClass('hovered').removeClass('open');
  914. $('#settings, #users').children('.close-button').hide()
  915. });
  916. $('.unselectable').attr('unselectable','on');
  917. $.contextMenu({
  918. selector: 'div.tab',
  919. items: {
  920. add: {
  921. name: 'New Tab',
  922. icon: 'add',
  923. callback: function(){
  924. $(this).contextMenu('hide');
  925. $o.ui.tabs.add(prompt('Channel'));
  926. }
  927. },
  928. s1: '',
  929. close: {
  930. name: 'Close',
  931. icon: 'delete',
  932. callback: function(){
  933. $(this).contextMenu('hide');
  934. $o.ui.tabs.remove($(this).data('id'));
  935. }
  936. }
  937. },
  938. zIndex: 99999,
  939. trigger: 'right'
  940. });
  941. $.contextMenu({
  942. selector: '#tabs-list',
  943. items: {
  944. add: {
  945. name: 'New Tab',
  946. icon: 'add',
  947. callback: function(){
  948. $(this).contextMenu('hide');
  949. $o.ui.tabs.add(prompt('channel'));
  950. }
  951. }
  952. },
  953. zIndex: 99999,
  954. trigger: 'right'
  955. });
  956. $('#tabs-scroll-right').click(function(){
  957. event('scroll right');
  958. $tl.scrollTop(($tl.scrollTop()||0)+20);
  959. if($tl.get(0).scrollHeight-20 == $tl.scrollTop()){
  960. $('#tabs-scroll-right').addClass('disabled');
  961. }
  962. $('#tabs-scroll-left').removeClass('disabled');
  963. });
  964. $('#tabs-scroll-left').click(function(){
  965. event('scroll left');
  966. $tl.scrollTop(($tl.scrollTop()||0)-20);
  967. if($tl.scrollTop() == 0){
  968. $('#tabs-scroll-left').addClass('disabled');
  969. }
  970. $('#tabs-scroll-right').removeClass('disabled');
  971. });
  972. (function scrollup(){
  973. $('#tabs-scroll-left').click();
  974. if($tl.scrollTop() != 0){
  975. setTimeout(scrollup,10);
  976. }
  977. })();
  978. event('Date '+new Date,'ready');
  979. $h.addClass('hovered');
  980. setTimeout(function(){
  981. $h.removeClass('hovered');
  982. },1000);
  983. $o.ui.render.settings();
  984. if(settings.autoconnect){
  985. $o.chat.connect();
  986. }
  987. });
  988. window.io = null;
  989. })(window,jQuery,io);
  990. if (!Date.prototype.toISOString) {
  991. Date.prototype.toISOString = function() {
  992. function pad(n) { return n < 10 ? '0' + n : n }
  993. return this.getUTCFullYear() + '-'
  994. + pad(this.getUTCMonth() + 1) + '-'
  995. + pad(this.getUTCDate()) + 'T'
  996. + pad(this.getUTCHours()) + ':'
  997. + pad(this.getUTCMinutes()) + ':'
  998. + pad(this.getUTCSeconds()) + 'Z';
  999. };
  1000. }
  1001. if(!String.prototype.isNumber){
  1002. String.prototype.isNumber = function(){return /^\d+$/.test(this);}
  1003. }