ui.js 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879
  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. ,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. updateTitle();
  83. }
  84. function onNetworkStateUpdated(isNetworkWorking) {
  85. isNetworkWorking ? document.body.classList.remove(R.klass.noNetwork) : document.body.classList.add(R.klass.noNetwork);
  86. updateTitle();
  87. }
  88. function onRoomSelected() {
  89. var name = SELECTED_ROOM.name || (SELECTED_ROOM.user ? SELECTED_ROOM.user.name : undefined);
  90. if (!name) {
  91. var members = [];
  92. for (var i in SELECTED_ROOM.members) {
  93. members.push(SELECTED_ROOM.members[i].name);
  94. }
  95. name = members.join(", ");
  96. }
  97. var roomLi = document.getElementById(SELECTED_ROOM.id);
  98. document.getElementById(R.id.currentRoom.title).textContent = name;
  99. onRoomUpdated();
  100. focusInput();
  101. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  102. markRoomAsRead(SELECTED_ROOM);
  103. if (REPLYING_TO) {
  104. REPLYING_TO = null;
  105. onReplyingToUpdated();
  106. }
  107. }
  108. function onReplyingToUpdated() {
  109. if (REPLYING_TO) {
  110. document.body.classList.add(R.klass.replyingTo);
  111. var domParent = document.getElementById(R.id.message.replyTo)
  112. ,closeLink = document.createElement("a");
  113. closeLink.addEventListener("click", function() {
  114. REPLYING_TO = null;
  115. onReplyingToUpdated();
  116. });
  117. closeLink.className = R.klass.msg.replyTo.close;
  118. closeLink.textContent = 'x';
  119. domParent.textContent = "";
  120. domParent.appendChild(closeLink);
  121. domParent.appendChild(createMessageDom("reply_" +SELECTED_ROOM.id, REPLYING_TO, true));
  122. } else {
  123. document.body.classList.remove(R.klass.replyingTo);
  124. }
  125. }
  126. /**
  127. * @param {string} chanId
  128. * @param {string} msgId
  129. * @param {string} reaction
  130. * @param {Array.<string>} users
  131. * @return {Element|null}
  132. **/
  133. function createReactionDom(chanId, msgId, reaction, users) {
  134. var emojiDom = makeEmojiDom(reaction);
  135. if (emojiDom) {
  136. var dom = document.createElement("li")
  137. ,a = document.createElement("a")
  138. ,emojiContainer = document.createElement("span")
  139. ,userList = document.createElement("span")
  140. ,userNames = [];
  141. for (var i =0, nbUser = users.length; i < nbUser; i++) {
  142. var user = SLACK.context.getMember(users[i]);
  143. if (user)
  144. userNames.push(user.name);
  145. }
  146. userNames.sort();
  147. userList.textContent = userNames.join(", ");
  148. emojiContainer.appendChild(emojiDom);
  149. emojiContainer.className = R.klass.emoji.small;
  150. a.href = "javascript:toggleReaction('" +chanId +"', '" +msgId +"', '" +reaction +"')";
  151. a.appendChild(emojiContainer);
  152. a.appendChild(userList);
  153. dom.className = R.klass.msg.reactions.item;
  154. dom.appendChild(a);
  155. return dom;
  156. }
  157. return null;
  158. }
  159. /**
  160. * @param {string} chanId
  161. * @param {string} msgId
  162. * @param {string} reaction
  163. **/
  164. window['toggleReaction'] = function(chanId, msgId, reaction) {
  165. var hist = SLACK.history[chanId];
  166. if (!hist)
  167. return;
  168. var msg = hist.getMessageById(msgId);
  169. if (msg) {
  170. if (msg.hasReactionForUser(reaction, SLACK.context.self.id)) {
  171. removeReaction(chanId, msgId, reaction);
  172. } else {
  173. addReaction(chanId, msgId, reaction);
  174. }
  175. }
  176. };
  177. /**
  178. * @param {string} channelId
  179. * @param {SlackMessage} msg
  180. * @param {boolean=} skipAttachment
  181. * @return {Element}
  182. **/
  183. function doCreateMessageDom(channelId, msg, skipAttachment) {
  184. var dom = document.createElement("div")
  185. ,ts = document.createElement("div")
  186. ,text = document.createElement("div")
  187. ,author = document.createElement("div")
  188. ,authorImg = document.createElement("img")
  189. ,authorName = document.createElement("span")
  190. ,hover = document.createElement("ul")
  191. ,hoverReply = document.createElement("li")
  192. ,attachments = document.createElement("ul")
  193. ,reactions = document.createElement("ul")
  194. ,sender = msg.raw["user"] ?
  195. SLACK.context.users[msg.raw["user"]] :
  196. SLACK.context.bots[msg.raw["bot_id"]];
  197. dom.id = channelId +"_" +msg.ts;
  198. dom.className = R.klass.msg.item;
  199. ts.className = R.klass.msg.ts;
  200. text.className = R.klass.msg.msg;
  201. author.className = R.klass.msg.author;
  202. authorImg.className = R.klass.msg.authorAvatar;
  203. authorName.className = R.klass.msg.authorname;
  204. hover.className = R.klass.msg.hover.container;
  205. hoverReply.className = R.klass.msg.hover.reply;
  206. ts.innerHTML = formatDate(msg.ts);
  207. text.innerHTML = formatSlackText(msg.raw["text"] || "");
  208. authorName.textContent = sender ? sender.name : (msg.raw["username"] || "?");
  209. if (!sender && !msg.raw["username"])
  210. text.textContent = msg.raw["subtype"] || JSON.stringify(msg.raw);
  211. authorImg.src = sender ? sender.icons.image_48 : "";
  212. author.appendChild(authorImg);
  213. author.appendChild(authorName);
  214. hover.appendChild(hoverReply);
  215. if ('makeEmoji' in window) {
  216. var hoverReaction = document.createElement("li")
  217. ,domReply = window['makeEmoji']("arrow_heading_down")
  218. ,domReaction = window['makeEmoji']("smile");
  219. hoverReaction.className = R.klass.msg.hover.reaction;
  220. if (domReaction) {
  221. hoverReaction.classList.add(R.klass.emoji.small);
  222. hoverReaction.appendChild(domReaction);
  223. } else {
  224. hoverReaction.style.backgroundImage = 'url("smile.svg")';
  225. }
  226. if (domReply) {
  227. hoverReply.classList.add(R.klass.emoji.small);
  228. hoverReply.appendChild(domReply);
  229. } else {
  230. hoverReply.style.backgroundImage = 'url("repl.svg")';
  231. }
  232. hover.appendChild(hoverReaction);
  233. } else {
  234. hoverReply.style.backgroundImage = 'url("repl.svg")';
  235. }
  236. dom.appendChild(author);
  237. dom.appendChild(text);
  238. dom.appendChild(ts);
  239. dom.appendChild(attachments);
  240. dom.appendChild(reactions);
  241. attachments.className = R.klass.msg.attachment.list;
  242. reactions.className = R.klass.msg.reactions.container;
  243. if (skipAttachment !== true) {
  244. if (msg.reactions) for (var reaction in msg.reactions) {
  245. var reac = createReactionDom(channelId, msg.id, reaction, msg.reactions[reaction]);
  246. reac && reactions.appendChild(reac);
  247. }
  248. if (msg.raw["attachments"]) {
  249. msg.raw["attachments"].forEach(function(attachment) {
  250. var domAttachment = createAttachmentDom(channelId, msg, attachment);
  251. if (domAttachment)
  252. attachments.appendChild(domAttachment);
  253. });
  254. }
  255. }
  256. dom.appendChild(hover);
  257. return dom;
  258. }
  259. /**
  260. * @param {string|number} ts seconds between EPOCH and event
  261. * @return {string}
  262. **/
  263. function formatDate(ts) {
  264. if (typeof(ts) !== "string")
  265. ts = parseFloat(ts);
  266. return (new Date(ts * 1000)).toLocaleTimeString();
  267. }
  268. /**
  269. * Try to resolve emoji from customized context
  270. * @param {string} emoji
  271. * @return {Element|string}
  272. **/
  273. function tryGetCustomEmoji(emoji) {
  274. var loop = {};
  275. while (!loop[emoji]) {
  276. var emojisrc= SLACK.context.emojis[emoji];
  277. if (emojisrc) {
  278. if (emojisrc.substr(0, 6) == "alias:") {
  279. loop[emoji] = true;
  280. emoji = emojisrc.substr(6);
  281. } else {
  282. var dom = document.createElement("span");
  283. dom.className = R.klass.emoji.custom +' ' +R.klass.emoji.emoji;
  284. dom.style.backgroundImage = "url('" +emojisrc +"')";
  285. return dom;
  286. }
  287. }
  288. return emoji; // Emoji not found, fallback to std emoji
  289. }
  290. return emoji; //loop detected, return first emoji
  291. }
  292. function makeEmojiDom(emojiCode) {
  293. var emoji = tryGetCustomEmoji(emojiCode);
  294. if (typeof emoji === "string" && "makeEmoji" in window)
  295. emoji = window['makeEmoji'](emoji);
  296. return typeof emoji === "string" ? null : emoji;
  297. }
  298. /**
  299. * replace all :emoji: codes with corresponding image
  300. * @param {string} inputString
  301. * @return {string}
  302. **/
  303. function formatEmojis(inputString) {
  304. return inputString.replace(/:([^ \t]+):/g, function(returnFailed, emoji) {
  305. var emojiDom = makeEmojiDom(emoji);
  306. if (emojiDom) {
  307. var domParent = document.createElement("span");
  308. domParent.className = returnFailed === inputString ? R.klass.emoji.medium : R.klass.emoji.small;
  309. domParent.appendChild(emojiDom);
  310. return domParent.outerHTML;
  311. }
  312. return returnFailed;
  313. });
  314. }
  315. /**
  316. * @param {string} fullText
  317. * @return {string}
  318. **/
  319. function formatSlackText(fullText) {
  320. var msgContents = fullText.split(/\r?\n/g);
  321. for (var msgContentIndex=0, nbMsgContents = msgContents.length; msgContentIndex < nbMsgContents; msgContentIndex++) {
  322. var msgContent = msgContents[msgContentIndex]
  323. ,_msgContent = ""
  324. ,currentMods = {}
  325. ,quote = false
  326. ,i =0
  327. msgContent = msgContent.replace(new RegExp('<([@#]?)([^>]*)>', 'g'),
  328. function(matched, type, entity) {
  329. var sub = entity.split('|');
  330. if (type === '@') {
  331. if (!sub[1]) {
  332. var user = SLACK.context.getMember(sub[0]);
  333. sub[1] = user ? ('@' +user.name) : locale.unknownMember;
  334. } else if ('@' !== sub[1][0]) {
  335. sub[1] = '@' +sub[1];
  336. }
  337. sub[0] = '#' +sub[0];
  338. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
  339. } else if (type === '#') {
  340. if (!sub[1]) {
  341. var chan = SLACK.context.getChannel(sub[0]);
  342. sub[1] = chan ? ('#' +chan.name) : locale.unknownChannel;
  343. } else if ('#' !== sub[1][0]) {
  344. sub[1] = '#' +sub[1];
  345. }
  346. sub[0] = '#' +sub[0];
  347. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkchan;
  348. } else if (sub[0].indexOf("://") !== -1) {
  349. if (!sub[1])
  350. sub[1] = sub[0];
  351. sub[2] = R.klass.msg.link;
  352. } else {
  353. return matched;
  354. }
  355. return '<a href="' +sub[0] +'" class="' +sub[2] +'"' +(!type ? ' target="_blank"' : '') +'>' +sub[1] +'</a>';
  356. });
  357. msgContent = formatEmojis(msgContent);
  358. var msgLength = msgContent.length;
  359. var checkEnd = function(str, pos, c) {
  360. while (str[pos]) {
  361. if (str[pos] != ' ' && str[pos] != c && str[pos +1] == c) {
  362. return true;
  363. }
  364. pos++;
  365. }
  366. return false;
  367. } ,appendMod = function(mods) {
  368. if (!Object.keys(currentMods).length)
  369. return "";
  370. return '<span class="' +Object.keys(mods).join(' ') +'">';
  371. };
  372. // Skip trailing
  373. while (i < msgLength && (msgContent[i] === ' ' || msgContent[i] === '\t'))
  374. i++;
  375. if (msgContent.substr(i, 4) === '&gt;') {
  376. quote = true;
  377. i += 4;
  378. }
  379. for (; i < msgLength; i++) {
  380. var c = msgContent[i];
  381. if (c === '<') {
  382. do {
  383. _msgContent += msgContent[i++];
  384. } while (msgContent[i -1] !== '>');
  385. i--;
  386. continue;
  387. }
  388. if (!(currentMods[R.klass.msg.style.bold]) && c === '*' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  389. if (Object.keys(currentMods).length)
  390. _msgContent += '</span>';
  391. currentMods[R.klass.msg.style.bold] = true;
  392. _msgContent += appendMod(currentMods);
  393. } else if (!(currentMods[R.klass.msg.style.strike]) && c === '~' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  394. if (Object.keys(currentMods).length)
  395. _msgContent += '</span>';
  396. currentMods[R.klass.msg.style.strike] = true;
  397. _msgContent += appendMod(currentMods);
  398. } else if (!(currentMods[R.klass.msg.style.code]) && c === '`' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  399. if (Object.keys(currentMods).length)
  400. _msgContent += '</span>';
  401. currentMods[R.klass.msg.style.code] = true;
  402. _msgContent += appendMod(currentMods);
  403. } else if (!(currentMods[R.klass.msg.style.italic]) && c === '_' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  404. if (Object.keys(currentMods).length)
  405. _msgContent += '</span>';
  406. currentMods[R.klass.msg.style.italic] = true;
  407. _msgContent += appendMod(currentMods);
  408. } else {
  409. var finalFound = false;
  410. _msgContent += c;
  411. do {
  412. if ((currentMods[R.klass.msg.style.bold]) && c !== '*' && msgContent[i +1] === '*') {
  413. delete currentMods[R.klass.msg.style.bold];
  414. finalFound = true;
  415. } else if ((currentMods[R.klass.msg.style.strike]) && c !== '~' && msgContent[i +1] === '~') {
  416. delete currentMods[R.klass.msg.style.strike];
  417. finalFound = true;
  418. } else if ((currentMods[R.klass.msg.style.code]) && c !== '`' && msgContent[i +1] === '`') {
  419. delete currentMods[R.klass.msg.style.code];
  420. finalFound = true;
  421. } else if ((currentMods[R.klass.msg.style.italic]) && c !== '_' && msgContent[i +1] === '_') {
  422. delete currentMods[R.klass.msg.style.italic];
  423. finalFound = true;
  424. } else {
  425. break;
  426. }
  427. c = msgContent[++i];
  428. } while (i < msgLength);
  429. if (finalFound)
  430. _msgContent += '</span>' +appendMod(currentMods);
  431. }
  432. }
  433. if (currentMods) {
  434. // Should not append
  435. _msgContent += '</span>';
  436. }
  437. if (quote)
  438. msgContents[msgContentIndex] = '<span class="' +R.klass.msg.style.quote +'">' +_msgContent +'</span>';
  439. else
  440. msgContents[msgContentIndex] = _msgContent;
  441. }
  442. return msgContents.join('<br/>');
  443. }
  444. /**
  445. * @param {string} channelId
  446. * @param {SlackMessage} msg
  447. * @param {*} attachment
  448. * @return {Element|null}
  449. **/
  450. function createAttachmentDom(channelId, msg, attachment) {
  451. var rootDom = document.createElement("li")
  452. ,attachmentBlock = document.createElement("div")
  453. ,pretext = document.createElement("div")
  454. ,titleBlock = document.createElement("a")
  455. ,authorBlock = document.createElement("div")
  456. ,authorImg = document.createElement("img")
  457. ,authorName = document.createElement("a")
  458. ,textBlock = document.createElement("div")
  459. ,textDom = document.createElement("div")
  460. ,thumbImgDom = document.createElement("img")
  461. ,imgDom = document.createElement("img")
  462. ,footerBlock = document.createElement("div")
  463. ,footerIcon = document.createElement("img")
  464. ,footerText = document.createElement("span")
  465. ,footerTs = document.createElement("span")
  466. ;
  467. rootDom.className = R.klass.msg.attachment.container;
  468. //Color
  469. var color = "#e3e4e6";
  470. if (attachment["color"]) {
  471. if (attachment["color"][0] === '#')
  472. color = attachment["color"][0];
  473. else if (attachment["color"] === "good")
  474. color = "#2fa44f";
  475. else if (attachment["color"] === "warning")
  476. color = "#de9e31";
  477. else if (attachment["color"] === "danger")
  478. color = "#d50200";
  479. }
  480. attachmentBlock.style.borderColor = color;
  481. //Pretext
  482. pretext.className = R.klass.msg.attachment.pretext;
  483. if (attachment["pretext"]) {
  484. pretext.innerHTML = formatSlackText(attachment["pretext"]);
  485. } else {
  486. pretext.classList.add(R.klass.hidden);
  487. }
  488. //Title
  489. titleBlock.target = "_blank";
  490. if (attachment["title"]) {
  491. titleBlock.innerHTML = formatSlackText(attachment["title"]);
  492. if (attachment["title_link"]) {
  493. titleBlock.href = attachment["title_link"];
  494. }
  495. titleBlock.className = R.klass.msg.attachment.title;
  496. } else {
  497. titleBlock.className = R.klass.hidden + " " +R.klass.msg.attachment.title;
  498. }
  499. //Author
  500. authorName.target = "_blank";
  501. authorBlock.className = R.klass.msg.author;
  502. if (attachment["author_name"]) {
  503. authorName.innerHTML = formatSlackText(attachment["author_name"]);
  504. authorName.href = attachment["author_link"] || "";
  505. authorName.className = R.klass.msg.authorname;
  506. authorImg.className = R.klass.msg.authorAvatar;
  507. if (attachment["author_icon"])
  508. authorImg.src = attachment["author_icon"];
  509. else
  510. authorImg.classList.add(R.klass.hidden);
  511. } else {
  512. authorBlock.classList.add(R.klass.hidden);
  513. }
  514. //Text
  515. textDom.innerHTML = formatSlackText(attachment["text"] || "");
  516. textDom.klassName = R.klass.msg.attachment.text;
  517. // Img (small one)
  518. thumbImgDom.className = R.klass.msg.attachment.thumbImg;
  519. if (attachment["thumb_url"])
  520. thumbImgDom.src = attachment["thumb_url"];
  521. else
  522. thumbImgDom.classList.add(R.klass.hidden);
  523. //Img (the big one)
  524. imgDom.className = R.klass.msg.attachment.img;
  525. if (attachment["image_url"])
  526. imgDom.src = attachment["image_url"];
  527. else
  528. imgDom.classList.add(R.klass.hidden);
  529. //Footer
  530. footerBlock.className = R.klass.msg.attachment.footer;
  531. footerText.className = R.klass.msg.attachment.footerText;
  532. footerIcon.className = R.klass.msg.attachment.footerIcon;
  533. if (attachment["footer"]) {
  534. footerText.innerHTML = formatSlackText(attachment["footer"]);
  535. if (attachment["footer_icon"])
  536. footerIcon.src = attachment["footer_icon"];
  537. else
  538. footerIcon.classList.add(R.klass.hidden);
  539. } else {
  540. footerIcon.classList.add(R.klass.hidden);
  541. footerText.classList.add(R.klass.hidden);
  542. }
  543. //Ts
  544. footerTs.className = R.klass.msg.ts;
  545. if (attachment["ts"])
  546. footerTs.innerHTML = formatDate(attachment["ts"]);
  547. else
  548. footerTs.classList.add(R.klass.hidden);
  549. // TODO Field [ {title, value, short } ]
  550. // TODO actions (button stuff)
  551. authorBlock.appendChild(authorImg);
  552. authorBlock.appendChild(authorName);
  553. textBlock.appendChild(textDom);
  554. textBlock.appendChild(thumbImgDom);
  555. footerBlock.appendChild(footerIcon);
  556. footerBlock.appendChild(footerText);
  557. footerBlock.appendChild(footerTs);
  558. attachmentBlock.appendChild(titleBlock);
  559. attachmentBlock.appendChild(authorBlock);
  560. attachmentBlock.appendChild(textBlock);
  561. attachmentBlock.appendChild(imgDom);
  562. attachmentBlock.appendChild(footerBlock);
  563. rootDom.appendChild(pretext);
  564. rootDom.appendChild(attachmentBlock);
  565. return rootDom;
  566. }
  567. /**
  568. * @param {string} channelId
  569. * @param {SlackMessage} msg
  570. * @param {boolean=} skipAttachment
  571. * @return {Element}
  572. **/
  573. function doCreateMeMessageDom(channelId, msg, skipAttachment) {
  574. var dom = doCreateMessageDom(channelId, msg, skipAttachment);
  575. dom.classList.add(R.klass.msg.meMessage);
  576. return dom;
  577. }
  578. /**
  579. * @param {string} channelId
  580. * @param {SlackMessage} msg
  581. * @param {boolean=} skipAttachment
  582. * @return {Element}
  583. **/
  584. function createMessageDom(channelId, msg, skipAttachment) {
  585. if (msg.subtype === "me_message") {
  586. return doCreateMeMessageDom(channelId, msg, skipAttachment);
  587. }
  588. return doCreateMessageDom(channelId, msg, skipAttachment);
  589. }
  590. /**
  591. * @param {number} unreadhi
  592. * @param {number} unread
  593. **/
  594. function setFavicon(unreadhi, unread) {
  595. if (!unreadhi && !unread)
  596. document.getElementById(R.id.favicon).href = "favicon_ok.png";
  597. else
  598. document.getElementById(R.id.favicon).href = "favicon.png?h="+unreadhi+"&m="+unread;
  599. }
  600. function setNetErrorFavicon() {
  601. document.getElementById(R.id.favicon).href = "favicon_err.png";
  602. }
  603. function updateTitle() {
  604. var hasUnread = 0
  605. ,hasHl = 0
  606. ,title = "";
  607. if (NEXT_RETRY) {
  608. title = '!' +locale.netErrorShort +' - ';
  609. setNetErrorFavicon();
  610. } else {
  611. for (var i in UNREAD_CHANS) {
  612. if (UNREAD_CHANS.hasOwnProperty(i)) {
  613. hasUnread += UNREAD_CHANS[i].unread;
  614. hasHl += UNREAD_CHANS[i].hl;
  615. }
  616. }
  617. if (hasHl)
  618. title = "(!" +hasHl +") - ";
  619. else if (hasUnread)
  620. title = "(" +hasUnread +") - ";
  621. setFavicon(hasHl, hasUnread);
  622. }
  623. title += SLACK.context.team.name;
  624. document.title = title;
  625. }
  626. function spawnNotification() {
  627. if (!("Notification" in window))
  628. {}
  629. else if (Notification.permission === "granted") {
  630. var now = Date.now();
  631. if (lastNotificationSpawn + NOTIFICATION_COOLDOWN < now) {
  632. var n = new Notification(locale.newMessage);
  633. lastNotificationSpawn = now;
  634. setTimeout(function() {
  635. n.close();
  636. }, NOTIFICATION_DELAY);
  637. }
  638. }
  639. else if (Notification.permission !== "denied")
  640. Notification.requestPermission();
  641. }
  642. function onRoomUpdated() {
  643. var chatFrag = document.createDocumentFragment()
  644. ,currentRoomId = SELECTED_ROOM.id;
  645. document.getElementById(R.id.currentRoom.content).textContent = "";
  646. if (SLACK.history[currentRoomId])
  647. SLACK.history[currentRoomId].messages.forEach(function(msg) {
  648. if (msg.type === "message") {
  649. var dom = createMessageDom(currentRoomId, msg);
  650. chatFrag.appendChild(dom);
  651. }
  652. });
  653. var content = document.getElementById(R.id.currentRoom.content);
  654. content.appendChild(chatFrag);
  655. //TODO scroll lock
  656. content.scrollTop = content.scrollHeight -content.clientHeight;
  657. }
  658. function chatClickDelegate(e) {
  659. var target = e.target
  660. ,getMessageId = function(e, target) {
  661. target = target || e.target;
  662. while (target !== e.currentTarget && target) {
  663. if (target.classList.contains(R.klass.msg.item)) {
  664. return target.id;
  665. }
  666. target = target.parentElement;
  667. }
  668. };
  669. while (target !== e.currentTarget && target) {
  670. if (target.classList.contains(R.klass.msg.hover.container)) {
  671. return;
  672. } else if (target.parentElement && target.parentElement.classList.contains(R.klass.msg.hover.container)) {
  673. var messageId = getMessageId(e, target);
  674. if (messageId) {
  675. messageId = parseFloat(messageId.split("_")[1]);
  676. var msg = SLACK.history[SELECTED_ROOM.id].getMessage(messageId);
  677. if (msg && target.classList.contains(R.klass.msg.hover.reply)) {
  678. if (REPLYING_TO !== msg) {
  679. REPLYING_TO = msg;
  680. onReplyingToUpdated();
  681. }
  682. } else if (msg && target.classList.contains(R.klass.msg.hover.reaction)) {
  683. EMOJI_BAR.spawn(document.body, function(emoji) {
  684. if (emoji)
  685. addReaction(SELECTED_ROOM.id, msg.id, emoji);
  686. });
  687. }
  688. }
  689. return;
  690. }
  691. target = target.parentElement;
  692. }
  693. }
  694. function focusInput() {
  695. document.getElementById(R.id.message.input).focus();
  696. }
  697. function setRoomFromHashBang() {
  698. var hashId = document.location.hash.substr(1)
  699. ,room = SLACK.context.getChannel(hashId)
  700. ,user = SLACK.context.getMember(hashId);
  701. if (room && room !== SELECTED_ROOM)
  702. selectRoom(room);
  703. else if (user && user.ims)
  704. selectRoom(user.ims);
  705. }
  706. document.addEventListener('DOMContentLoaded', function() {
  707. initLang();
  708. document.getElementById(R.id.currentRoom.content).addEventListener("click", chatClickDelegate);
  709. window.addEventListener("hashchange", function(e) {
  710. if (document.location.hash && document.location.hash[0] === '#') {
  711. setRoomFromHashBang();
  712. }
  713. });
  714. document.getElementById(R.id.message.file.cancel).addEventListener("click", function(e) {
  715. e.preventDefault();
  716. document.getElementById(R.id.message.file.error).classList.add(R.klass.hidden);
  717. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  718. document.getElementById(R.id.message.file.fileInput).value = "";
  719. return false;
  720. });
  721. document.getElementById(R.id.message.file.form).addEventListener("submit", function(e) {
  722. e.preventDefault();
  723. var fileInput = document.getElementById(R.id.message.file.fileInput)
  724. ,filename = fileInput.value;
  725. if (filename) {
  726. filename = filename.substr(filename.lastIndexOf('\\') +1);
  727. uploadFile(SELECTED_ROOM, filename, fileInput.files[0], function(errorMsg) {
  728. var error = document.getElementById(R.id.message.file.error);
  729. if (errorMsg) {
  730. error.textContent = errorMsg;
  731. error.classList.remove(R.klass.hidden);
  732. } else {
  733. error.classList.add(R.klass.hidden);
  734. document.getElementById(R.id.message.file.fileInput).value = "";
  735. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  736. }
  737. });
  738. }
  739. return false;
  740. });
  741. document.getElementById(R.id.message.file.bt).addEventListener("click", function(e) {
  742. e.preventDefault();
  743. if (SELECTED_ROOM) {
  744. document.getElementById(R.id.message.file.formContainer).classList.remove(R.klass.hidden);
  745. }
  746. return false;
  747. });
  748. document.getElementById(R.id.message.form).addEventListener("submit", function(e) {
  749. e.preventDefault();
  750. var input =document.getElementById(R.id.message.input);
  751. if (SELECTED_ROOM && input.value) {
  752. sendMsg(SELECTED_ROOM, input.value, REPLYING_TO);
  753. input.value = "";
  754. if (REPLYING_TO) {
  755. REPLYING_TO = null;
  756. onReplyingToUpdated();
  757. }
  758. }
  759. focusInput();
  760. return false;
  761. });
  762. window.addEventListener('blur', function() {
  763. window.hasFocus = false;
  764. });
  765. window.addEventListener('focus', function() {
  766. window.hasFocus = true;
  767. lastNotificationSpawn = 0;
  768. if (SELECTED_ROOM)
  769. markRoomAsRead(SELECTED_ROOM);
  770. focusInput();
  771. });
  772. window.hasFocus = true;
  773. //Emoji closure
  774. (function() {
  775. var emojiButton = document.getElementById(R.id.message.emoji);
  776. if ('makeEmoji' in window) {
  777. var emojiDom = window['makeEmoji']('smile');
  778. if (emojiDom) {
  779. emojiButton.innerHTML = "<span class='" +R.klass.emoji.small +"'>" +emojiDom.outerHTML +"</span>";
  780. } else {
  781. emojiButton.style.backgroundImage = 'url("smile.svg")';
  782. }
  783. emojiDom = window['makeEmoji']('paperclip');
  784. if (emojiDom) {
  785. document.getElementById(R.id.message.file.bt).innerHTML = "<span class='" +R.klass.emoji.small +"'>" +emojiDom.outerHTML +"</span>";
  786. } else {
  787. document.getElementById(R.id.message.file.bt).style.backgroundImage = 'url("public/paperclip.svg")';
  788. }
  789. emojiButton.addEventListener("click", function() {
  790. EMOJI_BAR.spawn(document.body, function(e) {
  791. if (e) document.getElementById(R.id.message.input).value += ":"+e+":";
  792. focusInput();
  793. });
  794. });
  795. } else {
  796. emojiButton.classList.add(R.klass.hidden);
  797. }
  798. })();
  799. startPolling();
  800. });