ui.js 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. /**
  2. * @param {SlackChan|SlackGroup} chan
  3. * @return {Element}
  4. **/
  5. function createChanListItem(chan) {
  6. var dom = document.createElement("li");
  7. dom.id = chan.id;
  8. if (chan.id[0] === 'D')
  9. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeDirect;
  10. else if (chan.id[0] === 'G')
  11. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeGroup;
  12. else if (chan.id[0] === 'C')
  13. dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeChannel;
  14. dom.textContent = chan.name;
  15. return dom;
  16. }
  17. /**
  18. * @param {SlackIms} ims
  19. * @return {Element}
  20. **/
  21. function createImsListItem(ims) {
  22. var dom = document.createElement("li");
  23. dom.id = ims.id;
  24. dom.className = R.klass.chatList.entry;
  25. dom.textContent = ims.user.name;
  26. return dom;
  27. }
  28. function onContextUpdated() {
  29. var chanListFram = document.createDocumentFragment();
  30. var sortedChans = SLACK.context.self ? Object.keys(SLACK.context.self.channels) : [];
  31. sortedChans.sort(function(a, b) {
  32. if (a[0] !== b[0]) {
  33. return a[0] - b[0];
  34. }
  35. return (SLACK.context.channels[a] || SLACK.context.groups[a]).name.localeCompare((SLACK.context.channels[b] || SLACK.context.groups[b]).name);
  36. });
  37. sortedChans.forEach(function(chanId) {
  38. var chan = SLACK.context.channels[chanId] || SLACK.context.groups[chanId]
  39. ,chanListItem = createChanListItem(chan);
  40. if (chanListItem) {
  41. chanListFram.appendChild(chanListItem);
  42. }
  43. });
  44. var sortedUsers = SLACK.context.users ? Object.keys(SLACK.context.users) : [];
  45. sortedUsers.sort(function(a, b) {
  46. return SLACK.context.users[a].name.localeCompare(SLACK.context.users[b].name);
  47. });
  48. sortedUsers.forEach(function(userId) {
  49. var ims = SLACK.context.users[userId].ims
  50. ,imsListItem = createImsListItem(ims);
  51. if (imsListItem) {
  52. chanListFram.appendChild(imsListItem);
  53. }
  54. });
  55. document.getElementById(R.id.chanList).textContent = "";
  56. document.getElementById(R.id.chanList).appendChild(chanListFram);
  57. }
  58. function onNetworkStateUpdated(isNetworkWorking) {
  59. isNetworkWorking ? document.body.classList.remove(R.klass.noNetwork) : document.body.classList.add(R.klass.noNetwork);
  60. }
  61. function onRoomSelected() {
  62. var name = SELECTED_ROOM.name || (SELECTED_ROOM.user ? SELECTED_ROOM.user.name : undefined);
  63. if (!name) {
  64. var members = [];
  65. for (var i in SELECTED_ROOM.members) {
  66. members.push(SELECTED_ROOM.members[i].name);
  67. }
  68. name = members.join(", ");
  69. }
  70. var roomLi = document.getElementById(SELECTED_ROOM.id);
  71. document.getElementById(R.id.currentRoom.title).textContent = name;
  72. onRoomUpdated();
  73. markRoomAsRead(SELECTED_ROOM);
  74. }
  75. function createMessageDom(channelId, msg) {
  76. var dom = document.createElement("div")
  77. ,ts = document.createElement("div")
  78. ,text = document.createElement("div")
  79. ,author = document.createElement("div")
  80. ,authorImg = document.createElement("img")
  81. ,authorName = document.createElement("span")
  82. ,hover = document.createElement("ul")
  83. ,hoverReply = document.createElement("li")
  84. ,sender = msg.raw["user"] ?
  85. SLACK.context.users[msg.raw["user"]] :
  86. SLACK.context.bots[msg.raw["bot_id"]];
  87. dom.id = channelId +"_" +msg.ts;
  88. dom.className = R.klass.msg.item;
  89. ts.className = R.klass.msg.ts;
  90. text.className = R.klass.msg.msg;
  91. author.className = R.klass.msg.author;
  92. authorImg.className = R.klass.msg.authorAvatar;
  93. authorName.className = R.klass.msg.authorname;
  94. hover.className = R.klass.msg.hover.container;
  95. hoverReply.className = R.klass.msg.hover.reply;
  96. ts.textContent = (new Date(msg.ts * 1000)).toLocaleTimeString();
  97. var msgContent = msg.raw["text"] || "";
  98. msgContent = msgContent
  99. .replace(new RegExp('<([@#]?)([^>]*)>', 'g'),
  100. function(_, type, entity) {
  101. var sub = entity.split('|');
  102. if (type === '@') {
  103. if (!sub[1]) {
  104. var user = SLACK.context.getMember(sub[0]);
  105. sub[1] = user ? user.name : "Unknown member"; // TODO locale
  106. }
  107. sub[0] = '#' +sub[0];
  108. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
  109. } else if (type === '#') {
  110. if (!sub[1]) {
  111. var chan = SLACK.context.getChannel(sub[0]);
  112. sub[1] = chan ? ('#' +chan.name) : "Unknown channel"; // TODO locale
  113. } else if ('#' !== sub[1][0]) {
  114. sub[1] = '#' +sub[1];
  115. }
  116. sub[0] = '#' +sub[0];
  117. sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkchan;
  118. } else {
  119. if (!sub[1])
  120. sub[1] = sub[0];
  121. sub[2] = R.klass.msg.link;
  122. }
  123. return '<a href="' +sub[0] +'" class="' +sub[2] +'"' +(!type ? ' target="_blank"' : '') +'>' +sub[1] +'</a>';
  124. })
  125. ;
  126. text.innerHTML = msgContent;
  127. authorName.textContent = sender ? sender.name : (msg.raw["username"] || "?");
  128. authorImg.src = sender ? sender.icons.image_48 : "";
  129. author.appendChild(authorImg);
  130. author.appendChild(authorName);
  131. hover.appendChild(hoverReply);
  132. dom.appendChild(author);
  133. dom.appendChild(text);
  134. dom.appendChild(ts);
  135. dom.appendChild(hover);
  136. return dom;
  137. }
  138. function updateTitle() {
  139. var hasUnread = 0;
  140. for (var i in UNREAD_CHANS) {
  141. if (UNREAD_CHANS.hasOwnProperty(i)) {
  142. hasUnread += UNREAD_CHANS[i];
  143. }
  144. }
  145. document.title = (hasUnread ? ("(" +hasUnread +") - ") : "") +SLACK.context.team.name;
  146. }
  147. /**
  148. * @param {SlackChan|SlackGroup|SlackIms} chan
  149. * @param {Array.<*>} msg
  150. **/
  151. function onMsgReceived(chan, msg) {
  152. if (chan && (chan !== SELECTED_ROOM || !window.hasFocus)) {
  153. document.getElementById(chan.id).classList.add(R.klass.unread);
  154. var count = UNREAD_CHANS[chan.id] || 0;
  155. UNREAD_CHANS[chan.id] = count + msg.length;
  156. updateTitle();
  157. }
  158. }
  159. /**
  160. * @param {SlackChan|SlackGroup|SlackIms} room
  161. **/
  162. function markRoomAsRead(room) {
  163. if (UNREAD_CHANS[room.id]) {
  164. UNREAD_CHANS[room.id] = 0;
  165. updateTitle();
  166. }
  167. var roomLi = document.getElementById(room.id);
  168. roomLi.classList.remove(R.klass.unread);
  169. roomLi.classList.remove(R.klass.unreadHi);
  170. }
  171. function onRoomUpdated() {
  172. var chatFrag = document.createDocumentFragment()
  173. ,currentRoomId = SELECTED_ROOM.id;
  174. document.getElementById(R.id.currentRoom.content).textContent = "";
  175. if (SLACK.history[currentRoomId])
  176. SLACK.history[currentRoomId].messages.forEach(function(msg) {
  177. chatFrag.appendChild(createMessageDom(currentRoomId, msg));
  178. });
  179. var content = document.getElementById(R.id.currentRoom.content);
  180. content.appendChild(chatFrag);
  181. //TODO scroll lock
  182. content.scrollTop = content.scrollHeight -content.clientHeight;
  183. }
  184. function onChanClick(e) {
  185. while (e.target !== e.currentTarget && e.target) {
  186. if (e.target.classList.contains(R.klass.chatList.entry)) {
  187. var room = (SLACK.context.channels[e.target.id] ||
  188. SLACK.context.ims[e.target.id] ||
  189. SLACK.context.groups[e.target.id]);
  190. if (room && room !== SELECTED_ROOM) {
  191. selectRoom(room);
  192. }
  193. return;
  194. }
  195. e.target = e.target.parentElement;
  196. }
  197. }
  198. function replyTo(msg) {
  199. console.log("Replying to ", msg);
  200. }
  201. function chatClickDelegate(e) {
  202. var target = e.target
  203. ,getMessageId = function(e, target) {
  204. target = target || e.target;
  205. while (target !== e.currentTarget && target) {
  206. if (target.classList.contains(R.klass.msg.item)) {
  207. return target.id;
  208. }
  209. target = target.parentElement;
  210. }
  211. };
  212. while (target !== e.currentTarget && target) {
  213. if (target.classList.contains(R.klass.msg.hover.container)) {
  214. return;
  215. } else if (target.classList.contains(R.klass.msg.hover.reply)) {
  216. var messageId = getMessageId(e, target);
  217. if (messageId) {
  218. messageId = parseFloat(messageId.split("_")[1]);
  219. var history = SLACK.history[SELECTED_ROOM.id].messages;
  220. for (var i =0, histLen =history.length; i < histLen && history[i].ts <= messageId; i++) {
  221. if (history[i].ts === messageId) {
  222. replyTo(history[i]);
  223. return;
  224. }
  225. }
  226. }
  227. return;
  228. }
  229. target = target.parentElement;
  230. }
  231. }
  232. document.addEventListener('DOMContentLoaded', function() {
  233. document.getElementById(R.id.chatList).addEventListener("click", onChanClick);
  234. document.getElementById(R.id.currentRoom.content).addEventListener("click", chatClickDelegate);
  235. document.getElementById(R.id.message.form).addEventListener("submit", function(e) {
  236. e.preventDefault();
  237. var input =document.getElementById(R.id.message.input);
  238. if (SELECTED_ROOM && input.value) {
  239. sendMsg(SELECTED_ROOM, input.value);
  240. input.value = "";
  241. }
  242. return false;
  243. });
  244. window.addEventListener('blur', function() {
  245. window.hasFocus = false;
  246. });
  247. window.addEventListener('focus', function() {
  248. window.hasFocus = true;
  249. if (SELECTED_ROOM)
  250. markRoomAsRead(SELECTED_ROOM);
  251. });
  252. window.hasFocus = true;
  253. startPolling();
  254. });