ui.js 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461
  1. var
  2. /** @type {SlackMessage|null} */
  3. REPLYING_TO = null
  4. ;
  5. /**
  6. * @param {SlackChan|SlackGroup} chan
  7. * @return {Element}
  8. **/
  9. function createChanListItem(chan) {
  10. var dom = document.createElement("li");
  11. dom.id = chan.id;
  12. if (chan.id[0] === 'D')
  13. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeDirect;
  14. else if (chan.id[0] === 'G')
  15. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeGroup;
  16. else if (chan.id[0] === 'C')
  17. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeChannel;
  18. dom.textContent = chan.name;
  19. return dom;
  20. }
  21. /**
  22. * @param {SlackIms} ims
  23. * @return {Element}
  24. **/
  25. function createImsListItem(ims) {
  26. var dom = document.createElement("li");
  27. dom.id = ims.id;
  28. dom.className = R.klass.chatList.entry;
  29. dom.textContent = ims.user.name;
  30. return dom;
  31. }
  32. function onContextUpdated() {
  33. var chanListFram = document.createDocumentFragment();
  34. var sortedChans = SLACK.context.self ? Object.keys(SLACK.context.self.channels) : [];
  35. sortedChans.sort(function(a, b) {
  36. if (a[0] !== b[0]) {
  37. return a[0] - b[0];
  38. }
  39. return (SLACK.context.channels[a] || SLACK.context.groups[a]).name.localeCompare((SLACK.context.channels[b] || SLACK.context.groups[b]).name);
  40. });
  41. sortedChans.forEach(function(chanId) {
  42. var chan = SLACK.context.channels[chanId] || SLACK.context.groups[chanId]
  43. ,chanListItem = createChanListItem(chan);
  44. if (chanListItem) {
  45. chanListFram.appendChild(chanListItem);
  46. }
  47. });
  48. var sortedUsers = SLACK.context.users ? Object.keys(SLACK.context.users) : [];
  49. sortedUsers.sort(function(a, b) {
  50. return SLACK.context.users[a].name.localeCompare(SLACK.context.users[b].name);
  51. });
  52. sortedUsers.forEach(function(userId) {
  53. var ims = SLACK.context.users[userId].ims
  54. ,imsListItem = createImsListItem(ims);
  55. if (imsListItem) {
  56. chanListFram.appendChild(imsListItem);
  57. }
  58. });
  59. document.getElementById(R.id.chanList).textContent = "";
  60. document.getElementById(R.id.chanList).appendChild(chanListFram);
  61. }
  62. function onNetworkStateUpdated(isNetworkWorking) {
  63. isNetworkWorking ? document.body.classList.remove(R.klass.noNetwork) : document.body.classList.add(R.klass.noNetwork);
  64. }
  65. function onRoomSelected() {
  66. var name = SELECTED_ROOM.name || (SELECTED_ROOM.user ? SELECTED_ROOM.user.name : undefined);
  67. if (!name) {
  68. var members = [];
  69. for (var i in SELECTED_ROOM.members) {
  70. members.push(SELECTED_ROOM.members[i].name);
  71. }
  72. name = members.join(", ");
  73. }
  74. var roomLi = document.getElementById(SELECTED_ROOM.id);
  75. document.getElementById(R.id.currentRoom.title).textContent = name;
  76. onRoomUpdated();
  77. focusInput();
  78. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  79. markRoomAsRead(SELECTED_ROOM);
  80. if (REPLYING_TO) {
  81. REPLYING_TO = null;
  82. onReplyingToUpdated();
  83. }
  84. }
  85. function onReplyingToUpdated() {
  86. if (REPLYING_TO) {
  87. document.body.classList.add(R.klass.replyingTo);
  88. var domParent = document.getElementById(R.id.message.replyTo)
  89. ,closeLink = document.createElement("a");
  90. closeLink.addEventListener("click", function() {
  91. REPLYING_TO = null;
  92. onReplyingToUpdated();
  93. });
  94. closeLink.className = R.klass.msg.replyTo.close;
  95. closeLink.textContent = 'x';
  96. domParent.textContent = "";
  97. domParent.appendChild(closeLink);
  98. domParent.appendChild(createMessageDom("reply_" +SELECTED_ROOM.id, REPLYING_TO));
  99. } else {
  100. document.body.classList.remove(R.klass.replyingTo);
  101. }
  102. }
  103. /**
  104. * @param {string} channelId
  105. * @param {SlackMessage} msg
  106. * @return {Element}
  107. **/
  108. function doCreateMessageDom(channelId, msg) {
  109. var dom = document.createElement("div")
  110. ,ts = document.createElement("div")
  111. ,text = document.createElement("div")
  112. ,author = document.createElement("div")
  113. ,authorImg = document.createElement("img")
  114. ,authorName = document.createElement("span")
  115. ,hover = document.createElement("ul")
  116. ,hoverReply = document.createElement("li")
  117. ,sender = msg.raw["user"] ?
  118. SLACK.context.users[msg.raw["user"]] :
  119. SLACK.context.bots[msg.raw["bot_id"]];
  120. dom.id = channelId +"_" +msg.ts;
  121. dom.className = R.klass.msg.item;
  122. ts.className = R.klass.msg.ts;
  123. text.className = R.klass.msg.msg;
  124. author.className = R.klass.msg.author;
  125. authorImg.className = R.klass.msg.authorAvatar;
  126. authorName.className = R.klass.msg.authorname;
  127. hover.className = R.klass.msg.hover.container;
  128. hoverReply.className = R.klass.msg.hover.reply;
  129. ts.textContent = (new Date(msg.ts * 1000)).toLocaleTimeString();
  130. var msgContents = msg.raw["text"] || "";
  131. msgContents = msgContents
  132. .replace(new RegExp('<([@#]?)([^>]*)>', 'g'),
  133. function(_, type, entity) {
  134. var sub = entity.split('|');
  135. if (type === '@') {
  136. if (!sub[1]) {
  137. var user = SLACK.context.getMember(sub[0]);
  138. sub[1] = user ? ('@' +user.name) : "Unknown member"; // TODO locale
  139. } else if ('@' !== sub[1][0]) {
  140. sub[1] = '@' +sub[1];
  141. }
  142. sub[0] = '#' +sub[0];
  143. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
  144. } else if (type === '#') {
  145. if (!sub[1]) {
  146. var chan = SLACK.context.getChannel(sub[0]);
  147. sub[1] = chan ? ('#' +chan.name) : "Unknown channel"; // TODO locale
  148. } else if ('#' !== sub[1][0]) {
  149. sub[1] = '#' +sub[1];
  150. }
  151. sub[0] = '#' +sub[0];
  152. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkchan;
  153. } else {
  154. if (!sub[1])
  155. sub[1] = sub[0];
  156. sub[2] = R.klass.msg.link;
  157. }
  158. return '<a href="' +sub[0] +'" class="' +sub[2] +'"' +(!type ? ' target="_blank"' : '') +'>' +sub[1] +'</a>';
  159. })
  160. .split(/\r?\n/g);
  161. ;
  162. for (var msgContentIndex=0, nbMsgContents = msgContents.length; msgContentIndex < nbMsgContents; msgContentIndex++) {
  163. var msgContent = msgContents[msgContentIndex]
  164. ,_msgContent = ""
  165. ,currentMods = {}
  166. ,quote = false
  167. ,i =0
  168. ,msgLength = msgContent.length;
  169. var checkEnd = function(str, pos, c) {
  170. while (str[pos]) {
  171. if (str[pos] != ' ' && str[pos] != c && str[pos +1] == c) {
  172. return true;
  173. }
  174. pos++;
  175. }
  176. return false;
  177. } ,appendMod = function(mods) {
  178. if (!Object.keys(currentMods).length)
  179. return "";
  180. return '<span class="' +Object.keys(mods).join(' ') +'">';
  181. };
  182. // Skip trailing
  183. while (i < msgLength && (msgContent[i] === ' ' || msgContent[i] === '\t'))
  184. i++;
  185. if (msgContent.substr(i, 4) === '&gt;') {
  186. quote = true;
  187. i += 4;
  188. }
  189. for (; i < msgLength; i++) {
  190. var c = msgContent[i];
  191. if (!(currentMods[R.klass.msg.style.bold]) && c === '*' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  192. if (Object.keys(currentMods).length)
  193. _msgContent += '</span>';
  194. currentMods[R.klass.msg.style.bold] = true;
  195. _msgContent += appendMod(currentMods);
  196. } else if (!(currentMods[R.klass.msg.style.strike]) && c === '~' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  197. if (Object.keys(currentMods).length)
  198. _msgContent += '</span>';
  199. currentMods[R.klass.msg.style.strike] = true;
  200. _msgContent += appendMod(currentMods);
  201. } else if (!(currentMods[R.klass.msg.style.code]) && c === '`' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  202. if (Object.keys(currentMods).length)
  203. _msgContent += '</span>';
  204. currentMods[R.klass.msg.style.code] = true;
  205. _msgContent += appendMod(currentMods);
  206. } else if (!(currentMods[R.klass.msg.style.italic]) && c === '_' && msgContent[i +1] && checkEnd(msgContent, i, c)) {
  207. if (Object.keys(currentMods).length)
  208. _msgContent += '</span>';
  209. currentMods[R.klass.msg.style.italic] = true;
  210. _msgContent += appendMod(currentMods);
  211. } else {
  212. var finalFound = false;
  213. _msgContent += c;
  214. do {
  215. if ((currentMods[R.klass.msg.style.bold]) && c !== '*' && msgContent[i +1] === '*') {
  216. delete currentMods[R.klass.msg.style.bold];
  217. finalFound = true;
  218. } else if ((currentMods[R.klass.msg.style.strike]) && c !== '~' && msgContent[i +1] === '~') {
  219. delete currentMods[R.klass.msg.style.strike];
  220. finalFound = true;
  221. } else if ((currentMods[R.klass.msg.style.code]) && c !== '`' && msgContent[i +1] === '`') {
  222. delete currentMods[R.klass.msg.style.code];
  223. finalFound = true;
  224. } else if ((currentMods[R.klass.msg.style.italic]) && c !== '_' && msgContent[i +1] === '_') {
  225. delete currentMods[R.klass.msg.style.italic];
  226. finalFound = true;
  227. } else {
  228. break;
  229. }
  230. c = msgContent[++i];
  231. } while (i < msgLength);
  232. if (finalFound)
  233. _msgContent += '</span>' +appendMod(currentMods);
  234. }
  235. }
  236. if (currentMods) {
  237. // Should not append
  238. _msgContent += '</span>';
  239. }
  240. if (quote)
  241. msgContents[msgContentIndex] = '<span class="' +R.klass.msg.style.quote +'">' +_msgContent +'</span>';
  242. else
  243. msgContents[msgContentIndex] = _msgContent;
  244. }
  245. text.innerHTML = msgContents.join('<br/>');
  246. authorName.textContent = sender ? sender.name : (msg.raw["username"] || "?");
  247. if (!sender && !msg.raw["username"])
  248. text.textContent = msg.raw["subtype"] || JSON.stringify(msg.raw);
  249. authorImg.src = sender ? sender.icons.image_48 : "";
  250. author.appendChild(authorImg);
  251. author.appendChild(authorName);
  252. hover.appendChild(hoverReply);
  253. dom.appendChild(author);
  254. dom.appendChild(text);
  255. dom.appendChild(ts);
  256. dom.appendChild(hover);
  257. return dom;
  258. }
  259. /**
  260. * @param {string} channelId
  261. * @param {SlackMessage} msg
  262. * @return {Element}
  263. **/
  264. function doCreateMeMessageDom(channelId, msg) {
  265. var dom = doCreateMessageDom(channelId, msg);
  266. dom.classList.add(R.klass.msg.meMessage);
  267. return dom;
  268. }
  269. /**
  270. * @param {string} channelId
  271. * @param {SlackMessage} msg
  272. * @return {Element}
  273. **/
  274. function createMessageDom(channelId, msg) {
  275. if (msg.subtype === "me_message") {
  276. return doCreateMeMessageDom(channelId, msg);
  277. }
  278. return doCreateMessageDom(channelId, msg);
  279. }
  280. function updateTitle() {
  281. var hasUnread = 0
  282. ,hasHl = 0
  283. ,title;
  284. for (var i in UNREAD_CHANS) {
  285. if (UNREAD_CHANS.hasOwnProperty(i)) {
  286. hasUnread += UNREAD_CHANS[i].unread;
  287. hasHl += UNREAD_CHANS[i].hl;
  288. }
  289. }
  290. if (hasHl) {
  291. title = "(!" +hasHl;
  292. }
  293. if (hasUnread) {
  294. title = (title ? (title+" - ") : "(") +hasUnread;
  295. }
  296. if (title)
  297. title += ") " +SLACK.context.team.name;
  298. else
  299. title = SLACK.context.team.name;
  300. document.title = title;
  301. }
  302. function onRoomUpdated() {
  303. var chatFrag = document.createDocumentFragment()
  304. ,currentRoomId = SELECTED_ROOM.id;
  305. document.getElementById(R.id.currentRoom.content).textContent = "";
  306. if (SLACK.history[currentRoomId])
  307. SLACK.history[currentRoomId].messages.forEach(function(msg) {
  308. if (msg.type === "message") {
  309. var dom = createMessageDom(currentRoomId, msg);
  310. chatFrag.appendChild(dom);
  311. }
  312. });
  313. var content = document.getElementById(R.id.currentRoom.content);
  314. content.appendChild(chatFrag);
  315. //TODO scroll lock
  316. content.scrollTop = content.scrollHeight -content.clientHeight;
  317. }
  318. function onChanClick(e) {
  319. while (e.target !== e.currentTarget && e.target) {
  320. if (e.target.classList.contains(R.klass.chatList.entry)) {
  321. var room = (SLACK.context.channels[e.target.id] ||
  322. SLACK.context.ims[e.target.id] ||
  323. SLACK.context.groups[e.target.id]);
  324. if (room && room !== SELECTED_ROOM) {
  325. selectRoom(room);
  326. }
  327. return;
  328. }
  329. e.target = e.target.parentElement;
  330. }
  331. }
  332. function chatClickDelegate(e) {
  333. var target = e.target
  334. ,getMessageId = function(e, target) {
  335. target = target || e.target;
  336. while (target !== e.currentTarget && target) {
  337. if (target.classList.contains(R.klass.msg.item)) {
  338. return target.id;
  339. }
  340. target = target.parentElement;
  341. }
  342. };
  343. while (target !== e.currentTarget && target) {
  344. if (target.classList.contains(R.klass.msg.hover.container)) {
  345. return;
  346. } else if (target.classList.contains(R.klass.msg.hover.reply)) {
  347. var messageId = getMessageId(e, target);
  348. if (messageId) {
  349. messageId = parseFloat(messageId.split("_")[1]);
  350. var history = SLACK.history[SELECTED_ROOM.id].messages;
  351. for (var i =0, histLen =history.length; i < histLen && history[i].ts <= messageId; i++) {
  352. if (history[i].ts === messageId) {
  353. if (REPLYING_TO !== history[i]) {
  354. REPLYING_TO = history[i];
  355. onReplyingToUpdated();
  356. }
  357. return;
  358. }
  359. }
  360. }
  361. return;
  362. }
  363. target = target.parentElement;
  364. }
  365. }
  366. function focusInput() {
  367. document.getElementById(R.id.message.input).focus();
  368. }
  369. document.addEventListener('DOMContentLoaded', function() {
  370. document.getElementById(R.id.chatList).addEventListener("click", onChanClick);
  371. document.getElementById(R.id.currentRoom.content).addEventListener("click", chatClickDelegate);
  372. document.getElementById(R.id.message.file.cancel).addEventListener("click", function(e) {
  373. e.preventDefault();
  374. document.getElementById(R.id.message.file.error).classList.add(R.klass.hidden);
  375. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  376. document.getElementById(R.id.message.file.fileInput).value = "";
  377. return false;
  378. });
  379. document.getElementById(R.id.message.file.form).addEventListener("submit", function(e) {
  380. e.preventDefault();
  381. var fileInput = document.getElementById(R.id.message.file.fileInput)
  382. ,filename = fileInput.value;
  383. if (filename) {
  384. filename = filename.substr(filename.lastIndexOf('\\') +1);
  385. uploadFile(SELECTED_ROOM, filename, fileInput.files[0], function(errorMsg) {
  386. var error = document.getElementById(R.id.message.file.error);
  387. if (errorMsg) {
  388. error.textContent = errorMsg;
  389. error.classList.remove(R.klass.hidden);
  390. } else {
  391. error.classList.add(R.klass.hidden);
  392. document.getElementById(R.id.message.file.fileInput).value = "";
  393. document.getElementById(R.id.message.file.formContainer).classList.add(R.klass.hidden);
  394. }
  395. });
  396. }
  397. return false;
  398. });
  399. document.getElementById(R.id.message.file.bt).addEventListener("click", function(e) {
  400. e.preventDefault();
  401. if (SELECTED_ROOM) {
  402. document.getElementById(R.id.message.file.formContainer).classList.remove(R.klass.hidden);
  403. }
  404. return false;
  405. });
  406. document.getElementById(R.id.message.form).addEventListener("submit", function(e) {
  407. e.preventDefault();
  408. var input =document.getElementById(R.id.message.input);
  409. if (SELECTED_ROOM && input.value) {
  410. sendMsg(SELECTED_ROOM, input.value, REPLYING_TO);
  411. input.value = "";
  412. if (REPLYING_TO) {
  413. REPLYING_TO = null;
  414. onReplyingToUpdated();
  415. }
  416. }
  417. focusInput();
  418. return false;
  419. });
  420. window.addEventListener('blur', function() {
  421. window.hasFocus = false;
  422. });
  423. window.addEventListener('focus', function() {
  424. window.hasFocus = true;
  425. if (SELECTED_ROOM)
  426. markRoomAsRead(SELECTED_ROOM);
  427. focusInput();
  428. });
  429. window.hasFocus = true;
  430. startPolling();
  431. });