ui.js 35 KB

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