Bläddra i källkod

TMP backup started slack abstraction

B Thibault 8 år sedan
förälder
incheckning
93083308ef
14 ändrade filer med 681 tillägg och 506 borttagningar
  1. 4 2
      Makefile
  2. 12 11
      cli/data.js
  3. 18 18
      cli/dom.js
  4. 1 1
      cli/resources.js
  5. 24 25
      cli/ui.js
  6. 17 17
      cli/workflow.js
  7. 62 75
      srv/public/slack.min.js
  8. 1 1
      srv/public/style.css
  9. 44 0
      srv/src/chatter.js
  10. 59 0
      srv/src/context.js
  11. 308 0
      srv/src/message.js
  12. 70 0
      srv/src/room.js
  13. 55 212
      srv/src/slackData.js
  14. 6 144
      srv/src/slackHistory.js

+ 4 - 2
Makefile

@@ -1,6 +1,8 @@
 
-SRC=		srv/src/slackData.js	\
-			srv/src/slackHistory.js	\
+SRC=		srv/src/context.js		\
+			srv/src/room.js			\
+			srv/src/message.js		\
+			srv/src/chatter.js		\
 			\
 			cli/lang/core.js		\
 			cli/lang/fr.js			\

+ 12 - 11
cli/data.js

@@ -6,7 +6,7 @@ var
     SLACK
 
     /**
-     * @type {Array.<SlackChan|SlackGroup|SlackIms>}
+     * @type {Array.<Room>}
     **/
     ,HIGHLIGHTED_CHANS = [];
 
@@ -19,10 +19,10 @@ function SlackWrapper() {
     /** @type {number} */
     this.lastServerVersion = 0;
 
-    /** @type {SlackData} */
-    this.context = new SlackData(null);
+    /** @type {ChatContext} */
+    this.context = new ChatContext();
 
-    /** @type {!Object.<string, SlackHistory>} **/
+    /** @type {!Object.<string, RoomHistory>} **/
     this.history = {};
 }
 
@@ -33,23 +33,24 @@ SlackWrapper.prototype.update = function(data) {
     if (data["static"]) {
         this.context.updateStatic(data["static"], Date.now());
     }
-    this.context.forEachChans(function(i) {
+    for (var chanId in this.context.channels) {
+        var i = this.context.channels[chanId];
         if (i.lastMsg === i.lastRead) {
             var pos = HIGHLIGHTED_CHANS.indexOf(i);
             if (pos !== -1)
                 HIGHLIGHTED_CHANS.splice(pos, 1);
         }
-    });
+    }
     if (data["live"]) {
         for (var i in data["live"]) {
             var history = this.history[i];
             if (!history)
-                history = this.history[i] = new SlackHistory(i, 250, data["live"][i], now);
+                history = this.history[i] = new RoomHistory(i, 250, data["live"][i], now);
             else
                 history.pushAll(data["live"][i], now);
         }
         for (var roomId in data["live"]) {
-            var chan = this.context.getChannel(roomId);
+            var chan = this.context.channels[roomId];
             if (chan) {
                 if (this.history[roomId].messages.length)
                     chan.lastMsg = Math.max(chan.lastMsg, this.history[roomId].lastMessage().ts);
@@ -89,7 +90,7 @@ function isHighlighted(text) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} chan
+ * @param {Room} chan
  * @param {Array.<*>} msg
 **/
 function onMsgReceived(chan, msg) {
@@ -103,7 +104,7 @@ function onMsgReceived(chan, msg) {
                 return;
             }
             areNew = true;
-            if (chan instanceof SlackIms || i.text.match(selfReg) || isHighlighted(i.text)) {
+            if (chan instanceof PrivateMessageRoom || i.text.match(selfReg) || isHighlighted(i.text)) {
                 if (HIGHLIGHTED_CHANS.indexOf(chan) === -1) {
                     newHighlited = true;
                     HIGHLIGHTED_CHANS.push(chan);
@@ -128,7 +129,7 @@ function onMsgReceived(chan, msg) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} room
+ * @param {Room} room
 **/
 function markRoomAsRead(room) {
     var highlightIndex = HIGHLIGHTED_CHANS.indexOf(room);

+ 18 - 18
cli/dom.js

@@ -22,7 +22,7 @@ function createTypingDisplay() {
 }
 
 /**
- * @param {SlackChan|SlackGroup} chan
+ * @param {Room} chan
  * @return {Element}
 **/
 function createChanListItem(chan) {
@@ -31,12 +31,12 @@ function createChanListItem(chan) {
 
     dom.id = chan.id;
     link.href = '#' +chan.id;
-    if (chan instanceof SlackGroup) {
-        dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeGroup;
-        dom.dataset["count"] = chan.nbMembers -1;
-    }
-    else if (chan instanceof SlackChan)
+    if (chan.isPrivate) {
+        dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typePrivate;
+        dom.dataset["count"] = chan.users.length;
+    } else {
         dom.className = R.klass.chatList.entry + " " +R.klass.chatList.typeChannel;
+    }
     if (SELECTED_ROOM === chan)
         dom.classList.add(R.klass.selected);
     link.textContent = chan.name;
@@ -51,7 +51,7 @@ function createChanListItem(chan) {
 }
 
 /**
- * @param {SlackIms} ims
+ * @param {PrivateMessageRoom} ims
  * @return {Element}
 **/
 function createImsListItem(ims) {
@@ -95,7 +95,7 @@ function createReactionDom(chanId, msgId, reaction, users) {
             ,userNames = [];
 
         for (var i =0, nbUser = users.length; i < nbUser; i++) {
-            var user = SLACK.context.getMember(users[i]);
+            var user = SLACK.context.users[users[i]];
             if (user)
                 userNames.push(user.name);
         }
@@ -115,7 +115,7 @@ function createReactionDom(chanId, msgId, reaction, users) {
 
 /**
  * @param {string} channelId
- * @param {SlackMessage} msg
+ * @param {Message} msg
  * @param {boolean=} skipAttachment
  * @return {Element}
 **/
@@ -130,7 +130,7 @@ function doCreateMessageDom(channelId, msg, skipAttachment) {
         ,hoverReply = document.createElement("li")
         ,attachments = document.createElement("ul")
         ,reactions = document.createElement("ul")
-        ,sender = SLACK.context.getMember(msg.userId);
+        ,sender = SLACK.context.users[msg.userId];
 
     dom.id = channelId +"_" +msg.ts;
     dom.className = R.klass.msg.item;
@@ -141,9 +141,9 @@ function doCreateMessageDom(channelId, msg, skipAttachment) {
     hover.className = R.klass.msg.hover.container;
     hoverReply.className = R.klass.msg.hover.reply;
     ts.innerHTML = locale.formatDate(msg.ts);
-    text.innerHTML = formatSlackText(msg.text);
+    text.innerHTML = formatText(msg.text);
     authorName.textContent = sender ? sender.name : (msg.username || "?");
-    authorImg.src = sender ? sender.icons.image_48 : "";
+    authorImg.src = sender ? sender.icons.small : "";
     hover.appendChild(hoverReply);
     if ('makeEmoji' in window) {
         var hoverReaction = document.createElement("li")
@@ -242,7 +242,7 @@ function doCreateMessageDom(channelId, msg, skipAttachment) {
 
 /**
  * @param {string} channelId
- * @param {SlackMessage} msg
+ * @param {Message} msg
  * @param {*} attachment
  * @return {Element|null}
 **/
@@ -288,7 +288,7 @@ function createAttachmentDom(channelId, msg, attachment) {
     //Pretext
     pretext.className = R.klass.msg.attachment.pretext;
     if (attachment["pretext"]) {
-        pretext.innerHTML = formatSlackText(attachment["pretext"]);
+        pretext.innerHTML = formatText(attachment["pretext"]);
     } else {
         pretext.classList.add(R.klass.hidden);
     }
@@ -296,7 +296,7 @@ function createAttachmentDom(channelId, msg, attachment) {
     //Title
     titleBlock.target = "_blank";
     if (attachment["title"]) {
-        titleBlock.innerHTML = formatSlackText(attachment["title"]);
+        titleBlock.innerHTML = formatText(attachment["title"]);
         if (attachment["title_link"]) {
             titleBlock.href = attachment["title_link"];
         }
@@ -310,7 +310,7 @@ function createAttachmentDom(channelId, msg, attachment) {
     authorName.target = "_blank";
     authorBlock.className = R.klass.msg.author;
     if (attachment["author_name"]) {
-        authorName.innerHTML = formatSlackText(attachment["author_name"]);
+        authorName.innerHTML = formatText(attachment["author_name"]);
         authorName.href = attachment["author_link"] || "";
         authorName.className = R.klass.msg.authorname;
         authorImg.className = R.klass.msg.authorAvatar;
@@ -323,7 +323,7 @@ function createAttachmentDom(channelId, msg, attachment) {
     }
 
     //Text
-    textDom.innerHTML = formatSlackText(attachment["text"] || "");
+    textDom.innerHTML = formatText(attachment["text"] || "");
     textDom.klassName = R.klass.msg.attachment.text;
 
     // Img (small one)
@@ -345,7 +345,7 @@ function createAttachmentDom(channelId, msg, attachment) {
     footerText.className = R.klass.msg.attachment.footerText;
     footerIcon.className = R.klass.msg.attachment.footerIcon;
     if (attachment["footer"]) {
-        footerText.innerHTML = formatSlackText(attachment["footer"]);
+        footerText.innerHTML = formatText(attachment["footer"]);
         if (attachment["footer_icon"])
             footerIcon.src = attachment["footer_icon"];
         else

+ 1 - 1
cli/resources.js

@@ -72,7 +72,7 @@ var R = {
         ,chatList: {
             entry: "slack-context-room"
             ,typeChannel: "slack-channel"
-            ,typeGroup: "slack-group"
+            ,typePrivate: "slack-group"
             ,typeDirect: "slack-ims"
             ,typing: "slack-context-typing"
         }

+ 24 - 25
cli/ui.js

@@ -20,21 +20,16 @@ var
 
 function onContextUpdated() {
     var chanListFram = document.createDocumentFragment()
-        ,sortedChans = SLACK.context.self ? Object.keys(SLACK.context.self.channels) : [];
+        ,sortedChans = (SLACK.context.self ? Object.keys(SLACK.context.self.channels) : []);
 
     sortedChans.sort(function(a, b) {
         if (a[0] !== b[0]) {
             return a[0] - b[0];
         }
-        return SLACK.context.getChannel(a).name.localeCompare(SLACK.context.getChannel(b).name);
+        return SLACK.context.channels[a].name.localeCompare(SLACK.context.channels[b].name);
     });
     sortedChans.forEach(function(chanId) {
-        var chan =
-            /**
-             * SortedChan does not contains ims ids
-             * @type {SlackChan|SlackGroup}
-            **/
-            (SLACK.context.getChannel(chanId));
+        var chan = SLACK.context.channels[chanId];
         if (!chan.archived) {
             var chanListItem = createChanListItem(chan);
             if (chanListItem)
@@ -46,7 +41,7 @@ function onContextUpdated() {
         return SLACK.context.users[a].name.localeCompare(SLACK.context.users[b].name);
     });
     sortedUsers.forEach(function(userId) {
-        var user = SLACK.context.getMember(userId);
+        var user = SLACK.context.users[userId];
 
         if (!user.deleted) {
             var ims = user.ims
@@ -95,7 +90,7 @@ function updateTypingChat() {
     if (SELECTED_ROOM && typing[SELECTED_ROOM.id]) {
         var typingUserNames = [], isOutOfSync = false;
         for (var i in typing[SELECTED_ROOM.id]) {
-            var member = SLACK.context.getMember(i);
+            var member = SLACK.context.users[i];
             if (member)
                 typingUserNames.push(member.name);
             else
@@ -117,10 +112,11 @@ function onNetworkStateUpdated(isNetworkWorking) {
 function onRoomSelected() {
     var name = SELECTED_ROOM.name || (SELECTED_ROOM.user ? SELECTED_ROOM.user.name : undefined);
     if (!name) {
+        /** @type {Array.<string>} */
         var members = [];
-        for (var i in SELECTED_ROOM.members) {
-            members.push(SELECTED_ROOM.members[i].name);
-        }
+        SELECTED_ROOM.users.forEach(function(i) {
+            members.push(i.name);
+        });
         name = members.join(", ");
     }
     var roomLi = document.getElementById(SELECTED_ROOM.id);
@@ -261,7 +257,7 @@ function formatEmojis(inputString) {
  * @param {string} fullText
  * @return {string}
 **/
-function formatSlackText(fullText) {
+function formatText(fullText) {
     var msgContents = fullText.split(/\r?\n/g);
 
     for (var msgContentIndex=0, nbMsgContents = msgContents.length; msgContentIndex < nbMsgContents; msgContentIndex++) {
@@ -277,7 +273,7 @@ function formatSlackText(fullText) {
 
                 if (type === '@') {
                     if (!sub[1]) {
-                        var user = SLACK.context.getMember(sub[0]);
+                        var user = SLACK.context.users[sub[0]];
                         sub[1] = user ? ('@' +user.name) : locale.unknownMember;
                     } else if ('@' !== sub[1][0]) {
                         sub[1] = '@' +sub[1];
@@ -286,7 +282,7 @@ function formatSlackText(fullText) {
                     sub[2] = R.klass.msg.link +' ' +R.klass.msg.linkuser;
                 } else if (type === '#') {
                     if (!sub[1]) {
-                        var chan = SLACK.context.getChannel(sub[0]);
+                        var chan = SLACK.context.channels[sub[0]];
                         sub[1] = chan ? ('#' +chan.name) : locale.unknownChannel;
                     } else if ('#' !== sub[1][0]) {
                         sub[1] = '#' +sub[1];
@@ -401,7 +397,7 @@ function formatSlackText(fullText) {
 
 /**
  * @param {string} channelId
- * @param {SlackMessage} msg
+ * @param {Message} msg
  * @param {boolean=} skipAttachment
  * @return {Element}
 **/
@@ -413,7 +409,7 @@ function doCreateMeMessageDom(channelId, msg, skipAttachment) {
 
 /**
  * @param {string} channelId
- * @param {SlackMessage} msg
+ * @param {Message} msg
  * @param {boolean=} skipAttachment
  * @return {Element}
 **/
@@ -456,10 +452,11 @@ function updateTitle() {
         setFavicon(hasHl, hasHl);
     } else {
         var hasUnread = 0;
-        SLACK.context.forEachChans(function(i) {
+        for (var chanId in SLACK.context.channels) {
+            var i = SLACK.context.channels[chanId];
             if (i.lastMsg > i.lastRead)
                 hasUnread++;
-        });
+        }
         if (hasUnread)
             title = "(" +hasUnread +") - ";
         setFavicon(0, hasUnread);
@@ -591,13 +588,15 @@ function focusInput() {
 
 function setRoomFromHashBang() {
     var hashId = document.location.hash.substr(1)
-        ,room = SLACK.context.getChannel(hashId)
-        ,user = SLACK.context.getMember(hashId);
+        ,room = SLACK.context.channels[hashId];
 
     if (room && room !== SELECTED_ROOM)
         selectRoom(room);
-    else if (user && user.ims)
-        selectRoom(user.ims);
+    else {
+        var user = SLACK.context.users[hashId];
+        if (user && user.ims)
+            selectRoom(user.ims);
+    }
 }
 
 document.addEventListener('DOMContentLoaded', function() {
@@ -678,7 +677,7 @@ document.addEventListener('DOMContentLoaded', function() {
     document.getElementById(R.id.message.input).addEventListener('input', function() {
         if (SELECTED_ROOM) {
             var now = Date.now();
-            if (lastKeyDown + 3000 < now && (SLACK.context.self.presence || !(SELECTED_ROOM instanceof SlackChan))) {
+            if (lastKeyDown + 3000 < now && (SLACK.context.self.presence || (SELECTED_ROOM instanceof PrivateMessageRoom))) {
                 sendTyping(SELECTED_ROOM);
                 lastKeyDown = now;
             }

+ 17 - 17
cli/workflow.js

@@ -6,19 +6,19 @@ var
     NEXT_RETRY = 0
 
     /**
-     * @type {SlackChan|SlackIms|SlackGroup}
+     * @type {Room}
     **/
     ,SELECTED_ROOM = null
 
-    /** @type {SlackMessage|null} */
+    /** @type {Message|null} */
     ,REPLYING_TO = null
 
-    /** @type {SlackMessage|null} */
+    /** @type {Message|null} */
     ,EDITING = null
 ;
 
 /**
- * @param {SlackChan|SlackIms|SlackGroup} room
+ * @param {Room} room
  * @param {function(boolean)} cb
 **/
 function fetchHistory(room, cb) {
@@ -75,7 +75,7 @@ function outOfSync() {
 }
 
 /**
- * @param {SlackChan|SlackIms|SlackGroup} room
+ * @param {Room} room
 **/
 function sendTyping(room) {
     var xhr = new XMLHttpRequest()
@@ -104,7 +104,7 @@ function startPolling() {
 }
 
 /**
- * @param {SlackChan|SlackIms|SlackGroup} room
+ * @param {Room} room
 **/
 function selectRoom(room) {
     if (SELECTED_ROOM)
@@ -122,7 +122,7 @@ function unselectRoom() {
 }
 
 /**
- * @param {SlackGroup|SlackChan|SlackIms} chan
+ * @param {Room} chan
  * @param {string} filename
  * @param {File} file
  * @param {function(string?)} callback
@@ -148,7 +148,7 @@ function uploadFile(chan, filename, file, callback) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} chan
+ * @param {Room} chan
  * @param {SlackCommand!} cmd
  * @param {string} args
 **/
@@ -160,15 +160,15 @@ function doCommand(chan, cmd, args) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} chan
+ * @param {Room} chan
  * @param {string} msg
- * @param {SlackMessage|null=} replyTo
+ * @param {Message|null=} replyTo
 **/
 function sendMsg(chan, msg, replyTo) {
     var xhr = new XMLHttpRequest();
     var url = 'api/msg?room=' +chan.id +"&text=" +encodeURIComponent(msg);
     if (replyTo) {
-        var sender = SLACK.context.getMember(replyTo.userId)
+        var sender = SLACK.context.users[replyTo.userId]
             ,footer = "Message";
 
         if (chan.id[0] === 'C') {
@@ -181,7 +181,7 @@ function sendMsg(chan, msg, replyTo) {
         var attachment = {
             "fallback": replyTo.text
             ,"author_name": "<@" +sender.id +"|" +sender.name +">"
-            ,"author_icon": sender.icons.image_48
+            ,"author_icon": sender.icons.small
             ,"text": replyTo.text
             ,"footer": footer
             ,"ts": replyTo.ts
@@ -220,9 +220,9 @@ function onTextEntered(input, skipCommand) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} chan
+ * @param {Room} chan
  * @param {string} text
- * @param {SlackMessage|null=} msg
+ * @param {Message|null=} msg
 **/
 function editMsg(chan, text, msg) {
     var xhr = new XMLHttpRequest();
@@ -232,8 +232,8 @@ function editMsg(chan, text, msg) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} chan
- * @param {SlackMessage|null=} msg
+ * @param {Room} chan
+ * @param {Message|null=} msg
 **/
 function removeMsg(chan, msg) {
     var xhr = new XMLHttpRequest();
@@ -243,7 +243,7 @@ function removeMsg(chan, msg) {
 }
 
 /**
- * @param {SlackChan|SlackGroup|SlackIms} chan
+ * @param {Room} chan
  * @param {number} ts
 **/
 function sendReadMArker(chan, ts) {

+ 62 - 75
srv/public/slack.min.js

@@ -1,75 +1,62 @@
-function aa(a){this.id=a;this.a={X:"",Z:"",aa:"",ba:"",R:"",S:"",U:"",ca:""};this.version=0}function m(a){this.id=a;this.b=this.a=0;this.c={};this.version=0}function q(a){this.id=a;this.c={};this.version=this.b=this.a=this.h=0}function x(a,b){this.id=a;this.g=b;this.version=this.b=this.a=0}function ba(a){this.id=a;this.a={V:"",W:"",s:"",F:"",T:"",$:""};this.g={};this.c=this.b=null;this.version=0}function ca(){this.b={};this.a=[]}
-function da(a,b,c){a.b=JSON.parse(b.emoji_use);b.highlight_words?a.a=(b.highlight_words||"").split(",").filter(function(a){return""!==a.trim()}):b.highlights&&(a.a=b.highlights);a.version=Math.max(a.version,c)}function ea(a){this.id=a;this.a={Y:"",s:"",F:""};this.c=this.b=null;this.version=0;this.f=!1}function fa(a,b){this.a=b.desc;this.name=b.name;this.type=b.type;this.usage=b.usage;this.B=ga(a,b)}
-function ga(a,b){if(b.service_name)return b.service_name;if(b.app){var c=b.app,d=[],g;for(g in a.f)a.f[g].l===c&&d.push(a.f[g]);if(d)for(c=0;c<d.length;c++)if(d[c].name)return d[c].name;console.log("Unknown app "+b.app);return""}return"Slack"}function y(){this.A=null;this.l={};this.w={};this.C={};this.b={};this.a=null;this.f={};this.h={version:0,data:{}};this.g={version:0,data:{}};this.c={};this.J=0}
-function ha(a,b){var c=Date.now();if(b.bots)for(var d=0,g=b.bots.length;d<g;d++){var e=a.f[b.bots[d].id];e||(e=a.f[b.bots[d].id]=new ea(b.bots[d].id));var f=b.bots[d],h=c;void 0!==f.deleted&&(e.h=f.deleted);void 0!==f.name&&(e.name=f.name);void 0!==f.app_id&&(e.l=f.app_id);f.icons&&(e.a.Y=f.icons.image_36,e.a.s=f.icons.image_48,e.a.F=f.icons.image_72);void 0!==f.presence&&(e.f="away"!==f.presence);void 0!==f.isPresent&&(e.f=f.isPresent);e.version=Math.max(e.version,h)}if(b.users)for(d=0,g=b.users.length;d<
-g;d++)(e=a.b[b.users[d].id])||(e=a.b[b.users[d].id]=new ba(b.users[d].id)),f=b.users[d],h=c,void 0!==f.name&&(e.name=f.name),void 0!==f.deleted&&(e.h=f.deleted),void 0!==f.status&&(e.status=f.status),void 0!==f.presence&&(e.f="away"!==f.presence),void 0!==f.isPresent&&(e.f=f.isPresent),f.profile&&(e.a.V=f.profile.image_24,e.a.W=f.profile.image_32,e.a.s=f.profile.image_48,e.a.F=f.profile.image_72,e.a.T=f.profile.image_192,e.a.$=f.profile.image_512),e.version=Math.max(e.version,h);if(b.ims)for(d=0,
-g=b.ims.length;d<g;d++)if(f=A(a,b.ims[d].user)){f.b||(a.C[b.ims[d].id]=f.b=new x(b.ims[d].id,f));var e=f.b,h=b.ims[d],l=c;void 0!==h.last_read&&(e.a=Math.max(parseFloat(h.last_read),e.a));void 0!==h.last_msg&&(e.b=parseFloat(h.last_msg));h.latest&&(e.b=parseFloat(h.latest.ts));e.f=f.h;e.version=Math.max(e.version,l)}if(b.channels)for(d=0,g=b.channels.length;d<g;d++){(e=a.l[b.channels[d].id])||(e=a.l[b.channels[d].id]=new m(b.channels[d].id));f=b.channels[d];h=a;l=c;void 0!==f.name&&(e.name=f.name);
-void 0!==f.is_archived&&(e.f=f.is_archived);void 0!==f.last_read&&(e.a=Math.max(parseFloat(f.last_read),e.a));void 0!==f.last_msg&&(e.b=parseFloat(f.last_msg));f.latest&&(e.b=parseFloat(f.latest.ts));if(f.members&&(e.c={},f.members))for(var n=0,k=f.members.length;n<k;n++){var p=A(h,f.members[n]);e.c[p.id]=p;p.g[e.id]=e}e.version=Math.max(e.version,l)}d=0;for(g=b.groups.length;d<g;d++){(e=a.w[b.groups[d].id])||(e=a.w[b.groups[d].id]=new q(b.groups[d].id));f=a;h=b.groups[d];l=c;n=[];if(h.members){e.c=
-{};k=e.h=0;for(p=h.members.length;k<p;k++){var r=A(f,h.members[k]);e.c[h.members[k]]=r;r.g[e.id]=e;n.push(r.name);e.h++}e.name=n.join(", ")}void 0!==h.is_archived&&(e.f=h.is_archived||!1===h.is_open);void 0!==h.last_read&&(e.a=Math.max(parseFloat(h.last_read),e.a));void 0!==h.last_msg?e.b=parseFloat(h.last_msg):h.latest&&(e.b=parseFloat(h.latest.ts));e.version=Math.max(e.version,l)}b.emojis&&(a.h.data=b.emojis,a.h.version=c);if(void 0!==b.commands){a.g.data={};for(d in b.commands)a.g.data[d]=new fa(a,
-b.commands[d]);a.g.version=c}b.team&&(a.A||(a.A=new aa(b.team.id)),g=a.A,e=b.team,void 0!==e.name&&(g.name=e.name),e.icon&&(g.a.X=e.icon.image_34,g.a.Z=e.icon.image_44,g.a.aa=e.icon.image_68,g.a.ba=e.icon.image_88,g.a.R=e.icon.image_102,g.a.S=e.icon.image_132,g.a.U=e.icon.image_230,g.a.ca=e.icon.image_default),g.version=Math.max(g.version,c));a.J=Math.max(a.J,c);b.self&&(a.a=A(a,b.self.id),a.a.c||(a.a.c=new ca),da(a.a.c,b.self.prefs,c));if(void 0!==b.typing)for(d in a.c={},b.typing){a.c[d]={};for(var u in b.typing[d])a.c[d][u]=
-c}}function A(a,b){return a.b[b]||a.f[b]||null}function C(a,b){return a.l[b]||a.C[b]||a.w[b]||null}function D(a,b){for(var c in a.l)if(!1===b(a.l[c]))return;for(c in a.C)if(!1===!b(a.C[c]))return;for(c in a.w)if(!1===!b(a.w[c]))break}"undefined"!==typeof module&&(module.P.da=y);function ia(a,b){this.m=a.user||a.bot_id;this.g=a.username;this.id=a.ts;this.i=parseFloat(a.ts);this.text=a.text||"";this.c=a.attachments||[];this.o=!!a.edited;this.b=a.removed||!1;this.h="me_message"===a.subtype||a.isMeMessage;this.a={};this.version=b;this.f=!0===a.is_ephemeral||!0===a.isNotice;var c=this;a.reactions&&a.reactions.forEach(function(a){c.a[a.name]=[];a.users.forEach(function(b){c.a[a.name].push(b)})})}
-function ja(a,b,c){b?(a.text=b.text||"",b.attachments&&(a.c=b.attachments),a.o=!!b.edited,a.b=!!b.removed,b.reactions&&(a.a={},b.reactions.forEach(function(b){a.a[b.name]=[];b.users.forEach(function(c){a.a[b.name].push(c)})}))):a.b=!0;a.version=c}function F(a,b,c,d){this.id="string"===typeof a?a:a.id;this.a=[];this.b=b;c&&ka(this,c,d)}
-function la(a,b,c,d){var g=!1;a.a[b]&&(1===a.a[b].length&&a.a[b][0]===c?(delete a.a[b],g=!0):a.a[b]=a.a[b].filter(function(a){return a!==c?!1:g=!0}));g&&(a.version=d)}function ka(a,b,c){var d=0;b.forEach(function(a){d=Math.max(this.push(a,c),d)}.bind(a));ma(a)}
-F.prototype.push=function(a,b){var c;if(a.type&&"message"!==a.type)if("reaction_added"===a.type){if(c=G(this,a.item.ts)){var d=a.reaction,g=a.user;c.a[d]||(c.a[d]=[]);c.a[d].push(g);c.version=b}}else if("reaction_removed"===a.type)(c=G(this,a.item.ts))&&la(c,a.reaction,a.user,b);else return 0;else{var d=!1,g=a.ts,e=a;"message_changed"===a.subtype&&a.previous_message?(g=a.previous_message.ts,e=a.message):"message_deleted"===a.subtype&&a.previous_message&&(g=a.previous_message.ts,e=null);for(var f=
-0,h=this.a.length;f<h;f++)if(this.a[f].id===g){ja(this.a[f],e,b);d=!0;break}d||(c=new ia(a,b),this.a.push(c))}for(;this.a.length>this.b;)this.a.shift();return c?c.i:0};function na(a){return a.a[a.a.length-1]}function oa(a){for(var b=H.b[I.id],c=0,d=b.a.length;c<d&&a>=b.a[c].i;c++)if(b.a[c].i===a)return b.a[c];return null}function G(a,b){for(var c=0,d=a.a.length;c<d;c++)if(a.a[c].id==b)return a.a[c];return null}function ma(a){a.a.sort(function(a,c){return a.i-c.i})}
-"undefined"!==typeof module&&(module.P.ea=F);var J={},K;function pa(){var a;if(!a){for(var b=0,c=navigator.languages.length;b<c;b++)if(J.hasOwnProperty(navigator.languages[b])){a=navigator.languages[b];break}a||(a="en")}K=J[a];console.log("Loading language pack: "+a);if(K.j)for(b in K.j)document.getElementById(b).textContent=K.j[b]};J.fr={O:"Utilisateur inconnu",N:"Channel inconnu",I:"Nouveau message",H:"Reseau",o:"edit&eacute;",K:"(visible seulement par vous)",G:function(a){return 1===a.length?a[0]+" est en train d'\u00e9crire":a.join(", ")+" sont en train d'\u00e9crire"},D:function(a){"string"!==typeof a&&(a=parseFloat(a));var b=new Date,c=new Date;a=new Date(1E3*a);b.setHours(0);b.setMinutes(0);b.setSeconds(0);b.setMilliseconds(0);c.setTime(b.getTime());c.setDate(c.getDate()-1);return a.getTime()>b.getTime()?a.toLocaleTimeString():
-a.getTime()>c.getTime()?"hier, "+a.toLocaleTimeString():a.toLocaleString()},j:{fileUploadCancel:"Annuler",neterror:"Impossible de se connecter au chat !"}};J.en={O:"Unknown member",N:"Unknown channel",I:"New message",H:"Network",o:"edited",K:"(only visible to you)",G:function(a){return 1===a.length?a[0]+" is typing":a.join(", ")+" are typing"},D:function(a){"string"!==typeof a&&(a=parseFloat(a));var b=new Date,c=new Date;a=new Date(1E3*a);b.setHours(0);b.setMinutes(0);b.setSeconds(0);b.setMilliseconds(0);c.setTime(b.getTime());c.setDate(c.getDate()-1);return a.getTime()>b.getTime()?a.toLocaleTimeString():a.getTime()>c.getTime()?"yesterday, "+a.toLocaleTimeString():
-a.toLocaleString()},j:{fileUploadCancel:"Cancel",neterror:"Cannot connect to chat !"}};var L=0;
-function qa(){var a=document.createDocumentFragment(),b=H.a.a?Object.keys(H.a.a.g):[];b.sort(function(a,b){return a[0]!==b[0]?a[0]-b[0]:C(H.a,a).name.localeCompare(C(H.a,b).name)});b.forEach(function(b){b=C(H.a,b);if(!b.f){var c=document.createElement("li"),g=document.createElement("a");c.id=b.id;g.href="#"+b.id;b instanceof q?(c.className="slack-context-room slack-group",c.dataset.count=b.h-1):b instanceof m&&(c.className="slack-context-room slack-channel");I===b&&c.classList.add("selected");g.textContent=
-b.name;c.appendChild(ra());c.appendChild(g);b.b>b.a&&(c.classList.add("unread"),0<=M.indexOf(b)&&c.classList.add("unreadHi"));c&&a.appendChild(c)}});b=H.a.b?Object.keys(H.a.b):[];b.sort(function(a,b){return H.a.b[a].name.localeCompare(H.a.b[b].name)});b.forEach(function(b){b=A(H.a,b);if(!b.h){b=b.b;var c=document.createElement("li"),g=document.createElement("a");c.id=b.id;g.href="#"+b.id;c.className="slack-context-room slack-ims";g.textContent=b.g.name;c.appendChild(ra());c.appendChild(g);b.g.f||
-c.classList.add("away");I===b&&c.classList.add("selected");b.b>b.a&&(c.classList.add("unread"),0<=M.indexOf(b)&&c.classList.add("unreadHi"));c&&a.appendChild(c)}});document.getElementById("chanList").textContent="";document.getElementById("chanList").appendChild(a);sa();N();ta(function(a){document.getElementById("slackCtx").style.backgroundImage="url("+a+")"})}
-function ua(){var a=H.a.c,b;for(b in H.a.a.g)if(!H.a.a.g[b].f){var c=document.getElementById(b);a[b]?c.classList.add("slack-context-typing"):c.classList.remove("slack-context-typing")}for(var d in H.a.b)(b=H.a.b[d].b)&&!b.f&&(c=document.getElementById(b.id),a[b.id]?c.classList.add("slack-context-typing"):c.classList.remove("slack-context-typing"));va()}
-function va(){var a=H.a.c;if(I&&a[I.id]){var b=[],c=!1,d;for(d in a[I.id])(a=A(H.a,d))?b.push(a.name):c=!0;c&&(H.c=0);document.getElementById("whoistyping").textContent=K.G(b)}else document.getElementById("whoistyping").textContent=""}function O(a){a?document.body.classList.remove("no-network"):document.body.classList.add("no-network");N()}
-function P(){if(Q){document.body.classList.add("replyingTo");var a=document.getElementById("replyToContainer"),b=document.createElement("a");b.addEventListener("click",function(){Q=null;P()});b.className="replyto-close";b.textContent="x";a.textContent="";a.appendChild(b);a.appendChild(R("reply_"+I.id,Q,!0))}else document.body.classList.remove("replyingTo"),document.getElementById("replyToContainer").textContent="";S()}
-function T(){if(U){document.body.classList.add("replyingTo");var a=document.getElementById("replyToContainer"),b=document.createElement("a");b.addEventListener("click",function(){U=null;T()});b.className="replyto-close";b.textContent="x";a.textContent="";a.appendChild(b);a.appendChild(R("edit_"+I.id,U,!0));document.getElementById("msgInput").value=U.text}else document.body.classList.remove("replyingTo"),document.getElementById("replyToContainer").textContent="";S()}
-window.toggleReaction=function(a,b,c){var d=H.b[a];if(d&&(d=G(d,b))){var g=H.a.a.id;d.a[c]&&-1!==d.a[c].indexOf(g)?(d=new XMLHttpRequest,d.open("DELETE","api/reaction?room="+a+"&msg="+b+"&reaction="+encodeURIComponent(c),!0),d.send(null)):wa(a,b,c)}};
-function xa(a){a:{for(var b=a,c={};!c[b];){if(a=H.a.h[b])if("alias:"==a.substr(0,6))c[b]=!0,b=a.substr(6);else{b=document.createElement("span");b.className="emoji-custom emoji";b.style.backgroundImage="url('"+a+"')";a=b;break a}break}a=b}"string"===typeof a&&"makeEmoji"in window&&(a=window.makeEmoji(a));return"string"===typeof a?null:a}
-function ya(a){return a.replace(/:([^ \t:]+):/g,function(b,c){var d=xa(c);if(d){var g=document.createElement("span");g.className=b===a?"emoji-medium":"emoji-small";g.appendChild(d);return g.outerHTML}return b})}
-function W(a){a=a.split(/\r?\n/g);for(var b=0,c=a.length;b<c;b++){for(var d=a[b].trim(),g="",e={},f=!1,h=0,d=d.replace(RegExp("<([@#]?)([^>]*)>","g"),function(a,b,c){c=c.split("|");if("@"===b)c[1]?"@"!==c[1][0]&&(c[1]="@"+c[1]):(a=A(H.a,c[0]),c[1]=a?"@"+a.name:K.O),c[0]="#"+c[0],c[2]="slackmsg-link slackmsg-link-user";else if("#"===b)c[1]?"#"!==c[1][0]&&(c[1]="#"+c[1]):(a=C(H.a,c[0]),c[1]=a?"#"+a.name:K.N),c[0]="#"+c[0],c[2]="slackmsg-link slackmsg-link-chan";else if(-1!==c[0].indexOf("://"))c[1]||
-(c[1]=c[0]),c[2]="slackmsg-link";else return a;return'<a href="'+c[0]+'" class="'+c[2]+'"'+(b?"":' target="_blank"')+">"+c[1]+"</a>"}),d=ya(d),l=d.length,n=function(a,b,c){for(;a[b];){var d=a[b];if(("A"<=d&&"Z">=d||"a"<=d&&"z">=d||"0"<=d&&"9">=d||-1!=="\u00e0\u00e8\u00ec\u00f2\u00f9\u00c0\u00c8\u00cc\u00d2\u00d9\u00e1\u00e9\u00ed\u00f3\u00fa\u00fd\u00c1\u00c9\u00cd\u00d3\u00da\u00dd\u00e2\u00ea\u00ee\u00f4\u00fb\u00c2\u00ca\u00ce\u00d4\u00db\u00e3\u00f1\u00f5\u00c3\u00d1\u00d5\u00e4\u00eb\u00ef\u00f6\u00fc\u00ff\u00c4\u00cb\u00cf\u00d6\u00dc\u0178\u00e7\u00c7\u00df\u00d8\u00f8\u00c5\u00e5\u00c6\u00e6\u0153".indexOf(d))&&
-a[b]!=c&&a[b+1]==c)return!0;b++}return!1},k=function(a){return Object.keys(e).length?'<span class="'+Object.keys(a).join(" ")+'">':""};h<l&&(" "===d[h]||"\t"===d[h]);)h++;"&gt;"===d.substr(h,4)&&(f=!0,h+=4);for(;h<l;h++){var p=d[h];if("<"===p){do g+=d[h++];while(">"!==d[h-1]);h--}else if(!e["slackmsg-style-bold"]&&"*"===p&&d[h+1]&&n(d,h,p))Object.keys(e).length&&(g+="</span>"),e["slackmsg-style-bold"]=!0,g+=k(e);else if(!e["slackmsg-style-strike"]&&"~"===p&&d[h+1]&&n(d,h,p))Object.keys(e).length&&
-(g+="</span>"),e["slackmsg-style-strike"]=!0,g+=k(e);else if(!e["slackmsg-style-code"]&&"`"===p&&d[h+1]&&n(d,h,p))Object.keys(e).length&&(g+="</span>"),e["slackmsg-style-code"]=!0,g+=k(e);else if(!e["slackmsg-style-italic"]&&"_"===p&&d[h+1]&&n(d,h,p))Object.keys(e).length&&(g+="</span>"),e["slackmsg-style-italic"]=!0,g+=k(e);else{var r=!1,g=g+p;do{if(e["slackmsg-style-bold"]&&"*"!==p&&"*"===d[h+1])delete e["slackmsg-style-bold"],r=!0;else if(e["slackmsg-style-strike"]&&"~"!==p&&"~"===d[h+1])delete e["slackmsg-style-strike"],
-r=!0;else if(e["slackmsg-style-code"]&&"`"!==p&&"`"===d[h+1])delete e["slackmsg-style-code"],r=!0;else if(e["slackmsg-style-italic"]&&"_"!==p&&"_"===d[h+1])delete e["slackmsg-style-italic"],r=!0;else break;p=d[++h]}while(h<l);r&&(g+="</span>"+k(e))}}e&&(g+="</span>");a[b]=f?'<span class="slackmsg-style-quote">'+g+"</span>":g}return a.join("<br/>")}
-function R(a,b,c){b.h?(a=za(a,b,c),a.classList.add("slackmsg-me_message")):a=za(a,b,c);b.o&&a.classList.add("slackmsg-edited");b.f&&a.classList.add("slackmsg-notice");return a}function Aa(a,b){document.getElementById("linkFavicon").href=a||b?"favicon.png?h="+a+"&m="+b:"favicon_ok.png"}
-function N(){var a=M.length,b="";if(X)b="!"+K.H+" - ",document.getElementById("linkFavicon").href="favicon_err.png";else if(a)b="(!"+a+") - ",Aa(a,a);else{var c=0;D(H.a,function(a){a.b>a.a&&c++});c&&(b="("+c+") - ");Aa(0,c)}H.a.A&&(b+=H.a.A.name);document.title=b}
-function Ba(){if("Notification"in window)if("granted"===Notification.permission){var a=Date.now();if(L+3E4<a){var b=new Notification(K.I);L=a;setTimeout(function(){b.close()},5E3)}}else"denied"!==Notification.permission&&Notification.requestPermission()}
-function Ca(){var a=document.createDocumentFragment(),b=I.id,c=null,d=0,g=null;H.b[b]&&H.b[b].a.forEach(function(e){if(!e.b){var f=R(b,e);c&&c.m===e.m&&e.m?(f.classList.add("slackmsg-same-author"),30>Math.abs(d-e.i)?g.classList.add("slackmsg-same-ts"):d=e.i):d=e.i;(!c||c.i<=I.a)&&e.i>I.a&&f.classList.add("slackmsg-first-unread");c=e;g=f;a.appendChild(f)}});var e=document.getElementById("chatWindow");e.textContent="";e.appendChild(a);e.scrollTop=e.scrollHeight-e.clientHeight;window.hasFocus&&Y()}
-function Da(a){function b(a,b){for(b=b||a.target;b!==a.currentTarget&&b;){if(b.classList.contains("slackmsg-item"))return b.id;b=b.parentElement}}for(var c=a.target;c!==a.currentTarget&&c&&!c.classList.contains("slackmsg-hover");){if(c.parentElement&&c.parentElement.classList.contains("slackmsg-hover")){if(a=b(a,c)){a=parseFloat(a.split("_")[1]);var d=oa(a);d&&c.classList.contains("slackmsg-hover-reply")?(U&&(U=null,T()),Q!==d&&(Q=d,P())):d&&c.classList.contains("slackmsg-hover-reaction")?Ea.M(document.body,
-function(a){a&&wa(I.id,d.id,a)}):d&&c.classList.contains("slackmsg-hover-edit")?(Q&&(Q=null,P()),U!==d&&(U=d,T())):d&&c.classList.contains("slackmsg-hover-remove")&&(Q&&(Q=null,P()),U&&(U=null,T()),Fa(d))}break}c=c.parentElement}}function S(){document.getElementById("msgInput").focus()}function sa(){var a=document.location.hash.substr(1),b=C(H.a,a),a=A(H.a,a);b&&b!==I?Ga(b):a&&a.b&&Ga(a.b)}
-document.addEventListener("DOMContentLoaded",function(){pa();document.getElementById("chatWindow").addEventListener("click",Da);window.addEventListener("hashchange",function(){document.location.hash&&"#"===document.location.hash[0]&&sa()});document.getElementById("fileUploadCancel").addEventListener("click",function(a){a.preventDefault();document.getElementById("fileUploadError").classList.add("hidden");document.getElementById("fileUploadContainer").classList.add("hidden");document.getElementById("fileUploadInput").value=
-"";return!1});document.getElementById("fileUploadForm").addEventListener("submit",function(a){a.preventDefault();a=document.getElementById("fileUploadInput");var b=a.value;b&&(b=b.substr(b.lastIndexOf("\\")+1),Ha(b,a.files[0],function(a){var b=document.getElementById("fileUploadError");a?(b.textContent=a,b.classList.remove("hidden")):(b.classList.add("hidden"),document.getElementById("fileUploadInput").value="",document.getElementById("fileUploadContainer").classList.add("hidden"))}));return!1});
-document.getElementById("attachFile").addEventListener("click",function(a){a.preventDefault();I&&document.getElementById("fileUploadContainer").classList.remove("hidden");return!1});document.getElementById("msgForm").addEventListener("submit",function(a){a.preventDefault();a=document.getElementById("msgInput");I&&a.value&&Ia(a.value)&&(a.value="",Q&&(Q=null,P()),U&&(U=null,P()),document.getElementById("slashList").textContent="");S();return!1});window.addEventListener("blur",function(){window.hasFocus=
-!1});window.addEventListener("focus",function(){window.hasFocus=!0;L=0;I&&Y();S()});var a=0;document.getElementById("msgInput").addEventListener("input",function(){if(I){var b=Date.now();!(a+3E3<b)||!H.a.a.f&&I instanceof m||(Ja(),a=b);var b=[],c=this.value;if("/"===this.value[0]){var d=c.indexOf(" "),g=-1!==d,d=-1===d?c.length:d,c=c.substr(0,d),e;for(e in H.a.g.data){var f=H.a.g.data[e];(!g&&f.name.substr(0,d)===c||g&&f.name===c)&&b.push(f)}}b.sort(function(a,b){return a.B.localeCompare(b.B)||a.name.localeCompare(b.name)});
-var d=document.getElementById("slashList"),g=document.createDocumentFragment(),h;d.textContent="";e=0;for(c=b.length;e<c;e++)f=b[e],h!==f.B&&(h=f.B,g.appendChild(Ka(f.B))),g.appendChild(La(f));d.appendChild(g)}});window.hasFocus=!0;(function(){var a=document.getElementById("emojiButton");if("makeEmoji"in window){var c=window.makeEmoji("smile");c?a.innerHTML="<span class='emoji-small'>"+c.outerHTML+"</span>":a.style.backgroundImage='url("smile.svg")';(c=window.makeEmoji("paperclip"))?document.getElementById("attachFile").innerHTML=
-"<span class='emoji-small'>"+c.outerHTML+"</span>":document.getElementById("attachFile").style.backgroundImage='url("public/paperclip.svg")';a.addEventListener("click",function(){Ea.M(document.body,function(a){a&&(document.getElementById("msgInput").value+=":"+a+":");S()})})}else a.classList.add("hidden")})();Z()});function ra(){var a=document.createElement("span"),b=document.createElement("span"),c=document.createElement("span"),d=document.createElement("span");a.className="typing-container";b.className="typing-dot1";c.className="typing-dot2";d.className="typing-dot3";b.textContent=c.textContent=d.textContent=".";a.appendChild(b);a.appendChild(c);a.appendChild(d);return a}
-function Ma(a,b,c,d){var g=xa(c);if(g){for(var e=document.createElement("li"),f=document.createElement("a"),h=document.createElement("span"),l=document.createElement("span"),n=[],k=0,p=d.length;k<p;k++){var r=A(H.a,d[k]);r&&n.push(r.name)}n.sort();l.textContent=n.join(", ");h.appendChild(g);h.className="emoji-small";f.href="javascript:toggleReaction('"+a+"', '"+b+"', '"+c+"')";f.appendChild(h);f.appendChild(l);e.className="slackmsg-reaction-item";e.appendChild(f);return e}return null}
-function za(a,b,c){var d=document.createElement("div"),g=document.createElement("div"),e=document.createElement("div"),f=document.createElement("div"),h=document.createElement("img"),l=document.createElement("span"),n=document.createElement("ul"),k=document.createElement("li"),p=document.createElement("ul"),r=document.createElement("ul"),u=A(H.a,b.m);d.id=a+"_"+b.i;d.className="slackmsg-item";e.className="slackmsg-ts";f.className="slackmsg-msg";h.className="slackmsg-author-img";l.className="slackmsg-author-name";
-n.className="slackmsg-hover";k.className="slackmsg-hover-reply";e.innerHTML=K.D(b.i);f.innerHTML=W(b.text);l.textContent=u?u.name:b.g||"?";h.src=u?u.a.s:"";n.appendChild(k);if("makeEmoji"in window){var v=document.createElement("li"),z=window.makeEmoji("arrow_heading_down"),E=window.makeEmoji("smile"),t=window.makeEmoji("pencil2"),u=window.makeEmoji("x");v.className="slackmsg-hover-reaction";E?(v.classList.add("emoji-small"),v.appendChild(E)):v.style.backgroundImage='url("smile.svg")';z?(k.classList.add("emoji-small"),
-k.appendChild(z)):k.style.backgroundImage='url("repl.svg")';n.appendChild(v);b.m===H.a.a.id&&(k=document.createElement("li"),k.className="slackmsg-hover-edit",t?k.classList.add("emoji-small"):k.style.backgroundImage='url("edit.svg")',k.appendChild(t),n.appendChild(k),k=document.createElement("li"),k.className="slackmsg-hover-remove",u?k.classList.add("emoji-small"):k.style.backgroundImage='url("remove.svg")',k.appendChild(u),n.appendChild(k))}else k.style.backgroundImage='url("repl.svg")',b.m===H.a.a.id&&
-(k=document.createElement("li"),k.className="slackmsg-hover-edit",k.style.backgroundImage='url("edit.svg")',n.appendChild(k),k=document.createElement("li"),k.className="slackmsg-hover-remove",k.style.backgroundImage='url("remove.svg")',n.appendChild(k));d.appendChild(h);b.f&&(h=document.createElement("span"),h.className="slackmsg-notice",h.textContent=K.K,g.appendChild(h));g.appendChild(l);g.appendChild(f);g.appendChild(e);g.appendChild(p);b.o&&(e=document.createElement("div"),e.textContent=K.o,e.className=
-"slackmsg-edited",g.appendChild(e));g.appendChild(r);g.className="slackmsg-content";p.className="slackmsg-attachments";r.className="slackmsg-reactions";if(!0!==c){if(b.a)for(var B in b.a)(c=Ma(a,b.id,B,b.a[B]))&&r.appendChild(c);b.c.forEach(function(a){var b=document.createElement("li"),c=document.createElement("div"),d=document.createElement("div"),e=document.createElement("a"),g=document.createElement("div"),f=document.createElement("img"),h=document.createElement("a"),k=document.createElement("div"),
-l=document.createElement("div"),t=document.createElement("img"),n=document.createElement("img"),w=document.createElement("div"),B=document.createElement("img"),r=document.createElement("span"),u=document.createElement("span");b.className="slackmsg-attachment";var v="#e3e4e6";a.color&&("#"===a.color[0]?v=a.color[0]:"good"===a.color?v="#2fa44f":"warning"===a.color?v="#de9e31":"danger"===a.color&&(v="#d50200"));c.style.borderColor=v;c.className="slackmsg-attachment-block";d.className="slackmsg-attachment-pretext";
-a.pretext?d.innerHTML=W(a.pretext):d.classList.add("hidden");e.target="_blank";a.title?(e.innerHTML=W(a.title),a.title_link&&(e.href=a.title_link),e.className="slackmsg-attachment-title"):e.className="hidden slackmsg-attachment-title";h.target="_blank";g.className="slackmsg-author";a.author_name?(h.innerHTML=W(a.author_name),h.href=a.author_link||"",h.className="slackmsg-author-name",f.className="slackmsg-author-img",a.author_icon?f.src=a.author_icon:f.classList.add("hidden")):g.classList.add("hidden");
-l.innerHTML=W(a.text||"");l.a="slackmsg-attachment-text";t.className="slackmsg-attachment-thumb";a.thumb_url?t.src=a.thumb_url:t.classList.add("hidden");n.className="slackmsg-attachment-img";a.image_url?n.src=a.image_url:n.classList.add("hidden");w.className="slackmsg-attachment-footer";r.className="slackmsg-attachment-footer-text";B.className="slackmsg-attachment-footer-icon";a.footer?(r.innerHTML=W(a.footer),a.footer_icon?B.src=a.footer_icon:B.classList.add("hidden")):(B.classList.add("hidden"),
-r.classList.add("hidden"));u.className="slackmsg-ts";a.ts?u.innerHTML=K.D(a.ts):u.classList.add("hidden");g.appendChild(f);g.appendChild(h);k.appendChild(l);k.appendChild(t);w.appendChild(B);w.appendChild(r);w.appendChild(u);c.appendChild(e);c.appendChild(g);c.appendChild(k);c.appendChild(n);c.appendChild(w);b.appendChild(d);b.appendChild(c);b&&p.appendChild(b)})}d.appendChild(g);d.appendChild(n);return d}
-function Ka(a){var b=document.createElement("lh");b.textContent=a;b.className="slack-command-header";return b}
-function La(a){var b=document.createElement("li"),c=document.createElement("span"),d=document.createElement("span"),g=document.createElement("span");c.textContent=a.name;d.textContent=a.usage;g.textContent=a.a;b.appendChild(c);b.appendChild(d);b.appendChild(g);b.className="slack-command-item";c.className="slack-command-name";d.className="slack-command-usage";g.className="slack-command-desc";return b};var Ea=function(){function a(a,b){for(var c=a.target;c!==l&&c&&"LI"!==c.nodeName;)c=c.parentElement;c&&"LI"===c.nodeName&&c.id&&"emojibar-"===c.id.substr(0,9)?b(c.id.substr(9)):b(null)}function b(){if(!c())return!1;w&&w(null);return!0}function c(){return l.parentElement?(l.parentElement.removeChild(n),l.parentElement.removeChild(l),!0):!1}function d(a){var b=0,c;a=void 0===a?u.value:a;if(h()){var d=window.searchEmojis(a);c=g(d);for(var f in v)v[f].visible&&(v[f].visible=!1,p.removeChild(v[f].j));
-f=0;for(var k=c.length;f<k;f++){var l=c[f].name,t=v[l];if(!t){var t=v,V=l,n=l,l=window.makeEmoji(d[l]),w=document.createElement("span");w.appendChild(l);w.className="emoji-medium";l=e(n,w);t=t[V]=l}t.visible||(t.visible=!0,p.appendChild(t.j));b++}}for(f in z)z[f].visible&&(z[f].visible=!1,r.removeChild(z[f].j));c=g(H.a.h.data);f=0;for(k=c.length;f<k;f++)l=c[f].name,""!==a&&l.substr(0,a.length)!==a||"alias:"===H.a.h.data[l].substr(0,6)||(t=z[l],t||(d=z,V=t=l,l=H.a.h.data[l],n=document.createElement("span"),
-w=document.createElement("span"),n.className="emoji emoji-custom",n.style.backgroundImage='url("'+l+'")',w.appendChild(n),w.className="emoji-medium",l=e(V,w),t=d[t]=l),t.visible||(t.visible=!0,r.appendChild(t.j)),b++);return b}function g(a){var b=H.a.a.c.b,c=[],d;for(d in a){var e={name:d,L:0,count:0};a[d].names.forEach(function(a){e.count+=b[a]||0});c.push(e)}return c=c.sort(function(a,b){var c=b.count-a.count;return c?c:a.L-b.L})}function e(a,b){var c=document.createElement("li");c.appendChild(b);
-c.className="emojibar-list-item";c.id="emojibar-"+a;return{visible:!1,j:c}}function f(a){var b=document.createElement("img"),c=document.createElement("div");b.src=a;c.appendChild(b);c.className="emojibar-header";return c}function h(){return"searchEmojis"in window}var l=document.createElement("div"),n=document.createElement("div"),k=document.createElement("div"),p=document.createElement("ul"),r=document.createElement("ul"),u=document.createElement("input"),v={},z={},E=document.createElement("div"),
-t=document.createElement("span"),B=document.createElement("span"),w;n.addEventListener("click",function(a){var c=l.getBoundingClientRect();(a.screenY<c.top||a.screenY>c.bottom||a.screenX<c.left||a.screenX>c.right)&&b()});n.className="emojibar-overlay";l.className="emojibar";k.className="emojibar-emojis";E.className="emojibar-detail";t.className="emojibar-detail-img";B.className="emojibar-detail-name";p.className=r.className="emojibar-list";u.className="emojibar-search";E.appendChild(t);E.appendChild(B);
-k.appendChild(f(window.emojiProviderHeader));k.appendChild(p);k.appendChild(f("emojicustom.png"));k.appendChild(r);l.appendChild(k);l.appendChild(E);l.appendChild(u);u.addEventListener("keyup",function(){d()});l.addEventListener("mousemove",function(b){a(b,function(a){var b=a?v[a]||z[a]:null;b?(t.innerHTML=b.j.outerHTML,B.textContent=":"+a+":"):(t.textContent="",B.textContent="")})});l.addEventListener("click",function(b){a(b,function(a){a&&c()&&w&&w(a)})});return{isSupported:h,M:function(a,b){return h()?
-(w=b,a.appendChild(n),a.appendChild(l),u.value="",d(),u.focus(),!0):!1},search:d,close:b}}();var H,M=[];function Na(a){var b=H,c=Date.now();a.v&&(b.c=a.v);a["static"]&&ha(b.a,a["static"]);D(b.a,function(a){a.b===a.a&&(a=M.indexOf(a),-1!==a&&M.splice(a,1))});if(a.live){for(var d in a.live){var g=b.b[d];g?ka(g,a.live[d],c):g=b.b[d]=new F(d,250,a.live[d],c)}for(var e in a.live)(c=C(b.a,e))?(b.b[e].a.length&&(c.b=Math.max(c.b,na(b.b[e]).i)),c.f||(Oa(c,a.live[e]),I&&a.live[I.id]&&Ca())):H.c=0}a["static"]&&(qa(),a["static"].typing&&ua())}
-setInterval(function(){var a=H.a,b=Date.now(),c=!1,d;for(d in a.c){var g=!0,e;for(e in a.c[d])a.c[d][e]+3E3<b?(delete a.c[d][e],c=!0):g=!1;g&&(delete a.c[d],c=!0)}c&&ua()},1E3);
-function Oa(a,b){if(a!==I||!window.hasFocus){var c=new RegExp("<@"+H.a.a.id),d=!1,g=!1,e=!1;b.forEach(function(b){if(!(parseFloat(b.ts)<=a.a)){g=!0;var f;if(!(f=a instanceof x||b.text.match(c)))a:{f=H.a.a.c.a;for(var h=0,k=f.length;h<k;h++)if(-1!==b.text.indexOf(f[h])){f=!0;break a}f=!1}f&&(-1===M.indexOf(a)&&(e=!0,M.push(a)),d=!0)}});if(g){N();var f=document.getElementById(a.id);f&&(f.classList.add("unread"),d&&f.classList.add("unreadHi"));e&&!window.hasFocus&&Ba()}}}
-function Y(){var a=I,b=M.indexOf(a);if(a.b>a.a){var c=new XMLHttpRequest;c.open("POST","api/markread?room="+a.id+"&ts="+a.b,!0);c.send(null);a.a=a.b}0<=b&&(M.splice(b,1),N());a=document.getElementById(a.id);a.classList.remove("unread");a.classList.remove("unreadHi")}H=new function(){this.c=0;this.a=new y;this.b={}};var ta=function(){function a(a,b){b.sort(function(){return Math.random()-.5});for(var c=0,d=20;d<l-40;d+=k)for(var f=0;f+k<=n;f+=k)e(a,b[c],d,f),c++,c===b.length&&(b.sort(function(a,b){return a.u?b.u?Math.random()-.5:-1:1}),c=0)}function b(a,d){for(var e=0,f=a.length;e<f;e++)if(void 0===a[e].u){c(a[e].src,function(c){a[e].u=c;b(a,d)});return}var g=[];a.forEach(function(a){a.u&&g.push(a.u)});d(g)}function c(a,b){var c=new XMLHttpRequest;c.responseType="blob";c.onreadystatechange=function(){if(4===
-c.readyState)if(c.response){var a=new Image;a.onload=function(){var c=document.createElement("canvas");c.height=c.width=u;c=c.getContext("2d");c.drawImage(a,0,0,u,u);for(var c=c.getImageData(0,0,u,u),d=0,e=0;e<c.width*c.height*4;e+=4)c.data[e]=c.data[e+1]=c.data[e+2]=(c.data[e]+c.data[e+1]+c.data[e+2])/3,c.data[e+3]=50,d+=c.data[e];if(50>d/(c.height*c.width))for(e=0;e<c.width*c.height*4;e+=4)c.data[e]=c.data[e+1]=c.data[e+2]=255-c.data[e];b(c)};a.onerror=function(){b(null)};a.src=window.URL.createObjectURL(c.response)}else b(null)};
-c.open("GET",a,!0);c.send(null)}function d(){var a=h.createLinearGradient(0,0,0,n);a.addColorStop(0,"#4D394B");a.addColorStop(1,"#201820");h.fillStyle=a;h.fillRect(0,0,l,n);return h.getImageData(0,0,l,n)}function g(a,b){for(var c=(a.height-b.height)/2,d=0;d<b.height;d++)for(var e=0;e<b.width;e++){var f=b.data[4*(d*b.width+e)]/255,g=4*((d+c)*a.width+e+c);a.data[g]*=f;a.data[g+1]*=f;a.data[g+2]*=f}return a}function e(a,b,c,d){var e=Math.floor(d);a=[a.data[e*l*4+0],a.data[e*l*4+1],a.data[e*l*4+2]];h.fillStyle=
-"#"+(1.1*a[0]<<16|1.1*a[1]<<8|1.1*a[2]).toString(16);h.beginPath();h.moveTo(c+k/2,d+p);h.lineTo(c-p+k,d+k/2);h.lineTo(c+k/2,d-p+k);h.lineTo(c+p,d+k/2);h.closePath();h.fill();h.putImageData(g(h.getImageData(c+p,d+p,r,r),b),c+p,d+p)}var f=document.createElement("canvas"),h=f.getContext("2d"),l=f.width=250,n=f.height=290,k=(l-40)/3,p=.1*k,r=Math.floor(k-2*p),u=.5*r,v,z=[],E=!1;return function(c){if(v)c(v);else if(E)z.push(c);else{var e=d(),g=[];E=!0;z.push(c);for(var h in H.a.b)H.a.b[h].h||g.push({src:"api/avatar?user="+
-h});b(g,function(b){a(e,b);v=f.toDataURL();z.forEach(function(a){a(v)})})}}}();var X=0,I=null,Q=null,U=null;function Pa(a){var b=new XMLHttpRequest;b.timeout=6E4;b.onreadystatechange=function(){if(4===b.readyState)if(b.status){var c=null,d=2===Math.floor(b.status/100);if(d){X&&(X=0,O(!0));c=b.response;try{c=JSON.parse(c)}catch(g){c=null}}else X?(X+=Math.floor((X||5)/2),X=Math.min(60,X)):(X=5,O(!1));a(d,c)}else X&&(X=0,O(!0)),Pa(a)};b.open("GET","api?v="+H.c,!0);b.send(null)}function Ja(){var a=new XMLHttpRequest;a.open("POST","api/typing?room="+I.id,!0);a.send(null)}
-function Qa(a,b){a?(b&&Na(b),Z()):setTimeout(Z,1E3*X)}function Z(){Pa(Qa)}
-function Ga(a){I&&document.getElementById(I.id).classList.remove("selected");document.getElementById(a.id).classList.add("selected");document.body.classList.remove("no-room-selected");I=a;a=I.name||(I.g?I.g.name:void 0);if(!a){a=[];for(var b in I.c)a.push(I.c[b].name);a=a.join(", ")}document.getElementById("currentRoomTitle").textContent=a;Ca();S();document.getElementById("fileUploadContainer").classList.add("hidden");Y();Q&&(Q=null,P());U&&(U=null,P());va();I.a&&!H.b[I.id]&&(b=new XMLHttpRequest,
-b.open("GET","api/hist?room="+I.id,!0),b.send(null))}function Ha(a,b,c){var d=I;new FileReader;var g=new FormData,e=new XMLHttpRequest;g.append("file",b);g.append("filename",a);e.onreadystatechange=function(){4===e.readyState&&(204===e.status?c(null):c(e.statusText))};e.open("POST","api/file?room="+d.id);e.send(g)}
-function Ia(a){if(U){var b=new XMLHttpRequest;b.open("PUT","api/msg?room="+I.id+"&ts="+U.id+"&text="+encodeURIComponent(a),!0);b.send(null);return!0}if("/"===a[0]){var c=a.indexOf(" "),b=-1===c?"":a.substr(c);return(a=H.a.g.data[a.substr(0,-1===c?void 0:c)])?(c=new XMLHttpRequest,c.open("POST","api/cmd?room="+I.id+"&cmd="+encodeURIComponent(a.name.substr(1))+"&args="+encodeURIComponent(b.trim()),!0),c.send(null),!0):!1}var b=I,c=Q,d=new XMLHttpRequest;a="api/msg?room="+b.id+"&text="+encodeURIComponent(a);
-if(c){var g=A(H.a,c.m),e="Message";"C"===b.id[0]?e="Channel message":"D"===b.id[0]?e="Direct message":"G"===b.id[0]&&(e="Group message");a+="&attachments="+encodeURIComponent(JSON.stringify([{fallback:c.text,author_name:"<@"+g.id+"|"+g.name+">",author_icon:g.a.s,text:c.text,footer:e,ts:c.i}]))}d.open("POST",a,!0);d.send(null);return!0}function Fa(a){var b=new XMLHttpRequest;b.open("DELETE","api/msg?room="+I.id+"&ts="+a.id,!0);b.send(null)}
-function wa(a,b,c){var d=new XMLHttpRequest;d.open("POST","api/reaction?room="+a+"&msg="+b+"&reaction="+encodeURIComponent(c),!0);d.send(null)};
+function aa(a){this.id=a;this.version=0}function n(){this.b={};this.a={};this.f={version:0,data:{}};this.B={version:0,data:{}};this.H={}}"undefined"!==typeof module&&(module.o.S=n,module.o.T=aa);function p(a){this.id=a;this.i=this.h=0;this.a=[];this.version=0}function w(a,b){p.call(this,a);this.w=b}w.prototype=Object.create(p);"undefined"!==typeof module&&(module.o.Z=p,module.o.Y=w);function z(a,b){this.l=a.l;this.a=a.userName;this.id=a.id;this.c=a.c;this.version=b;B(this,a,b)}function D(a,b){z.call(this,a,b)}function F(a,b){z.call(this,a,b)}function B(a,b,c){b?(a.text=b.text||"",b.A&&(a.A=b.A),a.m=!!b.m,a.D=!!b.D,b.j&&(a.j={},b.j.forEach(function(b){a.j[b.name]=[];b.users.forEach(function(c){a.j[b.name].push(c)})}))):a.D=!0;a.version=c}function G(a,b,c,d){this.id="string"===typeof a?a:a.id;this.a=[];this.b=b;c&&H(this,c,d)}
+function H(a,b,c){var d=0;b.forEach(function(a){d=Math.max(ba(this,a,c),d)}.bind(a));ca(a)}function ba(a,b,c){for(var d=!1,g,e=0,l=a.a.length;e<l;e++){var f=a.a[e];if(f.id===b){g=B(f,c,void 0);d=!0;break}}d||(f=(void 0)(c),a.a.push(f),g=f.c);for(;a.a.length>a.b;)a.a.shift();return g||0}function da(a){for(var b=I.b[J.id],c=0,d=b.a.length;c<d&&a>=b.a[c].c;c++)if(b.a[c].c===a)return b.a[c];return null}function ca(a){a.a.sort(function(a,c){return a.c-c.c})}D.prototype=Object.create(z);F.prototype=Object.create(z);
+"undefined"!==typeof module&&(module.o={W:z,V:D,X:F,$:G});function ea(a){this.id=a;this.a={small:"",fa:""};this.F={};this.M=null;this.version=0}"undefined"!==typeof module&&(module.o.U=ea);var K={},L;function fa(){var a;if(!a){for(var b=0,c=navigator.languages.length;b<c;b++)if(K.hasOwnProperty(navigator.languages[b])){a=navigator.languages[b];break}a||(a="en")}L=K[a];console.log("Loading language pack: "+a);if(L.g)for(b in L.g)document.getElementById(b).textContent=L.g[b]};K.fr={P:"Utilisateur inconnu",O:"Channel inconnu",J:"Nouveau message",I:"Reseau",m:"edit&eacute;",K:"(visible seulement par vous)",G:function(a){return 1===a.length?a[0]+" est en train d'\u00e9crire":a.join(", ")+" sont en train d'\u00e9crire"},C:function(a){"string"!==typeof a&&(a=parseFloat(a));var b=new Date,c=new Date;a=new Date(1E3*a);b.setHours(0);b.setMinutes(0);b.setSeconds(0);b.setMilliseconds(0);c.setTime(b.getTime());c.setDate(c.getDate()-1);return a.getTime()>b.getTime()?a.toLocaleTimeString():
+a.getTime()>c.getTime()?"hier, "+a.toLocaleTimeString():a.toLocaleString()},g:{fileUploadCancel:"Annuler",neterror:"Impossible de se connecter au chat !"}};K.en={P:"Unknown member",O:"Unknown channel",J:"New message",I:"Network",m:"edited",K:"(only visible to you)",G:function(a){return 1===a.length?a[0]+" is typing":a.join(", ")+" are typing"},C:function(a){"string"!==typeof a&&(a=parseFloat(a));var b=new Date,c=new Date;a=new Date(1E3*a);b.setHours(0);b.setMinutes(0);b.setSeconds(0);b.setMilliseconds(0);c.setTime(b.getTime());c.setDate(c.getDate()-1);return a.getTime()>b.getTime()?a.toLocaleTimeString():a.getTime()>c.getTime()?"yesterday, "+a.toLocaleTimeString():
+a.toLocaleString()},g:{fileUploadCancel:"Cancel",neterror:"Cannot connect to chat !"}};var M=0;
+function ga(){var a=document.createDocumentFragment(),b=[];b.sort(function(a,b){return a[0]!==b[0]?a[0]-b[0]:I.a.b[a].name.localeCompare(I.a.b[b].name)});b.forEach(function(b){b=I.a.b[b];if(!b.b){var c=document.createElement("li"),g=document.createElement("a");c.id=b.id;g.href="#"+b.id;b.f?(c.className="slack-context-room slack-group",c.dataset.count=b.a.length):c.className="slack-context-room slack-channel";J===b&&c.classList.add("selected");g.textContent=b.name;c.appendChild(ha());c.appendChild(g);b.i>
+b.h&&(c.classList.add("unread"),0<=N.indexOf(b)&&c.classList.add("unreadHi"));c&&a.appendChild(c)}});b=I.a.a?Object.keys(I.a.a):[];b.sort(function(a,b){return I.a.a[a].name.localeCompare(I.a.a[b].name)});b.forEach(function(b){if(!I.a.a[b].b){b=document.createElement("li");var c=document.createElement("a");b.id=null.id;c.href="#"+null.id;b.className="slack-context-room slack-ims";c.textContent=null.w.name;b.appendChild(ha());b.appendChild(c);null.w.R||b.classList.add("away");null===J&&b.classList.add("selected");
+null.i>null.h&&(b.classList.add("unread"),0<=N.indexOf(null)&&b.classList.add("unreadHi"));b&&a.appendChild(b)}});document.getElementById("chanList").textContent="";document.getElementById("chanList").appendChild(a);ia();O();ja(function(b){document.getElementById("slackCtx").style.backgroundImage="url("+b+")"})}
+function ka(){var a=I.a.H,b;for(b in null.F)if(!null.F[b].b){var c=document.getElementById(b);a[b]?c.classList.add("slack-context-typing"):c.classList.remove("slack-context-typing")}for(var d in I.a.a);la()}function la(){var a=I.a.H;if(J&&a[J.id]){var b=[],c=!1,d;for(d in a[J.id])(a=I.a.a[d])?b.push(a.name):c=!0;c&&(I.f=0);document.getElementById("whoistyping").textContent=L.G(b)}else document.getElementById("whoistyping").textContent=""}
+function P(a){a?document.body.classList.remove("no-network"):document.body.classList.add("no-network");O()}function ma(){var a=J.name||(J.w?J.w.name:void 0);if(!a){var b=[];J.a.forEach(function(a){b.push(a.name)});a=b.join(", ")}document.getElementById("currentRoomTitle").textContent=a;na();Q();document.getElementById("fileUploadContainer").classList.add("hidden");R();S&&(S=null,T());U&&(U=null,T());la()}
+function T(){if(S){document.body.classList.add("replyingTo");var a=document.getElementById("replyToContainer"),b=document.createElement("a");b.addEventListener("click",function(){S=null;T()});b.className="replyto-close";b.textContent="x";a.textContent="";a.appendChild(b);a.appendChild(V("reply_"+J.id,S,!0))}else document.body.classList.remove("replyingTo"),document.getElementById("replyToContainer").textContent="";Q()}
+function W(){if(U){document.body.classList.add("replyingTo");var a=document.getElementById("replyToContainer"),b=document.createElement("a");b.addEventListener("click",function(){U=null;W()});b.className="replyto-close";b.textContent="x";a.textContent="";a.appendChild(b);a.appendChild(V("edit_"+J.id,U,!0));document.getElementById("msgInput").value=U.text}else document.body.classList.remove("replyingTo"),document.getElementById("replyToContainer").textContent="";Q()}
+window.toggleReaction=function(a,b,c){var d=I.b[a];if(d){a:{for(var g=0,e=d.a.length;g<e;g++)if(d.a[g].id==b){d=d.a[g];break a}d=null}d&&(g=null.id,d.j[c]&&-1!==d.j[c].indexOf(g)?(d=new XMLHttpRequest,d.open("DELETE","api/reaction?room="+a+"&msg="+b+"&reaction="+encodeURIComponent(c),!0),d.send(null)):oa(a,b,c))}};
+function pa(a){a:{for(var b=a,c={};!c[b];){if(a=I.a.f[b])if("alias:"==a.substr(0,6))c[b]=!0,b=a.substr(6);else{b=document.createElement("span");b.className="emoji-custom emoji";b.style.backgroundImage="url('"+a+"')";a=b;break a}break}a=b}"string"===typeof a&&"makeEmoji"in window&&(a=window.makeEmoji(a));return"string"===typeof a?null:a}
+function qa(a){return a.replace(/:([^ \t:]+):/g,function(b,c){var d=pa(c);if(d){var g=document.createElement("span");g.className=b===a?"emoji-medium":"emoji-small";g.appendChild(d);return g.outerHTML}return b})}
+function X(a){a=a.split(/\r?\n/g);for(var b=0,c=a.length;b<c;b++){for(var d=a[b].trim(),g="",e={},l=!1,f=0,d=d.replace(RegExp("<([@#]?)([^>]*)>","g"),function(b,a,c){c=c.split("|");if("@"===a)c[1]?"@"!==c[1][0]&&(c[1]="@"+c[1]):(b=I.a.a[c[0]],c[1]=b?"@"+b.name:L.P),c[0]="#"+c[0],c[2]="slackmsg-link slackmsg-link-user";else if("#"===a)c[1]?"#"!==c[1][0]&&(c[1]="#"+c[1]):(b=I.a.b[c[0]],c[1]=b?"#"+b.name:L.O),c[0]="#"+c[0],c[2]="slackmsg-link slackmsg-link-chan";else if(-1!==c[0].indexOf("://"))c[1]||
+(c[1]=c[0]),c[2]="slackmsg-link";else return b;return'<a href="'+c[0]+'" class="'+c[2]+'"'+(a?"":' target="_blank"')+">"+c[1]+"</a>"}),d=qa(d),k=d.length,m=function(b,a,c){for(;b[a];){var d=b[a];if(("A"<=d&&"Z">=d||"a"<=d&&"z">=d||"0"<=d&&"9">=d||-1!=="\u00e0\u00e8\u00ec\u00f2\u00f9\u00c0\u00c8\u00cc\u00d2\u00d9\u00e1\u00e9\u00ed\u00f3\u00fa\u00fd\u00c1\u00c9\u00cd\u00d3\u00da\u00dd\u00e2\u00ea\u00ee\u00f4\u00fb\u00c2\u00ca\u00ce\u00d4\u00db\u00e3\u00f1\u00f5\u00c3\u00d1\u00d5\u00e4\u00eb\u00ef\u00f6\u00fc\u00ff\u00c4\u00cb\u00cf\u00d6\u00dc\u0178\u00e7\u00c7\u00df\u00d8\u00f8\u00c5\u00e5\u00c6\u00e6\u0153".indexOf(d))&&
+b[a]!=c&&b[a+1]==c)return!0;a++}return!1},h=function(b){return Object.keys(e).length?'<span class="'+Object.keys(b).join(" ")+'">':""};f<k&&(" "===d[f]||"\t"===d[f]);)f++;"&gt;"===d.substr(f,4)&&(l=!0,f+=4);for(;f<k;f++){var q=d[f];if("<"===q){do g+=d[f++];while(">"!==d[f-1]);f--}else if(!e["slackmsg-style-bold"]&&"*"===q&&d[f+1]&&m(d,f,q))Object.keys(e).length&&(g+="</span>"),e["slackmsg-style-bold"]=!0,g+=h(e);else if(!e["slackmsg-style-strike"]&&"~"===q&&d[f+1]&&m(d,f,q))Object.keys(e).length&&
+(g+="</span>"),e["slackmsg-style-strike"]=!0,g+=h(e);else if(!e["slackmsg-style-code"]&&"`"===q&&d[f+1]&&m(d,f,q))Object.keys(e).length&&(g+="</span>"),e["slackmsg-style-code"]=!0,g+=h(e);else if(!e["slackmsg-style-italic"]&&"_"===q&&d[f+1]&&m(d,f,q))Object.keys(e).length&&(g+="</span>"),e["slackmsg-style-italic"]=!0,g+=h(e);else{var r=!1,g=g+q;do{if(e["slackmsg-style-bold"]&&"*"!==q&&"*"===d[f+1])delete e["slackmsg-style-bold"],r=!0;else if(e["slackmsg-style-strike"]&&"~"!==q&&"~"===d[f+1])delete e["slackmsg-style-strike"],
+r=!0;else if(e["slackmsg-style-code"]&&"`"!==q&&"`"===d[f+1])delete e["slackmsg-style-code"],r=!0;else if(e["slackmsg-style-italic"]&&"_"!==q&&"_"===d[f+1])delete e["slackmsg-style-italic"],r=!0;else break;q=d[++f]}while(f<k);r&&(g+="</span>"+h(e))}}e&&(g+="</span>");a[b]=l?'<span class="slackmsg-style-quote">'+g+"</span>":g}return a.join("<br/>")}
+function V(a,b,c){b.f?(a=ra(a,b,c),a.classList.add("slackmsg-me_message")):a=ra(a,b,c);b.m&&a.classList.add("slackmsg-edited");b.b&&a.classList.add("slackmsg-notice");return a}function sa(a,b){document.getElementById("linkFavicon").href=a||b?"favicon.png?h="+a+"&m="+b:"favicon_ok.png"}
+function O(){var a=N.length,b="";if(Y)b="!"+L.I+" - ",document.getElementById("linkFavicon").href="favicon_err.png";else if(a)b="(!"+a+") - ",sa(a,a);else{var a=0,c;for(c in I.a.b){var d=I.a.b[c];d.i>d.h&&a++}a&&(b="("+a+") - ");sa(0,a)}document.title=b}
+function ta(){if("Notification"in window)if("granted"===Notification.permission){var a=Date.now();if(M+3E4<a){var b=new Notification(L.J);M=a;setTimeout(function(){b.close()},5E3)}}else"denied"!==Notification.permission&&Notification.requestPermission()}
+function na(){var a=document.createDocumentFragment(),b=J.id,c=null,d=0,g=null;I.b[b]&&I.b[b].a.forEach(function(e){if(!e.D){var f=V(b,e);c&&c.l===e.l&&e.l?(f.classList.add("slackmsg-same-author"),30>Math.abs(d-e.c)?g.classList.add("slackmsg-same-ts"):d=e.c):d=e.c;(!c||c.c<=J.h)&&e.c>J.h&&f.classList.add("slackmsg-first-unread");c=e;g=f;a.appendChild(f)}});var e=document.getElementById("chatWindow");e.textContent="";e.appendChild(a);e.scrollTop=e.scrollHeight-e.clientHeight;window.hasFocus&&R()}
+function ua(a){function b(b,a){for(a=a||b.target;a!==b.currentTarget&&a;){if(a.classList.contains("slackmsg-item"))return a.id;a=a.parentElement}}for(var c=a.target;c!==a.currentTarget&&c&&!c.classList.contains("slackmsg-hover");){if(c.parentElement&&c.parentElement.classList.contains("slackmsg-hover")){if(a=b(a,c)){a=parseFloat(a.split("_")[1]);var d=da(a);d&&c.classList.contains("slackmsg-hover-reply")?(U&&(U=null,W()),S!==d&&(S=d,T())):d&&c.classList.contains("slackmsg-hover-reaction")?va.N(document.body,
+function(b){b&&oa(J.id,d.id,b)}):d&&c.classList.contains("slackmsg-hover-edit")?(S&&(S=null,T()),U!==d&&(U=d,W())):d&&c.classList.contains("slackmsg-hover-remove")&&(S&&(S=null,T()),U&&(U=null,W()),wa(d))}break}c=c.parentElement}}function Q(){document.getElementById("msgInput").focus()}
+function ia(){var a=I.a.b[document.location.hash.substr(1)];a&&a!==J&&(J&&document.getElementById(J.id).classList.remove("selected"),document.getElementById(a.id).classList.add("selected"),document.body.classList.remove("no-room-selected"),J=a,ma(),J.h&&!I.b[J.id]&&(a=new XMLHttpRequest,a.open("GET","api/hist?room="+J.id,!0),a.send(null)))}
+document.addEventListener("DOMContentLoaded",function(){fa();document.getElementById("chatWindow").addEventListener("click",ua);window.addEventListener("hashchange",function(){document.location.hash&&"#"===document.location.hash[0]&&ia()});document.getElementById("fileUploadCancel").addEventListener("click",function(b){b.preventDefault();document.getElementById("fileUploadError").classList.add("hidden");document.getElementById("fileUploadContainer").classList.add("hidden");document.getElementById("fileUploadInput").value=
+"";return!1});document.getElementById("fileUploadForm").addEventListener("submit",function(b){b.preventDefault();b=document.getElementById("fileUploadInput");var a=b.value;a&&(a=a.substr(a.lastIndexOf("\\")+1),xa(a,b.files[0],function(b){var a=document.getElementById("fileUploadError");b?(a.textContent=b,a.classList.remove("hidden")):(a.classList.add("hidden"),document.getElementById("fileUploadInput").value="",document.getElementById("fileUploadContainer").classList.add("hidden"))}));return!1});
+document.getElementById("attachFile").addEventListener("click",function(b){b.preventDefault();J&&document.getElementById("fileUploadContainer").classList.remove("hidden");return!1});document.getElementById("msgForm").addEventListener("submit",function(b){b.preventDefault();b=document.getElementById("msgInput");J&&b.value&&ya(b.value)&&(b.value="",S&&(S=null,T()),U&&(U=null,T()),document.getElementById("slashList").textContent="");Q();return!1});window.addEventListener("blur",function(){window.hasFocus=
+!1});window.addEventListener("focus",function(){window.hasFocus=!0;M=0;J&&R();Q()});var a=0;document.getElementById("msgInput").addEventListener("input",function(){if(J){var b=Date.now();a+3E3<b&&(null.R||J instanceof w)&&(za(),a=b);var b=[],c=this.value;if("/"===this.value[0]){var d=c.indexOf(" "),g=-1!==d,d=-1===d?c.length:d,c=c.substr(0,d),e;for(e in I.a.B.data){var l=I.a.B.data[e];(!g&&l.name.substr(0,d)===c||g&&l.name===c)&&b.push(l)}}b.sort(function(b,a){return b.u.localeCompare(a.u)||b.name.localeCompare(a.name)});
+var d=document.getElementById("slashList"),g=document.createDocumentFragment(),f;d.textContent="";e=0;for(c=b.length;e<c;e++)l=b[e],f!==l.u&&(f=l.u,g.appendChild(Aa(l.u))),g.appendChild(Ba(l));d.appendChild(g)}});window.hasFocus=!0;(function(){var b=document.getElementById("emojiButton");if("makeEmoji"in window){var a=window.makeEmoji("smile");a?b.innerHTML="<span class='emoji-small'>"+a.outerHTML+"</span>":b.style.backgroundImage='url("smile.svg")';(a=window.makeEmoji("paperclip"))?document.getElementById("attachFile").innerHTML=
+"<span class='emoji-small'>"+a.outerHTML+"</span>":document.getElementById("attachFile").style.backgroundImage='url("public/paperclip.svg")';b.addEventListener("click",function(){va.N(document.body,function(a){a&&(document.getElementById("msgInput").value+=":"+a+":");Q()})})}else b.classList.add("hidden")})();Z()});function ha(){var a=document.createElement("span"),b=document.createElement("span"),c=document.createElement("span"),d=document.createElement("span");a.className="typing-container";b.className="typing-dot1";c.className="typing-dot2";d.className="typing-dot3";b.textContent=c.textContent=d.textContent=".";a.appendChild(b);a.appendChild(c);a.appendChild(d);return a}
+function Ca(a,b,c,d){var g=pa(c);if(g){for(var e=document.createElement("li"),l=document.createElement("a"),f=document.createElement("span"),k=document.createElement("span"),m=[],h=0,q=d.length;h<q;h++){var r=I.a.a[d[h]];r&&m.push(r.name)}m.sort();k.textContent=m.join(", ");f.appendChild(g);f.className="emoji-small";l.href="javascript:toggleReaction('"+a+"', '"+b+"', '"+c+"')";l.appendChild(f);l.appendChild(k);e.className="slackmsg-reaction-item";e.appendChild(l);return e}return null}
+function ra(a,b,c){var d=document.createElement("div"),g=document.createElement("div"),e=document.createElement("div"),l=document.createElement("div"),f=document.createElement("img"),k=document.createElement("span"),m=document.createElement("ul"),h=document.createElement("li"),q=document.createElement("ul"),r=document.createElement("ul"),t=I.a.a[b.l];d.id=a+"_"+b.c;d.className="slackmsg-item";e.className="slackmsg-ts";l.className="slackmsg-msg";f.className="slackmsg-author-img";k.className="slackmsg-author-name";
+m.className="slackmsg-hover";h.className="slackmsg-hover-reply";e.innerHTML=L.C(b.c);l.innerHTML=X(b.text);k.textContent=t?t.name:b.a||"?";f.src=t?t.a.small:"";m.appendChild(h);if("makeEmoji"in window){var u=document.createElement("li"),A=window.makeEmoji("arrow_heading_down"),E=window.makeEmoji("smile"),y=window.makeEmoji("pencil2"),t=window.makeEmoji("x");u.className="slackmsg-hover-reaction";E?(u.classList.add("emoji-small"),u.appendChild(E)):u.style.backgroundImage='url("smile.svg")';A?(h.classList.add("emoji-small"),
+h.appendChild(A)):h.style.backgroundImage='url("repl.svg")';m.appendChild(u);b.l===null.id&&(h=document.createElement("li"),h.className="slackmsg-hover-edit",y?h.classList.add("emoji-small"):h.style.backgroundImage='url("edit.svg")',h.appendChild(y),m.appendChild(h),h=document.createElement("li"),h.className="slackmsg-hover-remove",t?h.classList.add("emoji-small"):h.style.backgroundImage='url("remove.svg")',h.appendChild(t),m.appendChild(h))}else h.style.backgroundImage='url("repl.svg")',b.l===null.id&&
+(h=document.createElement("li"),h.className="slackmsg-hover-edit",h.style.backgroundImage='url("edit.svg")',m.appendChild(h),h=document.createElement("li"),h.className="slackmsg-hover-remove",h.style.backgroundImage='url("remove.svg")',m.appendChild(h));d.appendChild(f);b.b&&(f=document.createElement("span"),f.className="slackmsg-notice",f.textContent=L.K,g.appendChild(f));g.appendChild(k);g.appendChild(l);g.appendChild(e);g.appendChild(q);b.m&&(e=document.createElement("div"),e.textContent=L.m,e.className=
+"slackmsg-edited",g.appendChild(e));g.appendChild(r);g.className="slackmsg-content";q.className="slackmsg-attachments";r.className="slackmsg-reactions";if(!0!==c){if(b.j)for(var C in b.j)(c=Ca(a,b.id,C,b.j[C]))&&r.appendChild(c);b.A.forEach(function(a){var b=document.createElement("li"),c=document.createElement("div"),d=document.createElement("div"),e=document.createElement("a"),g=document.createElement("div"),f=document.createElement("img"),h=document.createElement("a"),y=document.createElement("div"),
+k=document.createElement("div"),l=document.createElement("img"),x=document.createElement("img"),m=document.createElement("div"),C=document.createElement("img"),r=document.createElement("span"),t=document.createElement("span");b.className="slackmsg-attachment";var u="#e3e4e6";a.color&&("#"===a.color[0]?u=a.color[0]:"good"===a.color?u="#2fa44f":"warning"===a.color?u="#de9e31":"danger"===a.color&&(u="#d50200"));c.style.borderColor=u;c.className="slackmsg-attachment-block";d.className="slackmsg-attachment-pretext";
+a.pretext?d.innerHTML=X(a.pretext):d.classList.add("hidden");e.target="_blank";a.title?(e.innerHTML=X(a.title),a.title_link&&(e.href=a.title_link),e.className="slackmsg-attachment-title"):e.className="hidden slackmsg-attachment-title";h.target="_blank";g.className="slackmsg-author";a.author_name?(h.innerHTML=X(a.author_name),h.href=a.author_link||"",h.className="slackmsg-author-name",f.className="slackmsg-author-img",a.author_icon?f.src=a.author_icon:f.classList.add("hidden")):g.classList.add("hidden");
+k.innerHTML=X(a.text||"");k.a="slackmsg-attachment-text";l.className="slackmsg-attachment-thumb";a.thumb_url?l.src=a.thumb_url:l.classList.add("hidden");x.className="slackmsg-attachment-img";a.image_url?x.src=a.image_url:x.classList.add("hidden");m.className="slackmsg-attachment-footer";r.className="slackmsg-attachment-footer-text";C.className="slackmsg-attachment-footer-icon";a.footer?(r.innerHTML=X(a.footer),a.footer_icon?C.src=a.footer_icon:C.classList.add("hidden")):(C.classList.add("hidden"),
+r.classList.add("hidden"));t.className="slackmsg-ts";a.ts?t.innerHTML=L.C(a.ts):t.classList.add("hidden");g.appendChild(f);g.appendChild(h);y.appendChild(k);y.appendChild(l);m.appendChild(C);m.appendChild(r);m.appendChild(t);c.appendChild(e);c.appendChild(g);c.appendChild(y);c.appendChild(x);c.appendChild(m);b.appendChild(d);b.appendChild(c);b&&q.appendChild(b)})}d.appendChild(g);d.appendChild(m);return d}
+function Aa(a){var b=document.createElement("lh");b.textContent=a;b.className="slack-command-header";return b}
+function Ba(a){var b=document.createElement("li"),c=document.createElement("span"),d=document.createElement("span"),g=document.createElement("span");c.textContent=a.name;d.textContent=a.usage;g.textContent=a.aa;b.appendChild(c);b.appendChild(d);b.appendChild(g);b.className="slack-command-item";c.className="slack-command-name";d.className="slack-command-usage";g.className="slack-command-desc";return b};var va=function(){function a(a,b){for(var c=a.target;c!==k&&c&&"LI"!==c.nodeName;)c=c.parentElement;c&&"LI"===c.nodeName&&c.id&&"emojibar-"===c.id.substr(0,9)?b(c.id.substr(9)):b(null)}function b(){if(!c())return!1;x&&x(null);return!0}function c(){return k.parentElement?(k.parentElement.removeChild(m),k.parentElement.removeChild(k),!0):!1}function d(a){var b=0,c;a=void 0===a?t.value:a;if(f()){var d=window.searchEmojis(a);c=g(d);for(var h in u)u[h].visible&&(u[h].visible=!1,q.removeChild(u[h].g));
+h=0;for(var y=c.length;h<y;h++){var v=c[h].name,k=u[v];if(!k){var k=u,l=v,x=v,v=window.makeEmoji(d[v]),m=document.createElement("span");m.appendChild(v);m.className="emoji-medium";v=e(x,m);k=k[l]=v}k.visible||(k.visible=!0,q.appendChild(k.g));b++}}for(h in A)A[h].visible&&(A[h].visible=!1,r.removeChild(A[h].g));c=g(I.a.f.data);h=0;for(y=c.length;h<y;h++)v=c[h].name,""!==a&&v.substr(0,a.length)!==a||"alias:"===I.a.f.data[v].substr(0,6)||(k=A[v],k||(d=A,l=k=v,v=I.a.f.data[v],x=document.createElement("span"),
+m=document.createElement("span"),x.className="emoji emoji-custom",x.style.backgroundImage='url("'+v+'")',m.appendChild(x),m.className="emoji-medium",v=e(l,m),k=d[k]=v),k.visible||(k.visible=!0,r.appendChild(k.g)),b++);return b}function g(a){var b=null.M.ba,c=[],d;for(d in a){var e={name:d,L:0,count:0};a[d].names.forEach(function(a){e.count+=b[a]||0});c.push(e)}return c=c.sort(function(a,b){var c=b.count-a.count;return c?c:a.L-b.L})}function e(a,b){var c=document.createElement("li");c.appendChild(b);
+c.className="emojibar-list-item";c.id="emojibar-"+a;return{visible:!1,g:c}}function l(a){var b=document.createElement("img"),c=document.createElement("div");b.src=a;c.appendChild(b);c.className="emojibar-header";return c}function f(){return"searchEmojis"in window}var k=document.createElement("div"),m=document.createElement("div"),h=document.createElement("div"),q=document.createElement("ul"),r=document.createElement("ul"),t=document.createElement("input"),u={},A={},E=document.createElement("div"),
+y=document.createElement("span"),C=document.createElement("span"),x;m.addEventListener("click",function(a){var c=k.getBoundingClientRect();(a.screenY<c.top||a.screenY>c.bottom||a.screenX<c.left||a.screenX>c.right)&&b()});m.className="emojibar-overlay";k.className="emojibar";h.className="emojibar-emojis";E.className="emojibar-detail";y.className="emojibar-detail-img";C.className="emojibar-detail-name";q.className=r.className="emojibar-list";t.className="emojibar-search";E.appendChild(y);E.appendChild(C);
+h.appendChild(l(window.emojiProviderHeader));h.appendChild(q);h.appendChild(l("emojicustom.png"));h.appendChild(r);k.appendChild(h);k.appendChild(E);k.appendChild(t);t.addEventListener("keyup",function(){d()});k.addEventListener("mousemove",function(b){a(b,function(a){var b=a?u[a]||A[a]:null;b?(y.innerHTML=b.g.outerHTML,C.textContent=":"+a+":"):(y.textContent="",C.textContent="")})});k.addEventListener("click",function(b){a(b,function(a){a&&c()&&x&&x(a)})});return{isSupported:f,N:function(a,b){return f()?
+(x=b,a.appendChild(m),a.appendChild(k),t.value="",d(),t.focus(),!0):!1},search:d,close:b}}();var I,N=[];setInterval(function(){I.a.ca(Date.now())&&ka()},1E3);
+function Da(a,b){if(a!==J||!window.hasFocus){var c=new RegExp("<@"+null.id),d=!1,g=!1,e=!1;b.forEach(function(b){if(!(parseFloat(b.ts)<=a.h)){g=!0;var f;if(!(f=a instanceof w||b.text.match(c)))a:{f=null.M.da;for(var l=0,h=f.length;l<h;l++)if(-1!==b.text.indexOf(f[l])){f=!0;break a}f=!1}f&&(-1===N.indexOf(a)&&(e=!0,N.push(a)),d=!0)}});if(g){O();var l=document.getElementById(a.id);l&&(l.classList.add("unread"),d&&l.classList.add("unreadHi"));e&&!window.hasFocus&&ta()}}}
+function R(){var a=J,b=N.indexOf(a);if(a.i>a.h){var c=new XMLHttpRequest;c.open("POST","api/markread?room="+a.id+"&ts="+a.i,!0);c.send(null);a.h=a.i}0<=b&&(N.splice(b,1),O());a=document.getElementById(a.id);a.classList.remove("unread");a.classList.remove("unreadHi")}I=new function(){this.f=0;this.a=new n;this.b={}};var ja=function(){function a(a,b){b.sort(function(){return Math.random()-.5});for(var c=0,d=20;d<k-40;d+=h)for(var f=0;f+h<=m;f+=h)e(a,b[c],d,f),c++,c===b.length&&(b.sort(function(a,b){return a.s?b.s?Math.random()-.5:-1:1}),c=0)}function b(a,d){for(var e=0,f=a.length;e<f;e++)if(void 0===a[e].s){c(a[e].src,function(c){a[e].s=c;b(a,d)});return}var g=[];a.forEach(function(a){a.s&&g.push(a.s)});d(g)}function c(a,b){var c=new XMLHttpRequest;c.responseType="blob";c.onreadystatechange=function(){if(4===
+c.readyState)if(c.response){var a=new Image;a.onload=function(){var c=document.createElement("canvas");c.height=c.width=t;c=c.getContext("2d");c.drawImage(a,0,0,t,t);for(var c=c.getImageData(0,0,t,t),d=0,e=0;e<c.width*c.height*4;e+=4)c.data[e]=c.data[e+1]=c.data[e+2]=(c.data[e]+c.data[e+1]+c.data[e+2])/3,c.data[e+3]=50,d+=c.data[e];if(50>d/(c.height*c.width))for(e=0;e<c.width*c.height*4;e+=4)c.data[e]=c.data[e+1]=c.data[e+2]=255-c.data[e];b(c)};a.onerror=function(){b(null)};a.src=window.URL.createObjectURL(c.response)}else b(null)};
+c.open("GET",a,!0);c.send(null)}function d(){var a=f.createLinearGradient(0,0,0,m);a.addColorStop(0,"#4D394B");a.addColorStop(1,"#201820");f.fillStyle=a;f.fillRect(0,0,k,m);return f.getImageData(0,0,k,m)}function g(a,b){for(var c=(a.height-b.height)/2,d=0;d<b.height;d++)for(var e=0;e<b.width;e++){var f=b.data[4*(d*b.width+e)]/255,g=4*((d+c)*a.width+e+c);a.data[g]*=f;a.data[g+1]*=f;a.data[g+2]*=f}return a}function e(a,b,c,d){var e=Math.floor(d);a=[a.data[e*k*4+0],a.data[e*k*4+1],a.data[e*k*4+2]];f.fillStyle=
+"#"+(1.1*a[0]<<16|1.1*a[1]<<8|1.1*a[2]).toString(16);f.beginPath();f.moveTo(c+h/2,d+q);f.lineTo(c-q+h,d+h/2);f.lineTo(c+h/2,d-q+h);f.lineTo(c+q,d+h/2);f.closePath();f.fill();f.putImageData(g(f.getImageData(c+q,d+q,r,r),b),c+q,d+q)}var l=document.createElement("canvas"),f=l.getContext("2d"),k=l.width=250,m=l.height=290,h=(k-40)/3,q=.1*h,r=Math.floor(h-2*q),t=.5*r,u,A=[],E=!1;return function(c){if(u)c(u);else if(E)A.push(c);else{var e=d(),f=[];E=!0;A.push(c);for(var g in I.a.a)I.a.a[g].b||f.push({src:"api/avatar?user="+
+g});b(f,function(b){a(e,b);u=l.toDataURL();A.forEach(function(a){a(u)})})}}}();var Y=0,J=null,S=null,U=null;function Ea(a){var b=new XMLHttpRequest;b.timeout=6E4;b.onreadystatechange=function(){if(4===b.readyState)if(b.status){var c=null,d=2===Math.floor(b.status/100);if(d){Y&&(Y=0,P(!0));c=b.response;try{c=JSON.parse(c)}catch(g){c=null}}else Y?(Y+=Math.floor((Y||5)/2),Y=Math.min(60,Y)):(Y=5,P(!1));a(d,c)}else Y&&(Y=0,P(!0)),Ea(a)};b.open("GET","api?v="+I.f,!0);b.send(null)}function za(){var a=new XMLHttpRequest;a.open("POST","api/typing?room="+J.id,!0);a.send(null)}
+function Fa(a,b){if(a){if(b){var c=I,d=Date.now();b.v&&(c.f=b.v);b["static"]&&c.a.ea(b["static"],Date.now());for(var g in c.a.b){var e=c.a.b[g];if(e.i===e.h){var l=N.indexOf(e);-1!==l&&N.splice(l,1)}}if(b.live){for(e in b.live)(g=c.b[e])?H(g,b.live[e],d):c.b[e]=new G(e,250,b.live[e],d);for(var f in b.live)(d=c.a.b[f])?(c.b[f].a.length&&(e=c.b[f],d.i=Math.max(d.i,e.a[e.a.length-1].c)),d.b||(Da(d,b.live[f]),J&&b.live[J.id]&&na())):I.f=0}b["static"]&&(ga(),b["static"].typing&&ka())}Z()}else setTimeout(Z,
+1E3*Y)}function Z(){Ea(Fa)}function xa(a,b,c){var d=J;new FileReader;var g=new FormData,e=new XMLHttpRequest;g.append("file",b);g.append("filename",a);e.onreadystatechange=function(){4===e.readyState&&(204===e.status?c(null):c(e.statusText))};e.open("POST","api/file?room="+d.id);e.send(g)}
+function ya(a){if(U){var b=new XMLHttpRequest;b.open("PUT","api/msg?room="+J.id+"&ts="+U.id+"&text="+encodeURIComponent(a),!0);b.send(null);return!0}if("/"===a[0]){var c=a.indexOf(" "),b=-1===c?"":a.substr(c);return(a=I.a.B.data[a.substr(0,-1===c?void 0:c)])?(c=new XMLHttpRequest,c.open("POST","api/cmd?room="+J.id+"&cmd="+encodeURIComponent(a.name.substr(1))+"&args="+encodeURIComponent(b.trim()),!0),c.send(null),!0):!1}var b=J,c=S,d=new XMLHttpRequest;a="api/msg?room="+b.id+"&text="+encodeURIComponent(a);
+if(c){var g=I.a.a[c.l],e="Message";"C"===b.id[0]?e="Channel message":"D"===b.id[0]?e="Direct message":"G"===b.id[0]&&(e="Group message");a+="&attachments="+encodeURIComponent(JSON.stringify([{fallback:c.text,author_name:"<@"+g.id+"|"+g.name+">",author_icon:g.a.small,text:c.text,footer:e,ts:c.c}]))}d.open("POST",a,!0);d.send(null);return!0}function wa(a){var b=new XMLHttpRequest;b.open("DELETE","api/msg?room="+J.id+"&ts="+a.id,!0);b.send(null)}
+function oa(a,b,c){var d=new XMLHttpRequest;d.open("POST","api/reaction?room="+a+"&msg="+b+"&reaction="+encodeURIComponent(c),!0);d.send(null)};

+ 1 - 1
srv/public/style.css

@@ -73,7 +73,7 @@ body { display: flex; margin: 0; padding: 0; font-family: Lato, sans-serif; heig
 .slackmsg-item.slackmsg-me_message { font-style: italic; margin-right: 10px; }
 .slackmsg-item.slackmsg-me_message .slackmsg-author-name { position: static; display: inline; }
 .slackmsg-item.slackmsg-me_message .slackmsg-author-name::before { content: "* "; }
-.slackmsg-item.slackmsg-me_message .slackmsg-msg { margin: 0 0 0 1em; }
+.slackmsg-item.slackmsg-me_message .slackmsg-msg { margin: 0 0 0 1em; display: inline; }
 .slackmsg-item.slackmsg-me_message .slackmsg-author-img { display: none; }
 .slackmsg-item.slackmsg-me_message .slackmsg-ts { display: block; }
 

+ 44 - 0
srv/src/chatter.js

@@ -0,0 +1,44 @@
+/**
+ * @constructor
+**/
+function Chatter(id) {
+    /** @const @type {string} */
+    this.id = id;
+    /** @type {string} */
+    this.name;
+    /** @type {boolean} */
+    this.deleted;
+    /** @type {string} */
+    this.status;
+    /** @type {string} */
+    this.realName;
+    /** @type {boolean} */
+    this.presence;
+    /** @type {Object.<string, string>} */
+    this.icons = {
+        small: ""
+        ,large: ""
+    };
+    /** @type {string} */
+    this.email;
+    /** @type {string} */
+    this.firstName;
+    /** @type {string} */
+    this.lastName;
+    /** @type {!Object.<string, Room>} */
+    this.channels = {};
+    /** @type {PrivateMessageRoom} */
+    this.ims = null;
+    /** @type {SelfPreferences|null} */
+    this.prefs = null;
+    /** @type {number} */
+    this.version = 0;
+}
+
+/** @suppress {undefinedVars,checkTypes} */
+(function() {
+    if (typeof module !== "undefined") {
+        module.exports.Chatter = Chatter;
+    }
+})();
+

+ 59 - 0
srv/src/context.js

@@ -0,0 +1,59 @@
+
+/**
+ * @constructor
+**/
+function ChatInfo(id) {
+    /** @const @type {string} */
+    this.id = id;
+    /** @type {string} */
+    this.name;
+    /** @type {string} */
+    this.domain;
+    /** @type {string} */
+    this.callApp;
+    /** @type {string} */
+    this.callAppName;
+    /** @type {boolean} */
+    this.fileUploadPermission;
+    /** @type {boolean} */
+    this.fileEditPermission;
+    /** @type {boolean} */
+    this.fileDeletePermission;
+    /** @type {string} */
+    this.icons = ""
+    /** @type {number} */
+    this.version = 0;
+}
+
+/**
+ * @constructor
+**/
+function ChatContext() {
+    /** @type {ChatInfo} */
+    this.team = null;
+    /** @type {Object.<string, Room>} */
+    this.channels = {};
+    /** @type {Object.<string, Chatter>} */
+    this.users = {};
+    /** @type {Chatter} */
+    this.self = null;
+    /** @type {{version: number, data: Object.<string, string>}} */
+    this.emojis = { version: 0, data: {} };
+    /** @type {{version: number, data: Object.<string, SlackCommand>}} */
+    this.commands = {version: 0, data: {} };
+    /** @type {Object.<string, Object.<string, number>>} */
+    this.typing = {};
+    /** @type {number} */
+    this.staticV = 0;
+    /** @type {number} */
+    this.liveV = 0;
+}
+
+/** @suppress {undefinedVars,checkTypes} */
+(function() {
+    if (typeof module !== "undefined") {
+        module.exports.ChatContext = ChatContext;
+        module.exports.ChatInfo = ChatInfo;
+    }
+})();
+

+ 308 - 0
srv/src/message.js

@@ -0,0 +1,308 @@
+/**
+ * Abstract class Message
+ * @constructor
+ * @param {*} e
+ * @param {number} ts
+**/
+function Message(e, ts) {
+    /** @const @type {string} **/
+    this.userId = e.userId;
+
+    /** @type {string} */
+    this.username = e.userName;
+
+    /** @const @type {string} **/
+    this.id = e.id;
+
+    /** @const @type {number} **/
+    this.ts = e.ts;
+
+    /** @type {string} */
+    this.text;
+
+    /** @type {Array.<*>} */
+    this.attachments;
+
+    /** @type {boolean} */
+    this.starred;
+
+    /** @type {boolean} */
+    this.pinned;
+
+    /** @type {boolean} */
+    this.edited;
+
+    /** @type {boolean} */
+    this.removed;
+
+    /**
+     * for each emoji code, an array of user id
+     * @type {Object.<string, Array.<string>>}
+    **/
+    this.reactions;
+
+    /** @type {number} */
+    this.version = ts;
+
+    this.update(e, ts);
+}
+
+/**
+ * @constructor
+ * @extends {Message}
+**/
+function MeMessage(e, ts) {
+    Message.call(this, e, ts);
+}
+
+/**
+ * @constructor
+ * @extends {Message}
+**/
+function NoticeMessage(e, ts) {
+    Message.call(this, e, ts);
+}
+
+Message.prototype.update = function(e, ts) {
+    if (e) {
+        this.text = e.text || "";
+        if (e.attachments) this.attachments = e.attachments;
+        this.starred = !!e.is_starred;
+        this.pinned = false; // TODO
+        this.edited = !!e.edited;
+        this.removed = !!e.removed;
+
+        var _this = this;
+        if (e.reactions) {
+            this.reactions = {};
+            e.reactions.forEach(function(r) {
+                _this.reactions[r["name"]] = [];
+                r["users"].forEach(function(userId) {
+                    _this.reactions[r["name"]].push(userId);
+                });
+            });
+        }
+    } else {
+        this.removed = true;
+    }
+    this.version = ts;
+}
+
+/** @return {*} */
+Message.prototype.toStatic = function() {
+    var reactions = [];
+    for (var reaction in this.reactions) {
+        reactions.push({
+            "name": reaction
+            ,"users": this.reactions[reaction]
+        });
+    }
+    return {
+        "user": this.userId
+        ,"username": this.username
+        ,"ts": this.id
+        ,"text": this.text
+        ,"attachments": this.attachments
+        ,"is_starred": this.is_starred || undefined
+        ,"edited": this.edited
+        ,"removed": this.removed
+        ,"reactions": reactions
+        ,"isMeMessage": this instanceof MeMessage
+        ,"isNotice": this instanceof NoticeMessage
+    };
+}
+
+/**
+ * @param {string} reaction
+ * @param {string} userId
+ * @param {number} ts
+**/
+Message.prototype.addReaction = function(reaction, userId, ts) {
+    if (!this.reactions[reaction])
+        this.reactions[reaction] = [];
+    this.reactions[reaction].push(userId);
+    this.version = ts;
+}
+
+/**
+ * @param {string} reaction
+ * @param {string} userId
+ * @param {number} ts
+**/
+Message.prototype.removeReaction = function(reaction, userId, ts) {
+    var updated = false;
+    if (this.reactions[reaction]) {
+        if (this.reactions[reaction].length === 1 && this.reactions[reaction][0] === userId) {
+            delete this.reactions[reaction];
+            updated = true;
+        }
+        else {
+            this.reactions[reaction] = this.reactions[reaction].filter(function(i) {
+                if (i !== userId)
+                    return false;
+                updated = true;
+                return true;
+            });
+        }
+    }
+    if (updated)
+        this.version = ts;
+}
+
+Message.prototype.hasReactionForUser = function(reaction, userId) {
+    if (this.reactions[reaction])
+        return this.reactions[reaction].indexOf(userId) !== -1;
+    return false;
+}
+
+/**
+ * @constructor
+ * @param {Room|string} room or roomId
+ * @param {number} keepMessages number of messages to keep in memory
+ * @param {Array|undefined} evts
+ * @param {number|undefined} now
+**/
+function RoomHistory(room, keepMessages, evts, now) {
+    /** @type {string} */
+    this.id = typeof room === "string" ? room : room.id;
+    /** @type {Array.<Message>} */
+    this.messages = [];
+    /** @type number */
+    this.v = 0;
+
+    /** @const @type {number} */
+    this.keepMessages = keepMessages;
+
+    if (evts) this.pushAll(evts, now);
+}
+
+/**
+ * @param {Array} evts
+ * @return {number} biggest ts
+**/
+RoomHistory.prototype.pushAll = function(evts, t) {
+    var result = 0;
+
+    evts.forEach(function(e) {
+        result = Math.max(this.push(e, t), result);
+    }.bind(this));
+    this.resort();
+    return result;
+}
+
+/**
+ * @param {string} msgId
+ * @param {Message} msg
+ * @param {function(*):Message} msgFactory
+ * @return {number} t
+**/
+RoomHistory.prototype.push = function(msgId, msg, msgFactory, t) {
+    var exists = false
+        ,ts;
+
+    for (var i =0, nbMsg = this.messages.length; i < nbMsg; i++) {
+        var msgObj = this.messages[i];
+        if (msgObj.id === msgId) {
+            ts = msgObj.update(msg, t);
+            exists = true;
+            break;
+        }
+    }
+    if (!exists) {
+        var msgObj = msgFactory(msg);
+        this.messages.push(msgObj);
+        ts = msgObj.ts;
+    }
+    while (this.messages.length > this.keepMessages)
+        this.messages.shift();
+    return ts || 0;
+}
+
+RoomHistory.prototype.lastMessage = function() {
+    return this.messages[this.messages.length -1];
+}
+
+/**
+ * @param {string} reaction
+ * @param {string} userId
+ * @param {string} msgId
+ * @param {number} ts
+ * @return {Message|null}
+**/
+RoomHistory.prototype.addReaction = function(reaction, userId, msgId, ts) {
+    var msg = this.getMessageById(msgId);
+    if (msg)
+        msg.addReaction(reaction, userId, ts);
+    return msg;
+}
+
+/**
+ * @param {string} reaction
+ * @param {string} userId
+ * @param {string} msgId
+ * @param {number} ts
+ * @return {Message|null}
+**/
+RoomHistory.prototype.removeReaction = function(reaction, userId, msgId, ts) {
+    var msg = this.getMessageById(msgId);
+    if (msg)
+        msg.removeReaction(reaction, userId, ts);
+    return msg;
+}
+
+/**
+ * @param {number} ts
+ * @return {Message|null}
+**/
+RoomHistory.prototype.getMessage = function(ts) {
+    for (var i =0, nbMessages = this.messages.length; i < nbMessages && ts >= this.messages[i].ts; i++) {
+        if (this.messages[i].ts === ts)
+            return this.messages[i];
+    }
+    return null;
+}
+
+/**
+ * @param {string} id
+ * @return {Message|null}
+**/
+RoomHistory.prototype.getMessageById = function(id) {
+    for (var i =0, nbMessages = this.messages.length; i < nbMessages; i++) {
+        if (this.messages[i].id == id)
+            return this.messages[i];
+    }
+    return null;
+}
+
+/**
+ * @return {Array.<*>}
+**/
+RoomHistory.prototype.toStatic = function(knownVersion) {
+    var result = [];
+
+    for (var i = this.messages.length -1; i >= 0; i--) {
+        if (this.messages[i].version > knownVersion)
+            result.push(this.messages[i].toStatic());
+    }
+    return result;
+}
+
+RoomHistory.prototype.resort = function() {
+    this.messages.sort(function(a, b) {
+        return a.ts -b.ts;
+    });
+};
+
+MeMessage.prototype = Object.create(Message);
+NoticeMessage.prototype = Object.create(Message);
+
+/** @suppress {undefinedVars,checkTypes} */
+(function() {
+    if (typeof module !== "undefined") module.exports = {
+        Message: Message
+        ,MeMessage: MeMessage
+        ,NoticeMessage: NoticeMessage
+        ,RoomHistory: RoomHistory
+    }
+})();
+

+ 70 - 0
srv/src/room.js

@@ -0,0 +1,70 @@
+/**
+ * @constructor
+**/
+function Room(id) {
+    /** @const @type {string} */
+    this.id = id;
+    /** @type {string} */
+    this.name;
+    /** @type {string} */
+    this.created;
+    /** @type {Chatter} */
+    this.creator;
+    /** @type {boolean} */
+    this.archived;
+    /** @type {boolean} */
+    this.isMember;
+    /** @type {number} */
+    this.lastRead = 0;
+    /** @type {number} */
+    this.lastMsg = 0;
+    /** @type {Array.<Chatter>} */
+    this.users = [];
+    /** @type {string|undefined} */
+    this.topic;
+    /** @type {number|undefined} */
+    this.topicTs;
+    /** @type {Chatter|undefined} */
+    this.topicCreator;
+    /** @type {string|undefined} */
+    this.purpose;
+    /** @type {number|undefined} */
+    this.purposeTs;
+    /** @type {Chatter|undefined} */
+    this.purposeCreator;
+    /** @type {number} */
+    this.version = 0;
+    /** @type {boolean} */
+    this.isPrivate;
+}
+
+Room.prototype.setLastMsg = function(lastMsg, t) {
+    if (this.lastMsg < lastMsg) {
+        this.lastMsg = lastMsg;
+        this.version = t;
+        return true;
+    }
+    return false;
+}
+
+/**
+ * @constructor
+ * @param {string} id
+ * @param {Chatter} user
+ * @extends {Room}
+**/
+function PrivateMessageRoom(id, user) {
+    Room.call(this, id);
+    /** @const @type {Chatter} */
+    this.user = user;
+}
+PrivateMessageRoom.prototype = Object.create(Room);
+
+/** @suppress {undefinedVars,checkTypes} */
+(function() {
+    if (typeof module !== "undefined") {
+        module.exports.Room = Room;
+        module.exports.PrivateMessageRoom = PrivateMessageRoom;
+    }
+})();
+

+ 55 - 212
srv/src/slackData.js

@@ -1,38 +1,15 @@
 
+const ChatContext = require('./context.js').ChatContext
+    ,ChatInfo = require('./context.js').ChatInfo;
+
 /**
  * @constructor
+ * @extends {ChatInfo}
 **/
 function SlackTeam(teamId) {
-    /** @const @type {string} */
-    this.id = teamId;
-    /** @type {string} */
-    this.name;
-    /** @type {string} */
-    this.domain;
-    /** @type {string} */
-    this.callApp;
-    /** @type {string} */
-    this.callAppName;
-    /** @type {boolean} */
-    this.fileUploadPermission;
-    /** @type {boolean} */
-    this.fileEditPermission;
-    /** @type {boolean} */
-    this.fileDeletePermission;
-    /** @type {Object.<string, string>} */
-    this.icons = {
-        image_34: ""
-        ,image_44: ""
-        ,image_68: ""
-        ,image_88: ""
-        ,image_102: ""
-        ,image_132: ""
-        ,image_230: ""
-        ,image_default: ""
-    };
-    /** @type {number} */
-    this.version = 0;
+    ChatInfo.call(this, teamId);
 }
+SlackTeam.prototype = Object.create(ChatInfo);
 
 SlackTeam.prototype.update = function(teamData, t) {
     if (teamData["name"] !== undefined) this.name = teamData["name"];
@@ -61,40 +38,11 @@ SlackTeam.prototype.update = function(teamData, t) {
  * @constructor
 **/
 function SlackChan(chanId) {
-    /** @const @type {string} */
-    this.id = chanId;
-    /** @type {string} */
-    this.name;
-    /** @type {string} */
-    this.created;
-    /** @type {SlackUser|SlackBot} */
-    this.creator;
-    /** @type {boolean} */
-    this.archived;
-    /** @type {boolean} */
-    this.isMember;
-    /** @type {number} */
-    this.lastRead = 0;
-    /** @type {number} */
-    this.lastMsg = 0;
-    /** @type {Object.<string, SlackBot|SlackUser>} */
-    this.members = {};
-    /** @type {string|undefined} */
-    this.topic;
-    /** @type {number|undefined} */
-    this.topicTs;
-    /** @type {SlackUser|SlackBot|undefined} */
-    this.topicCreator;
-    /** @type {string|undefined} */
-    this.purpose;
-    /** @type {number|undefined} */
-    this.purposeTs;
-    /** @type {SlackUser|SlackBot|undefined} */
-    this.purposeCreator;
-    /** @type {number} */
-    this.version = 0;
+    Room.call(this, chanId);
 }
 
+SlackChan.prototype = Object.create(Room);
+
 /**
  * @param {*} chanData
  * @param {SlackData} slackData
@@ -129,15 +77,6 @@ SlackChan.prototype.update = function(chanData, slackData, t) {
     this.version = Math.max(this.version, t);
 }
 
-SlackChan.prototype.setLastMsg = function(lastMsg, t) {
-    if (this.lastMsg < lastMsg) {
-        this.lastMsg = lastMsg;
-        this.version = t;
-        return true;
-    }
-    return false;
-}
-
 /**
  * @constructor
  * @param {string} id
@@ -215,15 +154,6 @@ SlackGroup.prototype.update = function(slack, groupData, t) {
     this.version = Math.max(this.version, t);
 }
 
-SlackGroup.prototype.setLastMsg = function(lastMsg, t) {
-    if (this.lastMsg < lastMsg) {
-        this.lastMsg = lastMsg;
-        this.version = t;
-        return true;
-    }
-    return false;
-}
-
 /**
  * @constructor
  * @param {string} id
@@ -259,55 +189,53 @@ SlackIms.prototype.update = function(user, imsData, t) {
     this.version = Math.max(this.version, t);
 }
 
-SlackIms.prototype.setLastMsg = function(lastMsg, t) {
-    if (this.lastMsg < lastMsg) {
-        this.lastMsg = lastMsg;
-        this.version = t;
-        return true;
-    }
-    return false;
+/**
+ * @constructor
+**/
+function SelfPreferences() {
+    /** @type {Object.<string, number>} */
+    this.favoriteEmojis = {};
+
+    /** @type {Array.<string>} */
+    this.highlights = [];
+}
+
+/**
+ * @param {*} prefs
+**/
+SelfPreferences.prototype.update = function(prefs, t) {
+    this.favoriteEmojis = /** @type {Object<string,number>} */ (JSON.parse(prefs["emoji_use"]));
+    if (prefs["highlight_words"])
+        this.highlights = (prefs["highlight_words"]||"").split(',').filter(function(i) {
+            return i.trim() !== '';
+        });
+    else if (prefs["highlights"])
+        this.highlights = prefs["highlights"];
+    this.version = Math.max(this.version, t);
 }
 
 /**
  * @constructor
+ * @extends {Chatter}
+**/
+function SlackChatter(id) {
+    Chatter.call(this, id);
+}
+SlackChatter.prototype = Object.create(Chatter);
+
+SlackChatter.prototype.setPresence = function(presenceStr, t) {
+    this.presence = presenceStr !== 'away';
+    this.version = Math.max(t, this.version);
+}
+
+/**
+ * @constructor
+ * @extends {SlackChatter}
 **/
 function SlackUser(id) {
-    /** @const @type {string} */
-    this.id = id;
-    /** @type {string} */
-    this.name;
-    /** @type {boolean} */
-    this.deleted;
-    /** @type {string} */
-    this.status;
-    /** @type {string} */
-    this.realName;
-    /** @type {boolean} */
-    this.presence;
-    /** @type {Object.<string, string>} */
-    this.icons = {
-        image_24: ""
-        ,image_32: ""
-        ,image_48: ""
-        ,image_72: ""
-        ,image_192: ""
-        ,image_512: ""
-    };
-    /** @type {string} */
-    this.email;
-    /** @type {string} */
-    this.firstName;
-    /** @type {string} */
-    this.lastName;
-    /** @type {!Object.<string, SlackChan|SlackGroup>} */
-    this.channels = {};
-    /** @type {SlackIms} */
-    this.ims = null;
-    /** @type {SelfPreferences|null} */
-    this.prefs = null;
-    /** @type {number} */
-    this.version = 0;
+    SlackChatter.call(this, id);
 }
+SlackUser.prototype = Object.create(SlackChatter);
 
 /**
  * @param {*} userData
@@ -338,65 +266,16 @@ SlackUser.prototype.update = function(userData, t) {
     this.version = Math.max(this.version, t);
 }
 
-SlackUser.prototype.setPresence = function(presenceStr, t) {
-    this.presence = presenceStr !== 'away';
-    this.version = Math.max(t, this.version);
-}
-
-/**
- * @constructor
-**/
-function SelfPreferences() {
-    /** @type {Object.<string, number>} */
-    this.favoriteEmojis = {};
-
-    /** @type {Array.<string>} */
-    this.highlights = [];
-}
-
-/**
- * @param {*} prefs
-**/
-SelfPreferences.prototype.update = function(prefs, t) {
-    this.favoriteEmojis = /** @type {Object<string,number>} */ (JSON.parse(prefs["emoji_use"]));
-    if (prefs["highlight_words"])
-        this.highlights = (prefs["highlight_words"]||"").split(',').filter(function(i) {
-            return i.trim() !== '';
-        });
-    else if (prefs["highlights"])
-        this.highlights = prefs["highlights"];
-    this.version = Math.max(this.version, t);
-}
-
 /**
  * @constructor
+ * @extends {SlackChatter}
 **/
 function SlackBot(id) {
-    /** @const @type {string} */
-    this.id = id;
-    /** @type {boolean} */
-    this.deleted;
-    /** @type {string} */
-    this.name;
+    SlackChatter.call(this, id);
     /** @type {string} */
     this.appId;
-    /** @type {Object.<string, string>} */
-    this.icons = {
-        image_36: ""
-        ,image_48: ""
-        ,image_72: ""
-    };
-    /** @type {!Object.<string, SlackGroup|SlackChan>} */
-    this.channels;
-    /** @type {SlackIms|null} */
-    this.ims = null;
-    /** @type {SelfPreferences|null} */
-    this.prefs = null;
-    /** @type {number} */
-    this.version = 0;
-    /** @type {boolean} */
-    this.presence = false;
 }
+SlackBot.prototype = Object.create(SlackChatter);
 
 /** @param {*} botData */
 SlackBot.prototype.update = function(botData, t) {
@@ -415,11 +294,6 @@ SlackBot.prototype.update = function(botData, t) {
     this.version = Math.max(this.version, t);
 }
 
-SlackBot.prototype.setPresence = function(presenceStr, t) {
-    this.presence = presenceStr !== 'away';
-    this.version = Math.max(t, this.version);
-}
-
 /**
  * @constructor
  * @param {*} data
@@ -475,38 +349,15 @@ SlackCommand.prototype.toStatic = function(t) {
  * @constructor
 **/
 function SlackData(slack) {
-    /** @type {SlackTeam} */
-    this.team = null;
-    /** @type {Object.<string, SlackChan>} */
-    this.channels = {};
-    /** @type {Object.<string, SlackGroup>} */
-    this.groups = {};
-    /** @type {Object.<string, SlackIms>} */
-    this.ims = {};
-    /** @type {Object.<string, SlackUser>} */
-    this.users = {};
-    /** @type {SlackUser|SlackBot} */
-    this.self = null;
-    /** @type {Object.<string, SlackBot>} */
-    this.bots = {};
-    /** @type {{version: number, data: Object.<string, string>}} */
-    this.emojis = { version: 0, data: {} };
-    /** @type {{version: number, data: Object.<string, SlackCommand>}} */
-    this.commands = {version: 0, data: {} };
-    /** @type {Object.<string, Object.<string, number>>} */
-    this.typing = {};
+    ChatContext.call(this);
 
     /**
      * Node serv handler
      * @type {*}
     **/
     this.slack = slack;
-
-    /** @type {number} */
-    this.staticV = 0;
-    /** @type {number} */
-    this.liveV = 0;
 }
+SlackData.prototype = Object.create(ChatContext);
 
 /**
  * @return {Object}
@@ -731,14 +582,6 @@ SlackData.prototype.getBotsByAppId = function(appId) {
     return bots;
 };
 
-/**
- * @param {string} mId
- * @return {SlackUser|SlackBot|null}
-**/
-SlackData.prototype.getMember = function(mId) {
-    return this.users[mId] || this.bots[mId] || null;
-};
-
 /**
  * @param {string} chanId
  * @return {SlackChan|SlackGroup|SlackIms|null}
@@ -748,7 +591,7 @@ SlackData.prototype.getChannel = function(chanId) {
 };
 
 /**
- * @param {function((SlackGroup|SlackIms|SlackChan)):(boolean|undefined)} cb should return false to stop iterating
+ * @param {function(Room):(boolean|undefined)} cb should return false to stop iterating
  * @return {boolean} true if iterated through all entities
 **/
 SlackData.prototype.forEachChans = function(cb) {

+ 6 - 144
srv/src/slackHistory.js

@@ -1,3 +1,4 @@
+const Message = require('./message.js');
 
 /**
  * @constructor
@@ -5,82 +6,13 @@
  * @param {number} ts
 **/
 function SlackMessage(e, ts) {
-    /** @const @type {string} **/
-    this.userId = e["user"] || e["bot_id"];
-
-    /** @type {string} */
-    this.username = e["username"];
-
-    /** @const @type {string} **/
-    this.id = e["ts"];
-
-    /** @const @type {number} **/
-    this.ts = parseFloat(e["ts"]);
-
-    /** @type {string} */
-    this.text = e["text"] || "";
-
-    /** @type {Array.<*>} */
-    this.attachments = e["attachments"] || [];
-
-    /** @type {boolean} */
-    this.starred = !!e["is_starred"];
-
-    /** @type {boolean} */
-    this.pinned = false; // TODO
-
-    /** @type {boolean} */
-    this.edited = !!e["edited"];
-
-    /** @type {boolean} */
-    this.removed = e["removed"] || false;
-
-    /** @const @type {boolean} */
-    this.isMeMessage = e["subtype"] === "me_message" || e["isMeMessage"];
-
-    /**
-     * for each emoji code, an array of user id
-     * @type {Object.<string, Array.<string>>}
-    **/
-    this.reactions = {};
-
-    /** @type {number} */
-    this.version = ts;
-
-    this.notice = e["is_ephemeral"] === true || e["isNotice"] === true;
-
-    var _this = this;
-    e["reactions"] && e["reactions"].forEach(function(r) {
-        _this.reactions[r["name"]] = [];
-        r["users"].forEach(function(userId) {
-            _this.reactions[r["name"]].push(userId);
-        });
-    });
+    Message.call(this, e, ts);
 }
 
+SlackMessage.prototype = Object.create(Message);
+SlackMessage.prototype.superMessageUpdate = SlackMessage.prototype.update;
 SlackMessage.prototype.update = function(e, ts) {
-    if (e) {
-        this.text = e["text"] || "";
-        if (e["attachments"]) this.attachments = e["attachments"];
-        this.starred = !!e["is_starred"];
-        this.pinned = false; // TODO
-        this.edited = !!e["edited"];
-        this.removed = !!e["removed"];
-
-        var _this = this;
-        if (e["reactions"]) {
-            this.reactions = {};
-            e["reactions"].forEach(function(r) {
-                _this.reactions[r["name"]] = [];
-                r["users"].forEach(function(userId) {
-                    _this.reactions[r["name"]].push(userId);
-                });
-            });
-        }
-    } else {
-        this.removed = true;
-    }
-    this.version = ts;
+    this.superMessageUpdate(e, ts);
 }
 
 /**
@@ -104,73 +36,6 @@ function SlackHistory(room, keepMessages, evts, now) {
     if (evts) this.pushAll(evts, now);
 }
 
-/** @return {*} */
-SlackMessage.prototype.toStatic = function() {
-    var reactions = [];
-    for (var reaction in this.reactions) {
-        reactions.push({
-            "name": reaction
-            ,"users": this.reactions[reaction]
-        });
-    }
-    return {
-        "user": this.userId
-        ,"username": this.username
-        ,"ts": this.id
-        ,"text": this.text
-        ,"attachments": this.attachments
-        ,"is_starred": this.is_starred || undefined
-        ,"edited": this.edited
-        ,"removed": this.removed
-        ,"reactions": reactions
-        ,"isMeMessage": this.isMeMessage
-        ,"isNotice": this.notice ? true : undefined
-    };
-}
-
-/**
- * @param {string} reaction
- * @param {string} userId
- * @param {number} ts
-**/
-SlackMessage.prototype.addReaction = function(reaction, userId, ts) {
-    if (!this.reactions[reaction])
-        this.reactions[reaction] = [];
-    this.reactions[reaction].push(userId);
-    this.version = ts;
-}
-
-/**
- * @param {string} reaction
- * @param {string} userId
- * @param {number} ts
-**/
-SlackMessage.prototype.removeReaction = function(reaction, userId, ts) {
-    var updated = false;
-    if (this.reactions[reaction]) {
-        if (this.reactions[reaction].length === 1 && this.reactions[reaction][0] === userId) {
-            delete this.reactions[reaction];
-            updated = true;
-        }
-        else {
-            this.reactions[reaction] = this.reactions[reaction].filter(function(i) {
-                if (i !== userId)
-                    return false;
-                updated = true;
-                return true;
-            });
-        }
-    }
-    if (updated)
-        this.version = ts;
-}
-
-SlackMessage.prototype.hasReactionForUser = function(reaction, userId) {
-    if (this.reactions[reaction])
-        return this.reactions[reaction].indexOf(userId) !== -1;
-    return false;
-}
-
 /**
  * @param {Array} evts
  * @return {number} biggest ts
@@ -300,8 +165,5 @@ SlackHistory.prototype.resort = function() {
     });
 };
 
-/** @suppress {undefinedVars,checkTypes} */
-(function() {
-    if (typeof module !== "undefined") module.exports.SlackHistory = SlackHistory;
-})();
+module.exports.SlackHistory = SlackHistory;