ui.js 32 KB

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