ui.js 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696
  1. var
  2. /** @type {SlackMessage|null} */
  3. REPLYING_TO = null
  4. /**
  5. * Minimum time between 2 notifications (ms)
  6. * @const
  7. * @type {number}
  8. **/
  9. ,NOTIFICATION_COOLDOWN = 30 * 1000 //30 sec
  10. /**
  11. * Maximum time the notification will stay visible (ms)
  12. * @const
  13. * @type {number}
  14. **/
  15. ,NOTIFICATION_DELAY = 5 * 1000 // 5 sec
  16. /** @type {number} */
  17. ,lastNotificationSpawn = 0
  18. ;
  19. /**
  20. * @param {SlackChan|SlackGroup} chan
  21. * @return {Element}
  22. **/
  23. function createChanListItem(chan) {
  24. var dom = document.createElement("li")
  25. ,link = document.createElement("a");
  26. dom.id = chan.id;
  27. link.href = '#' +chan.id;
  28. if (chan.id[0] === 'D')
  29. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeDirect;
  30. else if (chan.id[0] === 'G')
  31. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeGroup;
  32. else if (chan.id[0] === 'C')
  33. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeChannel;
  34. link.textContent = chan.name;
  35. dom.appendChild(link);
  36. return dom;
  37. }
  38. /**
  39. * @param {SlackIms} ims
  40. * @return {Element}
  41. **/
  42. function createImsListItem(ims) {
  43. var dom = document.createElement("li")
  44. ,link = document.createElement("a");
  45. dom.id = ims.id;
  46. link.href = '#' +ims.id;
  47. dom.className = R.klass.chatList.entry;
  48. link.textContent = ims.user.name;
  49. dom.appendChild(link);
  50. return dom;
  51. }
  52. function onContextUpdated() {
  53. var chanListFram = document.createDocumentFragment();
  54. var sortedChans = SLACK.context.self ? Object.keys(SLACK.context.self.channels) : [];
  55. sortedChans.sort(function(a, b) {
  56. if (a[0] !== b[0]) {
  57. return a[0] - b[0];
  58. }
  59. return (SLACK.context.channels[a] || SLACK.context.groups[a]).name.localeCompare((SLACK.context.channels[b] || SLACK.context.groups[b]).name);
  60. });
  61. sortedChans.forEach(function(chanId) {
  62. var chan = SLACK.context.channels[chanId] || SLACK.context.groups[chanId]
  63. ,chanListItem = createChanListItem(chan);
  64. if (chanListItem) {
  65. chanListFram.appendChild(chanListItem);
  66. }
  67. });
  68. var sortedUsers = SLACK.context.users ? Object.keys(SLACK.context.users) : [];
  69. sortedUsers.sort(function(a, b) {
  70. return SLACK.context.users[a].name.localeCompare(SLACK.context.users[b].name);
  71. });
  72. sortedUsers.forEach(function(userId) {
  73. var ims = SLACK.context.users[userId].ims
  74. ,imsListItem = createImsListItem(ims);
  75. if (imsListItem) {
  76. chanListFram.appendChild(imsListItem);
  77. }
  78. });
  79. document.getElementById(R.id.chanList).textContent = "";
  80. document.getElementById(R.id.chanList).appendChild(chanListFram);
  81. setRoomFromHashBang();
  82. }
  83. function onNetworkStateUpdated(isNetworkWorking) {
  84. isNetworkWorking ? document.body.classList.remove(R.klass.noNetwork) : document.body.classList.add(R.klass.noNetwork);
  85. }
  86. function onRoomSelected() {
  87. var name = SELECTED_ROOM.name || (SELECTED_ROOM.user ? SELECTED_ROOM.user.name : undefined);
  88. if (!name) {
  89. var members = [];
  90. for (var i in SELECTED_ROOM.members) {
  91. members.push(SELECTED_ROOM.members[i].name);
  92. }
  93. name = members.join(", ");
  94. }
  95. var roomLi = document.getElementById(SELECTED_ROOM.id);
  96. document.getElementById(R.id.currentRoom.title).textContent = name;
  97. onRoomUpdated();
  98. focusInput();
  99. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  100. markRoomAsRead(SELECTED_ROOM);
  101. if (REPLYING_TO) {
  102. REPLYING_TO = null;
  103. onReplyingToUpdated();
  104. }
  105. }
  106. function onReplyingToUpdated() {
  107. if (REPLYING_TO) {
  108. document.body.classList.add(R.klass.replyingTo);
  109. var domParent = document.getElementById(R.id.message.replyTo)
  110. ,closeLink = document.createElement("a");
  111. closeLink.addEventListener("click", function() {
  112. REPLYING_TO = null;
  113. onReplyingToUpdated();
  114. });
  115. closeLink.className = R.klass.msg.replyTo.close;
  116. closeLink.textContent = 'x';
  117. domParent.textContent = "";
  118. domParent.appendChild(closeLink);
  119. domParent.appendChild(createMessageDom("reply_" +SELECTED_ROOM.id, REPLYING_TO, true));
  120. } else {
  121. document.body.classList.remove(R.klass.replyingTo);
  122. }
  123. }
  124. /**
  125. * @param {string} channelId
  126. * @param {SlackMessage} msg
  127. * @param {boolean=} skipAttachment
  128. * @return {Element}
  129. **/
  130. function doCreateMessageDom(channelId, msg, skipAttachment) {
  131. var dom = document.createElement("div")
  132. ,ts = document.createElement("div")
  133. ,text = document.createElement("div")
  134. ,author = document.createElement("div")
  135. ,authorImg = document.createElement("img")
  136. ,authorName = document.createElement("span")
  137. ,hover = document.createElement("ul")
  138. ,hoverReply = document.createElement("li")
  139. ,attachments = document.createElement("ul")
  140. ,sender = msg.raw["user"] ?
  141. SLACK.context.users[msg.raw["user"]] :
  142. SLACK.context.bots[msg.raw["bot_id"]];
  143. dom.id = channelId +"_" +msg.ts;
  144. dom.className = R.klass.msg.item;
  145. ts.className = R.klass.msg.ts;
  146. text.className = R.klass.msg.msg;
  147. author.className = R.klass.msg.author;
  148. authorImg.className = R.klass.msg.authorAvatar;
  149. authorName.className = R.klass.msg.authorname;
  150. hover.className = R.klass.msg.hover.container;
  151. hoverReply.className = R.klass.msg.hover.reply;
  152. ts.innerHTML = formatDate(msg.ts);
  153. text.innerHTML = formatSlackText(msg.raw["text"] || "");
  154. authorName.textContent = sender ? sender.name : (msg.raw["username"] || "?");
  155. if (!sender && !msg.raw["username"])
  156. text.textContent = msg.raw["subtype"] || JSON.stringify(msg.raw);
  157. authorImg.src = sender ? sender.icons.image_48 : "";
  158. author.appendChild(authorImg);
  159. author.appendChild(authorName);
  160. hover.appendChild(hoverReply);
  161. dom.appendChild(author);
  162. dom.appendChild(text);
  163. dom.appendChild(ts);
  164. dom.appendChild(attachments);
  165. attachments.className = R.klass.msg.attachment.list;
  166. if (skipAttachment !== true && msg.raw["attachments"]) {
  167. msg.raw["attachments"].forEach(function(attachment) {
  168. var domAttachment = createAttachmentDom(channelId, msg, attachment);
  169. if (domAttachment)
  170. attachments.appendChild(domAttachment);
  171. });
  172. }
  173. dom.appendChild(hover);
  174. return dom;
  175. }
  176. /**
  177. * @param {string|number} ts seconds between EPOCH and event
  178. * @return {string}
  179. **/
  180. function formatDate(ts) {
  181. if (typeof(ts) !== "string")
  182. ts = parseFloat(ts);
  183. return (new Date(ts * 1000)).toLocaleTimeString();
  184. }
  185. /**
  186. * replace all :emoji: codes with corresponding image
  187. * @param {string} inputString
  188. * @return {string}
  189. **/
  190. function formatEmojis(inputString) {
  191. if (!("makeEmoji" in window))
  192. return inputString;
  193. return inputString.replace(/:(\w+):/g, function(returnFailed, emoji) {
  194. var emojiDom = window['makeEmoji'](emoji);
  195. console.log(emoji, emojiDom ? "found" : "not found");
  196. if (emojiDom) {
  197. var domParent = document.createElement("span");
  198. domParent.className = R.klass.emoji.small;
  199. domParent.appendChild(emojiDom);
  200. return domParent.outerHTML;
  201. }
  202. return returnFailed;
  203. });
  204. }
  205. /**
  206. * @param {string} fullText
  207. * @return {string}
  208. **/
  209. function formatSlackText(fullText) {
  210. var msgContents = fullText.split(/\r?\n/g);
  211. for (var msgContentIndex=0, nbMsgContents = msgContents.length; msgContentIndex < nbMsgContents; msgContentIndex++) {
  212. var msgContent = formatEmojis(msgContents[msgContentIndex])
  213. ,_msgContent = ""
  214. ,currentMods = {}
  215. ,quote = false
  216. ,i =0
  217. ,msgLength = msgContent.length;
  218. var checkEnd = function(str, pos, c) {
  219. while (str[pos]) {
  220. if (str[pos] != ' ' && str[pos] != c && str[pos +1] == c) {
  221. return true;
  222. }
  223. pos++;
  224. }
  225. return false;
  226. } ,appendMod = function(mods) {
  227. if (!Object.keys(currentMods).length)
  228. return "";
  229. return '<span class="' +Object.keys(mods).join(' ') +'">';
  230. };
  231. // Skip trailing
  232. while (i < msgLength && (msgContent[i] === ' ' || msgContent[i] === '\t'))
  233. i++;
  234. if (msgContent.substr(i, 4) === '&gt;') {
  235. quote = true;
  236. i += 4;
  237. }
  238. for (; i < msgLength; i++) {
  239. var c = msgContent[i];
  240. if (!(currentMods[R.klass.msg.style.bold]) && c === '*' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  241. if (Object.keys(currentMods).length)
  242. _msgContent += '</span>';
  243. currentMods[R.klass.msg.style.bold] = true;
  244. _msgContent += appendMod(currentMods);
  245. } else if (!(currentMods[R.klass.msg.style.strike]) && c === '~' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  246. if (Object.keys(currentMods).length)
  247. _msgContent += '</span>';
  248. currentMods[R.klass.msg.style.strike] = true;
  249. _msgContent += appendMod(currentMods);
  250. } else if (!(currentMods[R.klass.msg.style.code]) && c === '`' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  251. if (Object.keys(currentMods).length)
  252. _msgContent += '</span>';
  253. currentMods[R.klass.msg.style.code] = true;
  254. _msgContent += appendMod(currentMods);
  255. } else if (!(currentMods[R.klass.msg.style.italic]) && c === '_' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  256. if (Object.keys(currentMods).length)
  257. _msgContent += '</span>';
  258. currentMods[R.klass.msg.style.italic] = true;
  259. _msgContent += appendMod(currentMods);
  260. } else {
  261. var finalFound = false;
  262. _msgContent += c;
  263. do {
  264. if ((currentMods[R.klass.msg.style.bold]) && c !== '*' && msgContent[i +1] === '*') {
  265. delete currentMods[R.klass.msg.style.bold];
  266. finalFound = true;
  267. } else if ((currentMods[R.klass.msg.style.strike]) && c !== '~' && msgContent[i +1] === '~') {
  268. delete currentMods[R.klass.msg.style.strike];
  269. finalFound = true;
  270. } else if ((currentMods[R.klass.msg.style.code]) && c !== '`' && msgContent[i +1] === '`') {
  271. delete currentMods[R.klass.msg.style.code];
  272. finalFound = true;
  273. } else if ((currentMods[R.klass.msg.style.italic]) && c !== '_' && msgContent[i +1] === '_') {
  274. delete currentMods[R.klass.msg.style.italic];
  275. finalFound = true;
  276. } else {
  277. break;
  278. }
  279. c = msgContent[++i];
  280. } while (i < msgLength);
  281. if (finalFound)
  282. _msgContent += '</span>' +appendMod(currentMods);
  283. }
  284. }
  285. if (currentMods) {
  286. // Should not append
  287. _msgContent += '</span>';
  288. }
  289. _msgContent = _msgContent.replace(new RegExp('<([@#]?)([^>]*)>', 'g'),
  290. function(matched, type, entity) {
  291. var sub = entity.split('|');
  292. if (type === '@') {
  293. if (!sub[1]) {
  294. var user = SLACK.context.getMember(sub[0]);
  295. sub[1] = user ? ('@' +user.name) : locale.unknownMember;
  296. } else if ('@' !== sub[1][0]) {
  297. sub[1] = '@' +sub[1];
  298. }
  299. sub[0] = '#' +sub[0];
  300. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
  301. } else if (type === '#') {
  302. if (!sub[1]) {
  303. var chan = SLACK.context.getChannel(sub[0]);
  304. sub[1] = chan ? ('#' +chan.name) : locale.unknownChannel;
  305. } else if ('#' !== sub[1][0]) {
  306. sub[1] = '#' +sub[1];
  307. }
  308. sub[0] = '#' +sub[0];
  309. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkchan;
  310. } else if (sub[0].indexOf("://") !== -1) {
  311. if (!sub[1])
  312. sub[1] = sub[0];
  313. sub[2] = R.klass.msg.link;
  314. } else {
  315. return matched;
  316. }
  317. return '<a href="' +sub[0] +'" class="' +sub[2] +'"' +(!type ? ' target="_blank"' : '') +'>' +sub[1] +'</a>';
  318. });
  319. if (quote)
  320. msgContents[msgContentIndex] = '<span class="' +R.klass.msg.style.quote +'">' +_msgContent +'</span>';
  321. else
  322. msgContents[msgContentIndex] = _msgContent;
  323. }
  324. return msgContents.join('<br/>');
  325. }
  326. /**
  327. * @param {string} channelId
  328. * @param {SlackMessage} msg
  329. * @param {*} attachment
  330. * @return {Element|null}
  331. **/
  332. function createAttachmentDom(channelId, msg, attachment) {
  333. var rootDom = document.createElement("li")
  334. ,attachmentBlock = document.createElement("div")
  335. ,pretext = document.createElement("div")
  336. ,titleBlock = document.createElement("a")
  337. ,authorBlock = document.createElement("div")
  338. ,authorImg = document.createElement("img")
  339. ,authorName = document.createElement("a")
  340. ,textBlock = document.createElement("div")
  341. ,textDom = document.createElement("div")
  342. ,thumbImgDom = document.createElement("img")
  343. ,imgDom = document.createElement("img")
  344. ,footerBlock = document.createElement("div")
  345. ,footerIcon = document.createElement("img")
  346. ,footerText = document.createElement("span")
  347. ,footerTs = document.createElement("span")
  348. ;
  349. rootDom.className = R.klass.msg.attachment.container;
  350. //Color
  351. var color = "#e3e4e6";
  352. if (attachment["color"]) {
  353. if (attachment["color"][0] === '#')
  354. color = attachment["color"][0];
  355. else if (attachment["color"] === "good")
  356. color = "#2fa44f";
  357. else if (attachment["color"] === "warning")
  358. color = "#de9e31";
  359. else if (attachment["color"] === "danger")
  360. color = "#d50200";
  361. }
  362. attachmentBlock.style.borderColor = color;
  363. //Pretext
  364. pretext.className = R.klass.msg.attachment.pretext;
  365. if (attachment["pretext"]) {
  366. pretext.innerHTML = formatSlackText(attachment["pretext"]);
  367. } else {
  368. pretext.classList.add(R.klass.hidden);
  369. }
  370. //Title
  371. titleBlock.target = "_blank";
  372. if (attachment["title"]) {
  373. titleBlock.innerHTML = formatSlackText(attachment["title"]);
  374. if (attachment["title_link"]) {
  375. titleBlock.href = attachment["title_link"];
  376. }
  377. titleBlock.className = R.klass.msg.attachment.title;
  378. } else {
  379. titleBlock.className = R.klass.hidden + " " +R.klass.msg.attachment.title;
  380. }
  381. //Author
  382. authorName.target = "_blank";
  383. authorBlock.className = R.klass.msg.author;
  384. if (attachment["author_name"]) {
  385. authorName.innerHTML = formatSlackText(attachment["author_name"]);
  386. authorName.href = attachment["author_link"] || "";
  387. authorName.className = R.klass.msg.authorname;
  388. authorImg.className = R.klass.msg.authorAvatar;
  389. if (attachment["author_icon"])
  390. authorImg.src = attachment["author_icon"];
  391. else
  392. authorImg.classList.add(R.klass.hidden);
  393. } else {
  394. authorBlock.classList.add(R.klass.hidden);
  395. }
  396. //Text
  397. textDom.innerHTML = formatSlackText(attachment["text"] || "");
  398. textDom.klassName = R.klass.msg.attachment.text;
  399. // Img (small one)
  400. thumbImgDom.className = R.klass.msg.attachment.thumbImg;
  401. if (attachment["thumb_url"])
  402. thumbImgDom.src = attachment["thumb_url"];
  403. else
  404. thumbImgDom.classList.add(R.klass.hidden);
  405. //Img (the big one)
  406. imgDom.className = R.klass.msg.attachment.img;
  407. if (attachment["image_url"])
  408. imgDom.src = attachment["image_url"];
  409. else
  410. imgDom.classList.add(R.klass.hidden);
  411. //Footer
  412. footerBlock.className = R.klass.msg.attachment.footer;
  413. footerText.className = R.klass.msg.attachment.footerText;
  414. footerIcon.className = R.klass.msg.attachment.footerIcon;
  415. if (attachment["footer"]) {
  416. footerText.innerHTML = formatSlackText(attachment["footer"]);
  417. if (attachment["footer_icon"])
  418. footerIcon.src = attachment["footer_icon"];
  419. else
  420. footerIcon.classList.add(R.klass.hidden);
  421. } else {
  422. footerIcon.classList.add(R.klass.hidden);
  423. footerText.classList.add(R.klass.hidden);
  424. }
  425. //Ts
  426. footerTs.className = R.klass.msg.ts;
  427. if (attachment["ts"])
  428. footerTs.innerHTML = formatDate(attachment["ts"]);
  429. else
  430. footerTs.classList.add(R.klass.hidden);
  431. // TODO Field [ {title, value, short } ]
  432. // TODO actions (button stuff)
  433. authorBlock.appendChild(authorImg);
  434. authorBlock.appendChild(authorName);
  435. textBlock.appendChild(textDom);
  436. textBlock.appendChild(thumbImgDom);
  437. footerBlock.appendChild(footerIcon);
  438. footerBlock.appendChild(footerText);
  439. footerBlock.appendChild(footerTs);
  440. attachmentBlock.appendChild(titleBlock);
  441. attachmentBlock.appendChild(authorBlock);
  442. attachmentBlock.appendChild(textBlock);
  443. attachmentBlock.appendChild(imgDom);
  444. attachmentBlock.appendChild(footerBlock);
  445. rootDom.appendChild(pretext);
  446. rootDom.appendChild(attachmentBlock);
  447. return rootDom;
  448. }
  449. /**
  450. * @param {string} channelId
  451. * @param {SlackMessage} msg
  452. * @param {boolean=} skipAttachment
  453. * @return {Element}
  454. **/
  455. function doCreateMeMessageDom(channelId, msg, skipAttachment) {
  456. var dom = doCreateMessageDom(channelId, msg, skipAttachment);
  457. dom.classList.add(R.klass.msg.meMessage);
  458. return dom;
  459. }
  460. /**
  461. * @param {string} channelId
  462. * @param {SlackMessage} msg
  463. * @param {boolean=} skipAttachment
  464. * @return {Element}
  465. **/
  466. function createMessageDom(channelId, msg, skipAttachment) {
  467. if (msg.subtype === "me_message") {
  468. return doCreateMeMessageDom(channelId, msg, skipAttachment);
  469. }
  470. return doCreateMessageDom(channelId, msg, skipAttachment);
  471. }
  472. function updateTitle() {
  473. var hasUnread = 0
  474. ,hasHl = 0
  475. ,title = "";
  476. for (var i in UNREAD_CHANS) {
  477. if (UNREAD_CHANS.hasOwnProperty(i)) {
  478. hasUnread += UNREAD_CHANS[i].unread;
  479. hasHl += UNREAD_CHANS[i].hl;
  480. }
  481. }
  482. if (hasHl)
  483. title = "(!" +hasHl +") - ";
  484. else if (hasUnread)
  485. title = "(" +hasUnread +") - ";
  486. title += SLACK.context.team.name;
  487. document.title = title;
  488. }
  489. function spawnNotification() {
  490. if (!("Notification" in window))
  491. {}
  492. else if (Notification.permission === "granted") {
  493. var now = Date.now();
  494. if (lastNotificationSpawn + NOTIFICATION_COOLDOWN < now) {
  495. var n = new Notification(locale.newMessage);
  496. lastNotificationSpawn = now;
  497. setTimeout(function() {
  498. n.close();
  499. }, NOTIFICATION_DELAY);
  500. }
  501. }
  502. else if (Notification.permission !== "denied")
  503. Notification.requestPermission();
  504. }
  505. function onRoomUpdated() {
  506. var chatFrag = document.createDocumentFragment()
  507. ,currentRoomId = SELECTED_ROOM.id;
  508. document.getElementById(R.id.currentRoom.content).textContent = "";
  509. if (SLACK.history[currentRoomId])
  510. SLACK.history[currentRoomId].messages.forEach(function(msg) {
  511. if (msg.type === "message") {
  512. var dom = createMessageDom(currentRoomId, msg);
  513. chatFrag.appendChild(dom);
  514. }
  515. });
  516. var content = document.getElementById(R.id.currentRoom.content);
  517. content.appendChild(chatFrag);
  518. //TODO scroll lock
  519. content.scrollTop = content.scrollHeight -content.clientHeight;
  520. }
  521. function chatClickDelegate(e) {
  522. var target = e.target
  523. ,getMessageId = function(e, target) {
  524. target = target || e.target;
  525. while (target !== e.currentTarget && target) {
  526. if (target.classList.contains(R.klass.msg.item)) {
  527. return target.id;
  528. }
  529. target = target.parentElement;
  530. }
  531. };
  532. while (target !== e.currentTarget && target) {
  533. if (target.classList.contains(R.klass.msg.hover.container)) {
  534. return;
  535. } else if (target.classList.contains(R.klass.msg.hover.reply)) {
  536. var messageId = getMessageId(e, target);
  537. if (messageId) {
  538. messageId = parseFloat(messageId.split("_")[1]);
  539. var history = SLACK.history[SELECTED_ROOM.id].messages;
  540. for (var i =0, histLen =history.length; i < histLen && history[i].ts <= messageId; i++) {
  541. if (history[i].ts === messageId) {
  542. if (REPLYING_TO !== history[i]) {
  543. REPLYING_TO = history[i];
  544. onReplyingToUpdated();
  545. }
  546. return;
  547. }
  548. }
  549. }
  550. return;
  551. }
  552. target = target.parentElement;
  553. }
  554. }
  555. function focusInput() {
  556. document.getElementById(R.id.message.input).focus();
  557. }
  558. function setRoomFromHashBang() {
  559. var hashId = document.location.hash.substr(1)
  560. ,room = SLACK.context.getChannel(hashId)
  561. ,user = SLACK.context.getMember(hashId);
  562. if (room && room !== SELECTED_ROOM)
  563. selectRoom(room);
  564. else if (user && user.ims)
  565. selectRoom(user.ims);
  566. }
  567. document.addEventListener('DOMContentLoaded', function() {
  568. initLang();
  569. document.getElementById(R.id.currentRoom.content).addEventListener("click", chatClickDelegate);
  570. window.addEventListener("hashchange", function(e) {
  571. if (document.location.hash && document.location.hash[0] === '#') {
  572. setRoomFromHashBang();
  573. }
  574. });
  575. document.getElementById(R.id.message.file.cancel).addEventListener("click", function(e) {
  576. e.preventDefault();
  577. document.getElementById(R.id.message.file.error).classList.add(R.klass.hidden);
  578. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  579. document.getElementById(R.id.message.file.fileInput).value = "";
  580. return false;
  581. });
  582. document.getElementById(R.id.message.file.form).addEventListener("submit", function(e) {
  583. e.preventDefault();
  584. var fileInput = document.getElementById(R.id.message.file.fileInput)
  585. ,filename = fileInput.value;
  586. if (filename) {
  587. filename = filename.substr(filename.lastIndexOf('\\') +1);
  588. uploadFile(SELECTED_ROOM, filename, fileInput.files[0], function(errorMsg) {
  589. var error = document.getElementById(R.id.message.file.error);
  590. if (errorMsg) {
  591. error.textContent = errorMsg;
  592. error.classList.remove(R.klass.hidden);
  593. } else {
  594. error.classList.add(R.klass.hidden);
  595. document.getElementById(R.id.message.file.fileInput).value = "";
  596. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  597. }
  598. });
  599. }
  600. return false;
  601. });
  602. document.getElementById(R.id.message.file.bt).addEventListener("click", function(e) {
  603. e.preventDefault();
  604. if (SELECTED_ROOM) {
  605. document.getElementById(R.id.message.file.formContainer).classList.remove(R.klass.hidden);
  606. }
  607. return false;
  608. });
  609. document.getElementById(R.id.message.form).addEventListener("submit", function(e) {
  610. e.preventDefault();
  611. var input =document.getElementById(R.id.message.input);
  612. if (SELECTED_ROOM && input.value) {
  613. sendMsg(SELECTED_ROOM, input.value, REPLYING_TO);
  614. input.value = "";
  615. if (REPLYING_TO) {
  616. REPLYING_TO = null;
  617. onReplyingToUpdated();
  618. }
  619. }
  620. focusInput();
  621. return false;
  622. });
  623. window.addEventListener('blur', function() {
  624. window.hasFocus = false;
  625. });
  626. window.addEventListener('focus', function() {
  627. window.hasFocus = true;
  628. lastNotificationSpawn = 0;
  629. if (SELECTED_ROOM)
  630. markRoomAsRead(SELECTED_ROOM);
  631. focusInput();
  632. });
  633. window.hasFocus = true;
  634. startPolling();
  635. });