ui.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322
  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. markRoomAsRead(SELECTED_ROOM);
  79. if (REPLYING_TO) {
  80. REPLYING_TO = null;
  81. onReplyingToUpdated();
  82. }
  83. }
  84. function onReplyingToUpdated() {
  85. if (REPLYING_TO) {
  86. document.body.classList.add(R.klass.replyingTo);
  87. var domParent = document.getElementById(R.id.message.replyTo)
  88. ,closeLink = document.createElement("a");
  89. closeLink.addEventListener("click", function() {
  90. REPLYING_TO = null;
  91. onReplyingToUpdated();
  92. });
  93. closeLink.className = R.klass.msg.replyTo.close;
  94. closeLink.textContent = 'x';
  95. domParent.textContent = "";
  96. domParent.appendChild(closeLink);
  97. domParent.appendChild(createMessageDom("reply_" +SELECTED_ROOM.id, REPLYING_TO));
  98. } else {
  99. document.body.classList.remove(R.klass.replyingTo);
  100. }
  101. }
  102. /**
  103. * @param {string} channelId
  104. * @param {SlackMessage} msg
  105. **/
  106. function createMessageDom(channelId, msg) {
  107. var dom = document.createElement("div")
  108. ,ts = document.createElement("div")
  109. ,text = document.createElement("div")
  110. ,author = document.createElement("div")
  111. ,authorImg = document.createElement("img")
  112. ,authorName = document.createElement("span")
  113. ,hover = document.createElement("ul")
  114. ,hoverReply = document.createElement("li")
  115. ,sender = msg.raw["user"] ?
  116. SLACK.context.users[msg.raw["user"]] :
  117. SLACK.context.bots[msg.raw["bot_id"]];
  118. dom.id = channelId +"_" +msg.ts;
  119. dom.className = R.klass.msg.item;
  120. ts.className = R.klass.msg.ts;
  121. text.className = R.klass.msg.msg;
  122. author.className = R.klass.msg.author;
  123. authorImg.className = R.klass.msg.authorAvatar;
  124. authorName.className = R.klass.msg.authorname;
  125. hover.className = R.klass.msg.hover.container;
  126. hoverReply.className = R.klass.msg.hover.reply;
  127. ts.textContent = (new Date(msg.ts * 1000)).toLocaleTimeString();
  128. var msgContent = msg.raw["text"] || "";
  129. msgContent = msgContent
  130. .replace(new RegExp('<([@#]?)([^>]*)>', 'g'),
  131. function(_, type, entity) {
  132. var sub = entity.split('|');
  133. if (type === '@') {
  134. if (!sub[1]) {
  135. var user = SLACK.context.getMember(sub[0]);
  136. sub[1] = user ? ('@' +user.name) : "Unknown member"; // TODO locale
  137. } else if ('@' !== sub[1][0]) {
  138. sub[1] = '@' +sub[1];
  139. }
  140. sub[0] = '#' +sub[0];
  141. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
  142. } else if (type === '#') {
  143. if (!sub[1]) {
  144. var chan = SLACK.context.getChannel(sub[0]);
  145. sub[1] = chan ? ('#' +chan.name) : "Unknown channel"; // TODO locale
  146. } else if ('#' !== sub[1][0]) {
  147. sub[1] = '#' +sub[1];
  148. }
  149. sub[0] = '#' +sub[0];
  150. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkchan;
  151. } else {
  152. if (!sub[1])
  153. sub[1] = sub[0];
  154. sub[2] = R.klass.msg.link;
  155. }
  156. return '<a href="' +sub[0] +'" class="' +sub[2] +'"' +(!type ? ' target="_blank"' : '') +'>' +sub[1] +'</a>';
  157. })
  158. ;
  159. text.innerHTML = msgContent;
  160. authorName.textContent = sender ? sender.name : (msg.raw["username"] || "?");
  161. authorImg.src = sender ? sender.icons.image_48 : "";
  162. author.appendChild(authorImg);
  163. author.appendChild(authorName);
  164. hover.appendChild(hoverReply);
  165. dom.appendChild(author);
  166. dom.appendChild(text);
  167. dom.appendChild(ts);
  168. dom.appendChild(hover);
  169. return dom;
  170. }
  171. function updateTitle() {
  172. var hasUnread = 0;
  173. for (var i in UNREAD_CHANS) {
  174. if (UNREAD_CHANS.hasOwnProperty(i)) {
  175. hasUnread += UNREAD_CHANS[i];
  176. }
  177. }
  178. document.title = (hasUnread ? ("(" +hasUnread +") - ") : "") +SLACK.context.team.name;
  179. }
  180. /**
  181. * @param {SlackChan|SlackGroup|SlackIms} chan
  182. * @param {Array.<*>} msg
  183. **/
  184. function onMsgReceived(chan, msg) {
  185. if (chan && (chan !== SELECTED_ROOM || !window.hasFocus)) {
  186. document.getElementById(chan.id).classList.add(R.klass.unread);
  187. var count = UNREAD_CHANS[chan.id] || 0;
  188. UNREAD_CHANS[chan.id] = count + msg.length;
  189. updateTitle();
  190. }
  191. }
  192. /**
  193. * @param {SlackChan|SlackGroup|SlackIms} room
  194. **/
  195. function markRoomAsRead(room) {
  196. if (UNREAD_CHANS[room.id]) {
  197. UNREAD_CHANS[room.id] = 0;
  198. updateTitle();
  199. }
  200. var roomLi = document.getElementById(room.id);
  201. roomLi.classList.remove(R.klass.unread);
  202. roomLi.classList.remove(R.klass.unreadHi);
  203. }
  204. function onRoomUpdated() {
  205. var chatFrag = document.createDocumentFragment()
  206. ,currentRoomId = SELECTED_ROOM.id;
  207. document.getElementById(R.id.currentRoom.content).textContent = "";
  208. if (SLACK.history[currentRoomId])
  209. SLACK.history[currentRoomId].messages.forEach(function(msg) {
  210. chatFrag.appendChild(createMessageDom(currentRoomId, msg));
  211. });
  212. var content = document.getElementById(R.id.currentRoom.content);
  213. content.appendChild(chatFrag);
  214. //TODO scroll lock
  215. content.scrollTop = content.scrollHeight -content.clientHeight;
  216. }
  217. function onChanClick(e) {
  218. while (e.target !== e.currentTarget && e.target) {
  219. if (e.target.classList.contains(R.klass.chatList.entry)) {
  220. var room = (SLACK.context.channels[e.target.id] ||
  221. SLACK.context.ims[e.target.id] ||
  222. SLACK.context.groups[e.target.id]);
  223. if (room && room !== SELECTED_ROOM) {
  224. selectRoom(room);
  225. }
  226. return;
  227. }
  228. e.target = e.target.parentElement;
  229. }
  230. }
  231. function chatClickDelegate(e) {
  232. var target = e.target
  233. ,getMessageId = function(e, target) {
  234. target = target || e.target;
  235. while (target !== e.currentTarget && target) {
  236. if (target.classList.contains(R.klass.msg.item)) {
  237. return target.id;
  238. }
  239. target = target.parentElement;
  240. }
  241. };
  242. while (target !== e.currentTarget && target) {
  243. if (target.classList.contains(R.klass.msg.hover.container)) {
  244. return;
  245. } else if (target.classList.contains(R.klass.msg.hover.reply)) {
  246. var messageId = getMessageId(e, target);
  247. if (messageId) {
  248. messageId = parseFloat(messageId.split("_")[1]);
  249. var history = SLACK.history[SELECTED_ROOM.id].messages;
  250. for (var i =0, histLen =history.length; i < histLen && history[i].ts <= messageId; i++) {
  251. if (history[i].ts === messageId) {
  252. if (REPLYING_TO !== history[i]) {
  253. REPLYING_TO = history[i];
  254. onReplyingToUpdated();
  255. }
  256. return;
  257. }
  258. }
  259. }
  260. return;
  261. }
  262. target = target.parentElement;
  263. }
  264. }
  265. function focusInput() {
  266. document.getElementById(R.id.message.input).focus();
  267. }
  268. document.addEventListener('DOMContentLoaded', function() {
  269. document.getElementById(R.id.chatList).addEventListener("click", onChanClick);
  270. document.getElementById(R.id.currentRoom.content).addEventListener("click", chatClickDelegate);
  271. document.getElementById(R.id.message.form).addEventListener("submit", function(e) {
  272. e.preventDefault();
  273. var input =document.getElementById(R.id.message.input);
  274. if (SELECTED_ROOM && input.value) {
  275. sendMsg(SELECTED_ROOM, input.value, REPLYING_TO);
  276. input.value = "";
  277. if (REPLYING_TO) {
  278. REPLYING_TO = null;
  279. onReplyingToUpdated();
  280. }
  281. }
  282. focusInput();
  283. return false;
  284. });
  285. window.addEventListener('blur', function() {
  286. window.hasFocus = false;
  287. });
  288. window.addEventListener('focus', function() {
  289. window.hasFocus = true;
  290. if (SELECTED_ROOM)
  291. markRoomAsRead(SELECTED_ROOM);
  292. focusInput();
  293. });
  294. window.hasFocus = true;
  295. startPolling();
  296. });