ui.js 37 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065
  1. var
  2. /** @type {SlackMessage|null} */
  3. REPLYING_TO = null
  4. /** @type {SlackMessage|null} */
  5. ,EDITING = null
  6. /**
  7. * Minimum time between 2 notifications (ms)
  8. * @const
  9. * @type {number}
  10. **/
  11. ,NOTIFICATION_COOLDOWN = 30 * 1000 //30 sec
  12. /**
  13. * Maximum time the notification will stay visible (ms)
  14. * @const
  15. * @type {number}
  16. **/
  17. ,NOTIFICATION_DELAY = 5 * 1000 // 5 sec
  18. /** @type {number} */
  19. ,lastNotificationSpawn = 0
  20. ;
  21. /**
  22. * @return {Element!}
  23. **/
  24. function createTypingDisplay() {
  25. var dom = document.createElement("span")
  26. ,dot1 = document.createElement("span")
  27. ,dot2 = document.createElement("span")
  28. ,dot3 = document.createElement("span");
  29. dom.className = R.klass.typing.container;
  30. dot1.className = R.klass.typing.dot1;
  31. dot2.className = R.klass.typing.dot2;
  32. dot3.className = R.klass.typing.dot3;
  33. dot1.textContent = dot2.textContent = dot3.textContent = '.';
  34. dom.appendChild(dot1);
  35. dom.appendChild(dot2);
  36. dom.appendChild(dot3);
  37. return dom;
  38. }
  39. /**
  40. * @param {SlackChan|SlackGroup} chan
  41. * @return {Element}
  42. **/
  43. function createChanListItem(chan) {
  44. var dom = document.createElement("li")
  45. ,link = document.createElement("a");
  46. dom.id = chan.id;
  47. link.href = '#' +chan.id;
  48. if (chan.id[0] === 'D')
  49. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeDirect;
  50. else if (chan.id[0] === 'G')
  51. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeGroup;
  52. else if (chan.id[0] === 'C')
  53. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeChannel;
  54. if (SELECTED_ROOM === chan)
  55. dom.classList.add(R.klass.selected);
  56. link.textContent = chan.name;
  57. dom.appendChild(createTypingDisplay());
  58. dom.appendChild(link);
  59. if (UNREAD_CHANS[chan.id]) {
  60. if (UNREAD_CHANS[chan.id].hl)
  61. dom.classList.add(R.klass.unreadHi);
  62. if (UNREAD_CHANS[chan.id].unread)
  63. dom.classList.add(R.klass.unread);
  64. }
  65. return dom;
  66. }
  67. /**
  68. * @param {SlackIms} ims
  69. * @return {Element}
  70. **/
  71. function createImsListItem(ims) {
  72. var dom = document.createElement("li")
  73. ,link = document.createElement("a");
  74. dom.id = ims.id;
  75. link.href = '#' +ims.id;
  76. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeDirect;
  77. link.textContent = ims.user.name;
  78. dom.appendChild(createTypingDisplay());
  79. dom.appendChild(link);
  80. if (!ims.user.presence)
  81. dom.classList.add(R.klass.presenceAway);
  82. if (SELECTED_ROOM === ims)
  83. dom.classList.add(R.klass.selected);
  84. if (UNREAD_CHANS[ims.id]) {
  85. if (UNREAD_CHANS[ims.id].hl)
  86. dom.classList.add(R.klass.unreadHi);
  87. if (UNREAD_CHANS[ims.id].unread)
  88. dom.classList.add(R.klass.unread);
  89. }
  90. return dom;
  91. }
  92. function onContextUpdated() {
  93. var chanListFram = document.createDocumentFragment()
  94. ,sortedChans = SLACK.context.self ? Object.keys(SLACK.context.self.channels) : [];
  95. sortedChans.sort(function(a, b) {
  96. if (a[0] !== b[0]) {
  97. return a[0] - b[0];
  98. }
  99. return SLACK.context.getChannel(a).name.localeCompare(SLACK.context.getChannel(b).name);
  100. });
  101. sortedChans.forEach(function(chanId) {
  102. var chan =
  103. /**
  104. * SortedChan does not contains ims ids
  105. * @type {SlackChan|SlackGroup}
  106. **/
  107. (SLACK.context.getChannel(chanId));
  108. if (!chan.archived) {
  109. var chanListItem = createChanListItem(chan);
  110. if (chanListItem)
  111. chanListFram.appendChild(chanListItem);
  112. }
  113. });
  114. var sortedUsers = SLACK.context.users ? Object.keys(SLACK.context.users) : [];
  115. sortedUsers.sort(function(a, b) {
  116. return SLACK.context.users[a].name.localeCompare(SLACK.context.users[b].name);
  117. });
  118. sortedUsers.forEach(function(userId) {
  119. var user = SLACK.context.getMember(userId);
  120. if (!user.deleted) {
  121. var ims = user.ims
  122. ,imsListItem = createImsListItem(ims);
  123. if (imsListItem) {
  124. chanListFram.appendChild(imsListItem);
  125. }
  126. }
  127. });
  128. document.getElementById(R.id.chanList).textContent = "";
  129. document.getElementById(R.id.chanList).appendChild(chanListFram);
  130. setRoomFromHashBang();
  131. updateTitle();
  132. createContextBackground(function(imgData) {
  133. document.getElementById(R.id.context).style.backgroundImage = 'url(' +imgData +')';
  134. });
  135. }
  136. function onTypingUpdated() {
  137. var typing = SLACK.context.typing;
  138. for (var chanId in SLACK.context.self.channels) {
  139. if (!SLACK.context.self.channels[chanId].archived) {
  140. var dom = document.getElementById(chanId);
  141. if (typing[chanId])
  142. dom.classList.add(R.klass.chatList.typing);
  143. else
  144. dom.classList.remove(R.klass.chatList.typing);
  145. }
  146. }
  147. for (var userId in SLACK.context.users) {
  148. var ims = SLACK.context.users[userId].ims;
  149. if (ims && !ims.archived) {
  150. var dom = document.getElementById(ims.id);
  151. if (typing[ims.id])
  152. dom.classList.add(R.klass.chatList.typing);
  153. else
  154. dom.classList.remove(R.klass.chatList.typing);
  155. }
  156. }
  157. }
  158. function onNetworkStateUpdated(isNetworkWorking) {
  159. isNetworkWorking ? document.body.classList.remove(R.klass.noNetwork) : document.body.classList.add(R.klass.noNetwork);
  160. updateTitle();
  161. }
  162. function onRoomSelected() {
  163. var name = SELECTED_ROOM.name || (SELECTED_ROOM.user ? SELECTED_ROOM.user.name : undefined);
  164. if (!name) {
  165. var members = [];
  166. for (var i in SELECTED_ROOM.members) {
  167. members.push(SELECTED_ROOM.members[i].name);
  168. }
  169. name = members.join(", ");
  170. }
  171. var roomLi = document.getElementById(SELECTED_ROOM.id);
  172. document.getElementById(R.id.currentRoom.title).textContent = name;
  173. onRoomUpdated();
  174. focusInput();
  175. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  176. markRoomAsRead(SELECTED_ROOM);
  177. if (REPLYING_TO) {
  178. REPLYING_TO = null;
  179. onReplyingToUpdated();
  180. }
  181. if (EDITING) {
  182. EDITING = null;
  183. onReplyingToUpdated();
  184. }
  185. }
  186. function onReplyingToUpdated() {
  187. if (REPLYING_TO) {
  188. document.body.classList.add(R.klass.replyingTo);
  189. var domParent = document.getElementById(R.id.message.replyTo)
  190. ,closeLink = document.createElement("a");
  191. closeLink.addEventListener("click", function() {
  192. REPLYING_TO = null;
  193. onReplyingToUpdated();
  194. });
  195. closeLink.className = R.klass.msg.replyTo.close;
  196. closeLink.textContent = 'x';
  197. domParent.textContent = "";
  198. domParent.appendChild(closeLink);
  199. domParent.appendChild(createMessageDom("reply_" +SELECTED_ROOM.id, REPLYING_TO, true));
  200. focusInput();
  201. } else {
  202. document.body.classList.remove(R.klass.replyingTo);
  203. document.getElementById(R.id.message.replyTo).textContent = "";
  204. focusInput();
  205. }
  206. }
  207. function onEditingUpdated() {
  208. if (EDITING) {
  209. document.body.classList.add(R.klass.replyingTo);
  210. var domParent = document.getElementById(R.id.message.replyTo)
  211. ,closeLink = document.createElement("a");
  212. closeLink.addEventListener("click", function() {
  213. EDITING = null;
  214. onEditingUpdated();
  215. });
  216. closeLink.className = R.klass.msg.replyTo.close;
  217. closeLink.textContent = 'x';
  218. domParent.textContent = "";
  219. domParent.appendChild(closeLink);
  220. domParent.appendChild(createMessageDom("edit_" +SELECTED_ROOM.id, EDITING, true));
  221. document.getElementById(R.id.message.input).value = EDITING.text;
  222. focusInput();
  223. } else {
  224. document.body.classList.remove(R.klass.replyingTo);
  225. document.getElementById(R.id.message.replyTo).textContent = "";
  226. focusInput();
  227. }
  228. }
  229. /**
  230. * @param {string} chanId
  231. * @param {string} msgId
  232. * @param {string} reaction
  233. * @param {Array.<string>} users
  234. * @return {Element|null}
  235. **/
  236. function createReactionDom(chanId, msgId, reaction, users) {
  237. var emojiDom = makeEmojiDom(reaction);
  238. if (emojiDom) {
  239. var dom = document.createElement("li")
  240. ,a = document.createElement("a")
  241. ,emojiContainer = document.createElement("span")
  242. ,userList = document.createElement("span")
  243. ,userNames = [];
  244. for (var i =0, nbUser = users.length; i < nbUser; i++) {
  245. var user = SLACK.context.getMember(users[i]);
  246. if (user)
  247. userNames.push(user.name);
  248. }
  249. userNames.sort();
  250. userList.textContent = userNames.join(", ");
  251. emojiContainer.appendChild(emojiDom);
  252. emojiContainer.className = R.klass.emoji.small;
  253. a.href = "javascript:toggleReaction('" +chanId +"', '" +msgId +"', '" +reaction +"')";
  254. a.appendChild(emojiContainer);
  255. a.appendChild(userList);
  256. dom.className = R.klass.msg.reactions.item;
  257. dom.appendChild(a);
  258. return dom;
  259. }
  260. return null;
  261. }
  262. /**
  263. * @param {string} chanId
  264. * @param {string} msgId
  265. * @param {string} reaction
  266. **/
  267. window['toggleReaction'] = function(chanId, msgId, reaction) {
  268. var hist = SLACK.history[chanId];
  269. if (!hist)
  270. return;
  271. var msg = hist.getMessageById(msgId);
  272. if (msg) {
  273. if (msg.hasReactionForUser(reaction, SLACK.context.self.id)) {
  274. removeReaction(chanId, msgId, reaction);
  275. } else {
  276. addReaction(chanId, msgId, reaction);
  277. }
  278. }
  279. };
  280. /**
  281. * @param {string} channelId
  282. * @param {SlackMessage} msg
  283. * @param {boolean=} skipAttachment
  284. * @return {Element}
  285. **/
  286. function doCreateMessageDom(channelId, msg, skipAttachment) {
  287. var dom = document.createElement("div")
  288. ,msgBlock = document.createElement("div")
  289. ,ts = document.createElement("div")
  290. ,text = document.createElement("div")
  291. ,authorImg = document.createElement("img")
  292. ,authorName = document.createElement("span")
  293. ,hover = document.createElement("ul")
  294. ,hoverReply = document.createElement("li")
  295. ,attachments = document.createElement("ul")
  296. ,reactions = document.createElement("ul")
  297. ,sender = SLACK.context.getMember(msg.userId);
  298. dom.id = channelId +"_" +msg.ts;
  299. dom.className = R.klass.msg.item;
  300. ts.className = R.klass.msg.ts;
  301. text.className = R.klass.msg.msg;
  302. authorImg.className = R.klass.msg.authorAvatar;
  303. authorName.className = R.klass.msg.authorname;
  304. hover.className = R.klass.msg.hover.container;
  305. hoverReply.className = R.klass.msg.hover.reply;
  306. ts.innerHTML = locale.formatDate(msg.ts);
  307. text.innerHTML = formatSlackText(msg.text);
  308. authorName.textContent = sender ? sender.name : (msg.username || "?");
  309. authorImg.src = sender ? sender.icons.image_48 : "";
  310. hover.appendChild(hoverReply);
  311. if ('makeEmoji' in window) {
  312. var hoverReaction = document.createElement("li")
  313. ,domReply = window['makeEmoji']("arrow_heading_down")
  314. ,domReaction = window['makeEmoji']("smile")
  315. ,domEdit = window['makeEmoji']("pencil2")
  316. ,domRemove = window['makeEmoji']("x");
  317. hoverReaction.className = R.klass.msg.hover.reaction;
  318. if (domReaction) {
  319. hoverReaction.classList.add(R.klass.emoji.small);
  320. hoverReaction.appendChild(domReaction);
  321. } else {
  322. hoverReaction.style.backgroundImage = 'url("smile.svg")';
  323. }
  324. if (domReply) {
  325. hoverReply.classList.add(R.klass.emoji.small);
  326. hoverReply.appendChild(domReply);
  327. } else {
  328. hoverReply.style.backgroundImage = 'url("repl.svg")';
  329. }
  330. hover.appendChild(hoverReaction);
  331. if (msg.userId === SLACK.context.self.id) {
  332. var hoverEdit = document.createElement("li");
  333. hoverEdit.className = R.klass.msg.hover.edit;
  334. if (domEdit)
  335. hoverEdit.classList.add(R.klass.emoji.small);
  336. else
  337. hoverEdit.style.backgroundImage = 'url("edit.svg")';
  338. hoverEdit.appendChild(domEdit);
  339. hover.appendChild(hoverEdit);
  340. var hoverRemove = document.createElement("li");
  341. hoverRemove.className = R.klass.msg.hover.remove;
  342. if (domRemove)
  343. hoverRemove.classList.add(R.klass.emoji.small);
  344. else
  345. hoverRemove.style.backgroundImage = 'url("remove.svg")';
  346. hoverRemove.appendChild(domRemove);
  347. hover.appendChild(hoverRemove);
  348. }
  349. } else {
  350. hoverReply.style.backgroundImage = 'url("repl.svg")';
  351. if (msg.userId === SLACK.context.self.id) {
  352. var hoverEdit = document.createElement("li");
  353. hoverEdit.className = R.klass.msg.hover.edit;
  354. hoverEdit.style.backgroundImage = 'url("edit.svg")';
  355. hover.appendChild(hoverEdit);
  356. var hoverRemove = document.createElement("li")
  357. hoverRemove.className = R.klass.msg.hover.remove;
  358. hoverRemove.style.backgroundImage = 'url("remove.svg")';
  359. hover.appendChild(hoverRemove);
  360. }
  361. }
  362. dom.appendChild(authorImg);
  363. msgBlock.appendChild(authorName);
  364. msgBlock.appendChild(text);
  365. msgBlock.appendChild(ts);
  366. msgBlock.appendChild(attachments);
  367. if (msg.edited) {
  368. var edited = document.createElement("div");
  369. edited.textContent = locale.edited;
  370. edited.className = R.klass.msg.edited;
  371. msgBlock.appendChild(edited);
  372. }
  373. msgBlock.appendChild(reactions);
  374. msgBlock.className = R.klass.msg.content;
  375. attachments.className = R.klass.msg.attachment.list;
  376. reactions.className = R.klass.msg.reactions.container;
  377. if (skipAttachment !== true) {
  378. if (msg.reactions) for (var reaction in msg.reactions) {
  379. var reac = createReactionDom(channelId, msg.id, reaction, msg.reactions[reaction]);
  380. reac && reactions.appendChild(reac);
  381. }
  382. msg.attachments.forEach(function(attachment) {
  383. var domAttachment = createAttachmentDom(channelId, msg, attachment);
  384. if (domAttachment)
  385. attachments.appendChild(domAttachment);
  386. });
  387. }
  388. dom.appendChild(msgBlock);
  389. dom.appendChild(hover);
  390. return dom;
  391. }
  392. /**
  393. * Try to resolve emoji from customized context
  394. * @param {string} emoji
  395. * @return {Element|string}
  396. **/
  397. function tryGetCustomEmoji(emoji) {
  398. var loop = {};
  399. while (!loop[emoji]) {
  400. var emojisrc= SLACK.context.emojis[emoji];
  401. if (emojisrc) {
  402. if (emojisrc.substr(0, 6) == "alias:") {
  403. loop[emoji] = true;
  404. emoji = emojisrc.substr(6);
  405. } else {
  406. var dom = document.createElement("span");
  407. dom.className = R.klass.emoji.custom +' ' +R.klass.emoji.emoji;
  408. dom.style.backgroundImage = "url('" +emojisrc +"')";
  409. return dom;
  410. }
  411. }
  412. return emoji; // Emoji not found, fallback to std emoji
  413. }
  414. return emoji; //loop detected, return first emoji
  415. }
  416. function makeEmojiDom(emojiCode) {
  417. var emoji = tryGetCustomEmoji(emojiCode);
  418. if (typeof emoji === "string" && "makeEmoji" in window)
  419. emoji = window['makeEmoji'](emoji);
  420. return typeof emoji === "string" ? null : emoji;
  421. }
  422. /**
  423. * replace all :emoji: codes with corresponding image
  424. * @param {string} inputString
  425. * @return {string}
  426. **/
  427. function formatEmojis(inputString) {
  428. return inputString.replace(/:([^ \t:]+):/g, function(returnFailed, emoji) {
  429. var emojiDom = makeEmojiDom(emoji);
  430. if (emojiDom) {
  431. var domParent = document.createElement("span");
  432. domParent.className = returnFailed === inputString ? R.klass.emoji.medium : R.klass.emoji.small;
  433. domParent.appendChild(emojiDom);
  434. return domParent.outerHTML;
  435. }
  436. return returnFailed;
  437. });
  438. }
  439. /**
  440. * @param {string} fullText
  441. * @return {string}
  442. **/
  443. function formatSlackText(fullText) {
  444. var msgContents = fullText.split(/\r?\n/g);
  445. for (var msgContentIndex=0, nbMsgContents = msgContents.length; msgContentIndex < nbMsgContents; msgContentIndex++) {
  446. var msgContent = msgContents[msgContentIndex].trim()
  447. ,_msgContent = ""
  448. ,currentMods = {}
  449. ,quote = false
  450. ,i =0
  451. msgContent = msgContent.replace(new RegExp('<([@#]?)([^>]*)>', 'g'),
  452. function(matched, type, entity) {
  453. var sub = entity.split('|');
  454. if (type === '@') {
  455. if (!sub[1]) {
  456. var user = SLACK.context.getMember(sub[0]);
  457. sub[1] = user ? ('@' +user.name) : locale.unknownMember;
  458. } else if ('@' !== sub[1][0]) {
  459. sub[1] = '@' +sub[1];
  460. }
  461. sub[0] = '#' +sub[0];
  462. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
  463. } else if (type === '#') {
  464. if (!sub[1]) {
  465. var chan = SLACK.context.getChannel(sub[0]);
  466. sub[1] = chan ? ('#' +chan.name) : locale.unknownChannel;
  467. } else if ('#' !== sub[1][0]) {
  468. sub[1] = '#' +sub[1];
  469. }
  470. sub[0] = '#' +sub[0];
  471. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkchan;
  472. } else if (sub[0].indexOf("://") !== -1) {
  473. if (!sub[1])
  474. sub[1] = sub[0];
  475. sub[2] = R.klass.msg.link;
  476. } else {
  477. return matched;
  478. }
  479. return '<a href="' +sub[0] +'" class="' +sub[2] +'"' +(!type ? ' target="_blank"' : '') +'>' +sub[1] +'</a>';
  480. });
  481. msgContent = formatEmojis(msgContent);
  482. var msgLength = msgContent.length;
  483. var isAlphadec = function(c) {
  484. return ((c >= 'A' && c <= 'Z') ||
  485. (c >= 'a' && c <= 'z') ||
  486. (c >= '0' && c <= '9') ||
  487. "àèìòùÀÈÌÒÙáéíóúýÁÉÍÓÚÝâêîôûÂÊÎÔÛãñõÃÑÕäëïöüÿÄËÏÖÜŸçÇߨøÅ寿œ".indexOf(c) !== -1);
  488. }
  489. ,checkEnd = function(str, pos, c) {
  490. while (str[pos]) {
  491. if (isAlphadec(str[pos]) && str[pos] != c && str[pos +1] == c) {
  492. return true;
  493. }
  494. pos++;
  495. }
  496. return false;
  497. } ,appendMod = function(mods) {
  498. if (!Object.keys(currentMods).length)
  499. return "";
  500. return '<span class="' +Object.keys(mods).join(' ') +'">';
  501. };
  502. // Skip trailing
  503. while (i < msgLength && (msgContent[i] === ' ' || msgContent[i] === '\t'))
  504. i++;
  505. if (msgContent.substr(i, 4) === '&gt;') {
  506. quote = true;
  507. i += 4;
  508. }
  509. for (; i < msgLength; i++) {
  510. var c = msgContent[i];
  511. if (c === '<') {
  512. do {
  513. _msgContent += msgContent[i++];
  514. } while (msgContent[i -1] !== '>');
  515. i--;
  516. continue;
  517. }
  518. if (!(currentMods[R.klass.msg.style.bold]) && c === '*' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  519. if (Object.keys(currentMods).length)
  520. _msgContent += '</span>';
  521. currentMods[R.klass.msg.style.bold] = true;
  522. _msgContent += appendMod(currentMods);
  523. } else if (!(currentMods[R.klass.msg.style.strike]) && c === '~' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  524. if (Object.keys(currentMods).length)
  525. _msgContent += '</span>';
  526. currentMods[R.klass.msg.style.strike] = true;
  527. _msgContent += appendMod(currentMods);
  528. } else if (!(currentMods[R.klass.msg.style.code]) && c === '`' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  529. if (Object.keys(currentMods).length)
  530. _msgContent += '</span>';
  531. currentMods[R.klass.msg.style.code] = true;
  532. _msgContent += appendMod(currentMods);
  533. } else if (!(currentMods[R.klass.msg.style.italic]) && c === '_' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  534. if (Object.keys(currentMods).length)
  535. _msgContent += '</span>';
  536. currentMods[R.klass.msg.style.italic] = true;
  537. _msgContent += appendMod(currentMods);
  538. } else {
  539. var finalFound = false;
  540. _msgContent += c;
  541. do {
  542. if ((currentMods[R.klass.msg.style.bold]) && c !== '*' && msgContent[i +1] === '*') {
  543. delete currentMods[R.klass.msg.style.bold];
  544. finalFound = true;
  545. } else if ((currentMods[R.klass.msg.style.strike]) && c !== '~' && msgContent[i +1] === '~') {
  546. delete currentMods[R.klass.msg.style.strike];
  547. finalFound = true;
  548. } else if ((currentMods[R.klass.msg.style.code]) && c !== '`' && msgContent[i +1] === '`') {
  549. delete currentMods[R.klass.msg.style.code];
  550. finalFound = true;
  551. } else if ((currentMods[R.klass.msg.style.italic]) && c !== '_' && msgContent[i +1] === '_') {
  552. delete currentMods[R.klass.msg.style.italic];
  553. finalFound = true;
  554. } else {
  555. break;
  556. }
  557. c = msgContent[++i];
  558. } while (i < msgLength);
  559. if (finalFound)
  560. _msgContent += '</span>' +appendMod(currentMods);
  561. }
  562. }
  563. if (currentMods) {
  564. // Should not append
  565. _msgContent += '</span>';
  566. }
  567. if (quote)
  568. msgContents[msgContentIndex] = '<span class="' +R.klass.msg.style.quote +'">' +_msgContent +'</span>';
  569. else
  570. msgContents[msgContentIndex] = _msgContent;
  571. }
  572. return msgContents.join('<br/>');
  573. }
  574. /**
  575. * @param {string} channelId
  576. * @param {SlackMessage} msg
  577. * @param {*} attachment
  578. * @return {Element|null}
  579. **/
  580. function createAttachmentDom(channelId, msg, attachment) {
  581. var rootDom = document.createElement("li")
  582. ,attachmentBlock = document.createElement("div")
  583. ,pretext = document.createElement("div")
  584. ,titleBlock = document.createElement("a")
  585. ,authorBlock = document.createElement("div")
  586. ,authorImg = document.createElement("img")
  587. ,authorName = document.createElement("a")
  588. ,textBlock = document.createElement("div")
  589. ,textDom = document.createElement("div")
  590. ,thumbImgDom = document.createElement("img")
  591. ,imgDom = document.createElement("img")
  592. ,footerBlock = document.createElement("div")
  593. ,footerIcon = document.createElement("img")
  594. ,footerText = document.createElement("span")
  595. ,footerTs = document.createElement("span")
  596. ;
  597. rootDom.className = R.klass.msg.attachment.container;
  598. //Color
  599. var color = "#e3e4e6";
  600. if (attachment["color"]) {
  601. if (attachment["color"][0] === '#')
  602. color = attachment["color"][0];
  603. else if (attachment["color"] === "good")
  604. color = "#2fa44f";
  605. else if (attachment["color"] === "warning")
  606. color = "#de9e31";
  607. else if (attachment["color"] === "danger")
  608. color = "#d50200";
  609. }
  610. attachmentBlock.style.borderColor = color;
  611. //Pretext
  612. pretext.className = R.klass.msg.attachment.pretext;
  613. if (attachment["pretext"]) {
  614. pretext.innerHTML = formatSlackText(attachment["pretext"]);
  615. } else {
  616. pretext.classList.add(R.klass.hidden);
  617. }
  618. //Title
  619. titleBlock.target = "_blank";
  620. if (attachment["title"]) {
  621. titleBlock.innerHTML = formatSlackText(attachment["title"]);
  622. if (attachment["title_link"]) {
  623. titleBlock.href = attachment["title_link"];
  624. }
  625. titleBlock.className = R.klass.msg.attachment.title;
  626. } else {
  627. titleBlock.className = R.klass.hidden + " " +R.klass.msg.attachment.title;
  628. }
  629. //Author
  630. authorName.target = "_blank";
  631. authorBlock.className = R.klass.msg.author;
  632. if (attachment["author_name"]) {
  633. authorName.innerHTML = formatSlackText(attachment["author_name"]);
  634. authorName.href = attachment["author_link"] || "";
  635. authorName.className = R.klass.msg.authorname;
  636. authorImg.className = R.klass.msg.authorAvatar;
  637. if (attachment["author_icon"])
  638. authorImg.src = attachment["author_icon"];
  639. else
  640. authorImg.classList.add(R.klass.hidden);
  641. } else {
  642. authorBlock.classList.add(R.klass.hidden);
  643. }
  644. //Text
  645. textDom.innerHTML = formatSlackText(attachment["text"] || "");
  646. textDom.klassName = R.klass.msg.attachment.text;
  647. // Img (small one)
  648. thumbImgDom.className = R.klass.msg.attachment.thumbImg;
  649. if (attachment["thumb_url"])
  650. thumbImgDom.src = attachment["thumb_url"];
  651. else
  652. thumbImgDom.classList.add(R.klass.hidden);
  653. //Img (the big one)
  654. imgDom.className = R.klass.msg.attachment.img;
  655. if (attachment["image_url"])
  656. imgDom.src = attachment["image_url"];
  657. else
  658. imgDom.classList.add(R.klass.hidden);
  659. //Footer
  660. footerBlock.className = R.klass.msg.attachment.footer;
  661. footerText.className = R.klass.msg.attachment.footerText;
  662. footerIcon.className = R.klass.msg.attachment.footerIcon;
  663. if (attachment["footer"]) {
  664. footerText.innerHTML = formatSlackText(attachment["footer"]);
  665. if (attachment["footer_icon"])
  666. footerIcon.src = attachment["footer_icon"];
  667. else
  668. footerIcon.classList.add(R.klass.hidden);
  669. } else {
  670. footerIcon.classList.add(R.klass.hidden);
  671. footerText.classList.add(R.klass.hidden);
  672. }
  673. //Ts
  674. footerTs.className = R.klass.msg.ts;
  675. if (attachment["ts"])
  676. footerTs.innerHTML = locale.formatDate(attachment["ts"]);
  677. else
  678. footerTs.classList.add(R.klass.hidden);
  679. // TODO Field [ {title, value, short } ]
  680. // TODO actions (button stuff)
  681. authorBlock.appendChild(authorImg);
  682. authorBlock.appendChild(authorName);
  683. textBlock.appendChild(textDom);
  684. textBlock.appendChild(thumbImgDom);
  685. footerBlock.appendChild(footerIcon);
  686. footerBlock.appendChild(footerText);
  687. footerBlock.appendChild(footerTs);
  688. attachmentBlock.appendChild(titleBlock);
  689. attachmentBlock.appendChild(authorBlock);
  690. attachmentBlock.appendChild(textBlock);
  691. attachmentBlock.appendChild(imgDom);
  692. attachmentBlock.appendChild(footerBlock);
  693. rootDom.appendChild(pretext);
  694. rootDom.appendChild(attachmentBlock);
  695. return rootDom;
  696. }
  697. /**
  698. * @param {string} channelId
  699. * @param {SlackMessage} msg
  700. * @param {boolean=} skipAttachment
  701. * @return {Element}
  702. **/
  703. function doCreateMeMessageDom(channelId, msg, skipAttachment) {
  704. var dom = doCreateMessageDom(channelId, msg, skipAttachment);
  705. dom.classList.add(R.klass.msg.meMessage);
  706. return dom;
  707. }
  708. /**
  709. * @param {string} channelId
  710. * @param {SlackMessage} msg
  711. * @param {boolean=} skipAttachment
  712. * @return {Element}
  713. **/
  714. function createMessageDom(channelId, msg, skipAttachment) {
  715. var dom = (msg.isMeMessage ?
  716. doCreateMeMessageDom(channelId, msg, skipAttachment):
  717. doCreateMessageDom(channelId, msg, skipAttachment));
  718. if (msg.edited)
  719. dom.classList.add(R.klass.msg.edited);
  720. return dom;
  721. }
  722. /**
  723. * @param {number} unreadhi
  724. * @param {number} unread
  725. **/
  726. function setFavicon(unreadhi, unread) {
  727. if (!unreadhi && !unread)
  728. document.getElementById(R.id.favicon).href = "favicon_ok.png";
  729. else
  730. document.getElementById(R.id.favicon).href = "favicon.png?h="+unreadhi+"&m="+unread;
  731. }
  732. function setNetErrorFavicon() {
  733. document.getElementById(R.id.favicon).href = "favicon_err.png";
  734. }
  735. function updateTitle() {
  736. var hasUnread = 0
  737. ,hasHl = 0
  738. ,title = "";
  739. if (NEXT_RETRY) {
  740. title = '!' +locale.netErrorShort +' - ';
  741. setNetErrorFavicon();
  742. } else {
  743. for (var i in UNREAD_CHANS) {
  744. if (UNREAD_CHANS.hasOwnProperty(i)) {
  745. hasUnread += UNREAD_CHANS[i].unread;
  746. hasHl += UNREAD_CHANS[i].hl;
  747. }
  748. }
  749. if (hasHl)
  750. title = "(!" +hasHl +") - ";
  751. else if (hasUnread)
  752. title = "(" +hasUnread +") - ";
  753. setFavicon(hasHl, hasUnread);
  754. }
  755. title += SLACK.context.team.name;
  756. document.title = title;
  757. }
  758. function spawnNotification() {
  759. if (!("Notification" in window))
  760. {}
  761. else if (Notification.permission === "granted") {
  762. var now = Date.now();
  763. if (lastNotificationSpawn + NOTIFICATION_COOLDOWN < now) {
  764. var n = new Notification(locale.newMessage);
  765. lastNotificationSpawn = now;
  766. setTimeout(function() {
  767. n.close();
  768. }, NOTIFICATION_DELAY);
  769. }
  770. }
  771. else if (Notification.permission !== "denied")
  772. Notification.requestPermission();
  773. }
  774. function onRoomUpdated() {
  775. var chatFrag = document.createDocumentFragment()
  776. ,currentRoomId = SELECTED_ROOM.id
  777. ,prevMsg = null
  778. ,firstTsCombo = 0
  779. ,prevMsgDom = null;
  780. if (SLACK.history[currentRoomId])
  781. SLACK.history[currentRoomId].messages.forEach(function(msg) {
  782. if (!msg.removed) {
  783. var dom = createMessageDom(currentRoomId, msg);
  784. if (prevMsg && prevMsg.userId === msg.userId && msg.userId) {
  785. dom.classList.add(R.klass.msg.same.author);
  786. if (Math.abs(firstTsCombo -msg.ts) < 30)
  787. prevMsgDom.classList.add(R.klass.msg.same.ts);
  788. else
  789. firstTsCombo = msg.ts;
  790. } else
  791. firstTsCombo = msg.ts;
  792. prevMsg = msg;
  793. prevMsgDom = dom;
  794. chatFrag.appendChild(dom);
  795. }
  796. });
  797. var content = document.getElementById(R.id.currentRoom.content);
  798. content.textContent = "";
  799. content.appendChild(chatFrag);
  800. //TODO scroll lock
  801. content.scrollTop = content.scrollHeight -content.clientHeight;
  802. }
  803. function chatClickDelegate(e) {
  804. var target = e.target
  805. ,getMessageId = function(e, target) {
  806. target = target || e.target;
  807. while (target !== e.currentTarget && target) {
  808. if (target.classList.contains(R.klass.msg.item)) {
  809. return target.id;
  810. }
  811. target = target.parentElement;
  812. }
  813. };
  814. while (target !== e.currentTarget && target) {
  815. if (target.classList.contains(R.klass.msg.hover.container)) {
  816. return;
  817. } else if (target.parentElement && target.parentElement.classList.contains(R.klass.msg.hover.container)) {
  818. var messageId = getMessageId(e, target);
  819. if (messageId) {
  820. messageId = parseFloat(messageId.split("_")[1]);
  821. var msg = SLACK.history[SELECTED_ROOM.id].getMessage(messageId);
  822. if (msg && target.classList.contains(R.klass.msg.hover.reply)) {
  823. if (EDITING) {
  824. EDITING = null;
  825. onEditingUpdated();
  826. }
  827. if (REPLYING_TO !== msg) {
  828. REPLYING_TO = msg;
  829. onReplyingToUpdated();
  830. }
  831. } else if (msg && target.classList.contains(R.klass.msg.hover.reaction)) {
  832. EMOJI_BAR.spawn(document.body, function(emoji) {
  833. if (emoji)
  834. addReaction(SELECTED_ROOM.id, msg.id, emoji);
  835. });
  836. } else if (msg && target.classList.contains(R.klass.msg.hover.edit)) {
  837. if (REPLYING_TO) {
  838. REPLYING_TO = null;
  839. onReplyingToUpdated();
  840. }
  841. if (EDITING !== msg) {
  842. EDITING = msg;
  843. onEditingUpdated();
  844. }
  845. } else if (msg && target.classList.contains(R.klass.msg.hover.remove)) {
  846. //TODO promt confirm
  847. if (REPLYING_TO) {
  848. REPLYING_TO = null;
  849. onReplyingToUpdated();
  850. }
  851. if (EDITING) {
  852. EDITING = null;
  853. onEditingUpdated();
  854. }
  855. removeMsg(SELECTED_ROOM, msg);
  856. }
  857. }
  858. return;
  859. }
  860. target = target.parentElement;
  861. }
  862. }
  863. function focusInput() {
  864. document.getElementById(R.id.message.input).focus();
  865. }
  866. function setRoomFromHashBang() {
  867. var hashId = document.location.hash.substr(1)
  868. ,room = SLACK.context.getChannel(hashId)
  869. ,user = SLACK.context.getMember(hashId);
  870. if (room && room !== SELECTED_ROOM)
  871. selectRoom(room);
  872. else if (user && user.ims)
  873. selectRoom(user.ims);
  874. }
  875. document.addEventListener('DOMContentLoaded', function() {
  876. initLang();
  877. document.getElementById(R.id.currentRoom.content).addEventListener("click", chatClickDelegate);
  878. window.addEventListener("hashchange", function(e) {
  879. if (document.location.hash && document.location.hash[0] === '#') {
  880. setRoomFromHashBang();
  881. }
  882. });
  883. document.getElementById(R.id.message.file.cancel).addEventListener("click", function(e) {
  884. e.preventDefault();
  885. document.getElementById(R.id.message.file.error).classList.add(R.klass.hidden);
  886. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  887. document.getElementById(R.id.message.file.fileInput).value = "";
  888. return false;
  889. });
  890. document.getElementById(R.id.message.file.form).addEventListener("submit", function(e) {
  891. e.preventDefault();
  892. var fileInput = document.getElementById(R.id.message.file.fileInput)
  893. ,filename = fileInput.value;
  894. if (filename) {
  895. filename = filename.substr(filename.lastIndexOf('\\') +1);
  896. uploadFile(SELECTED_ROOM, filename, fileInput.files[0], function(errorMsg) {
  897. var error = document.getElementById(R.id.message.file.error);
  898. if (errorMsg) {
  899. error.textContent = errorMsg;
  900. error.classList.remove(R.klass.hidden);
  901. } else {
  902. error.classList.add(R.klass.hidden);
  903. document.getElementById(R.id.message.file.fileInput).value = "";
  904. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  905. }
  906. });
  907. }
  908. return false;
  909. });
  910. document.getElementById(R.id.message.file.bt).addEventListener("click", function(e) {
  911. e.preventDefault();
  912. if (SELECTED_ROOM) {
  913. document.getElementById(R.id.message.file.formContainer).classList.remove(R.klass.hidden);
  914. }
  915. return false;
  916. });
  917. document.getElementById(R.id.message.form).addEventListener("submit", function(e) {
  918. e.preventDefault();
  919. var input =document.getElementById(R.id.message.input);
  920. if (SELECTED_ROOM && input.value) {
  921. if (onTextEntered(input.value)) {
  922. input.value = "";
  923. if (REPLYING_TO) {
  924. REPLYING_TO = null;
  925. onReplyingToUpdated();
  926. }
  927. if (EDITING) {
  928. EDITING = null;
  929. onReplyingToUpdated();
  930. }
  931. }
  932. }
  933. focusInput();
  934. return false;
  935. });
  936. window.addEventListener('blur', function() {
  937. window.hasFocus = false;
  938. });
  939. window.addEventListener('focus', function() {
  940. window.hasFocus = true;
  941. lastNotificationSpawn = 0;
  942. if (SELECTED_ROOM)
  943. markRoomAsRead(SELECTED_ROOM);
  944. focusInput();
  945. });
  946. window.hasFocus = true;
  947. //Emoji closure
  948. (function() {
  949. var emojiButton = document.getElementById(R.id.message.emoji);
  950. if ('makeEmoji' in window) {
  951. var emojiDom = window['makeEmoji']('smile');
  952. if (emojiDom) {
  953. emojiButton.innerHTML = "<span class='" +R.klass.emoji.small +"'>" +emojiDom.outerHTML +"</span>";
  954. } else {
  955. emojiButton.style.backgroundImage = 'url("smile.svg")';
  956. }
  957. emojiDom = window['makeEmoji']('paperclip');
  958. if (emojiDom) {
  959. document.getElementById(R.id.message.file.bt).innerHTML = "<span class='" +R.klass.emoji.small +"'>" +emojiDom.outerHTML +"</span>";
  960. } else {
  961. document.getElementById(R.id.message.file.bt).style.backgroundImage = 'url("public/paperclip.svg")';
  962. }
  963. emojiButton.addEventListener("click", function() {
  964. EMOJI_BAR.spawn(document.body, function(e) {
  965. if (e) document.getElementById(R.id.message.input).value += ":"+e+":";
  966. focusInput();
  967. });
  968. });
  969. } else {
  970. emojiButton.classList.add(R.klass.hidden);
  971. }
  972. })();
  973. startPolling();
  974. });