/* Copyright 2014 OpenMarket Ltd Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ 'use strict'; /* This service handles what should happen when you get an event. This service does not care where the event came from, it only needs enough context to be able to process them. Events may be coming from the event stream, the REST API (via direct GETs or via a pagination stream API), etc. Typically, this service will store events or broadcast them to any listeners (e.g. controllers) via $broadcast. Alternatively, it may update the $rootScope if typically all the $on method would do is update its own $scope. */ angular.module('eventHandlerService', []) .factory('eventHandlerService', ['matrixService', '$rootScope', '$q', '$timeout', 'mPresence', function(matrixService, $rootScope, $q, $timeout, mPresence) { var ROOM_CREATE_EVENT = "ROOM_CREATE_EVENT"; var MSG_EVENT = "MSG_EVENT"; var MEMBER_EVENT = "MEMBER_EVENT"; var PRESENCE_EVENT = "PRESENCE_EVENT"; var POWERLEVEL_EVENT = "POWERLEVEL_EVENT"; var CALL_EVENT = "CALL_EVENT"; var NAME_EVENT = "NAME_EVENT"; var TOPIC_EVENT = "TOPIC_EVENT"; var RESET_EVENT = "RESET_EVENT"; // eventHandlerService has been resetted // used for dedupping events - could be expanded in future... // FIXME: means that we leak memory over time (along with lots of the rest // of the app, given we never try to reap memory yet) var eventMap = {}; $rootScope.presence = {}; // TODO: This is attached to the rootScope so .html can just go containsBingWord // for determining classes so it is easy to highlight bing messages. It seems a // bit strange to put the impl in this service though, but I can't think of a better // file to put it in. $rootScope.containsBingWord = function(content) { if (!content || $.type(content) != "string") { return false; } var bingWords = matrixService.config().bingWords; var shouldBing = false; // case-insensitive name check for user_id OR display_name if they exist var myUserId = matrixService.config().user_id; if (myUserId) { myUserId = myUserId.toLocaleLowerCase(); } var myDisplayName = matrixService.config().display_name; if (myDisplayName) { myDisplayName = myDisplayName.toLocaleLowerCase(); } if ( (myDisplayName && content.toLocaleLowerCase().indexOf(myDisplayName) != -1) || (myUserId && content.toLocaleLowerCase().indexOf(myUserId) != -1) ) { shouldBing = true; } // bing word list check if (bingWords && !shouldBing) { for (var i=0; i eventTs) { // ignore it, we have a newer one already. latestData = false; } } } if (latestData) { $rootScope.events.rooms[event.room_id][event.type] = event; } }; var handleRoomCreate = function(event, isLiveEvent) { // For now, we do not use the event data. Simply signal it to the app controllers $rootScope.$broadcast(ROOM_CREATE_EVENT, event, isLiveEvent); }; var handleRoomAliases = function(event, isLiveEvent) { matrixService.createRoomIdToAliasMapping(event.room_id, event.content.aliases[0]); }; var handleMessage = function(event, isLiveEvent) { // Check for empty event content var hasContent = false; for (var prop in event.content) { hasContent = true; break; } if (!hasContent) { // empty json object is a redacted event, so ignore. return; } if (isLiveEvent) { if (event.user_id === matrixService.config().user_id && (event.content.msgtype === "m.text" || event.content.msgtype === "m.emote") ) { // Assume we've already echoed it. So, there is a fake event in the messages list of the room // Replace this fake event by the true one var index = getRoomEventIndex(event.room_id, event.event_id); if (index) { $rootScope.events.rooms[event.room_id].messages[index] = event; } else { $rootScope.events.rooms[event.room_id].messages.push(event); } } else { $rootScope.events.rooms[event.room_id].messages.push(event); } if (window.Notification && event.user_id != matrixService.config().user_id) { var shouldBing = $rootScope.containsBingWord(event.content.body); // Ideally we would notify only when the window is hidden (i.e. document.hidden = true). // // However, Chrome on Linux and OSX currently returns document.hidden = false unless the window is // explicitly showing a different tab. So we need another metric to determine hiddenness - we // simply use idle time. If the user has been idle enough that their presence goes to idle, then // we also display notifs when things happen. // // This is far far better than notifying whenever anything happens anyway, otherwise you get spammed // to death with notifications when the window is in the foreground, which is horrible UX (especially // if you have not defined any bingers and so get notified for everything). var isIdle = (document.hidden || matrixService.presence.unavailable === mPresence.getState()); // We need a way to let people get notifications for everything, if they so desire. The way to do this // is to specify zero bingwords. var bingWords = matrixService.config().bingWords; if (bingWords === undefined || bingWords.length === 0) { shouldBing = true; } if (shouldBing && isIdle) { console.log("Displaying notification for "+JSON.stringify(event)); var member = getMember(event.room_id, event.user_id); var displayname = getUserDisplayName(event.room_id, event.user_id); var message = event.content.body; if (event.content.msgtype === "m.emote") { message = "* " + displayname + " " + message; } var roomTitle = matrixService.getRoomIdToAliasMapping(event.room_id); var theRoom = $rootScope.events.rooms[event.room_id]; if (!roomTitle && theRoom && theRoom["m.room.name"] && theRoom["m.room.name"].content) { roomTitle = theRoom["m.room.name"].content.name; } if (!roomTitle) { roomTitle = event.room_id; } var notification = new window.Notification( displayname + " (" + roomTitle + ")", { "body": message, "icon": member ? member.avatar_url : undefined }); notification.onclick = function() { console.log("notification.onclick() room=" + event.room_id); $rootScope.goToPage('room/' + (event.room_id)); }; $timeout(function() { notification.close(); }, 5 * 1000); } } } else { $rootScope.events.rooms[event.room_id].messages.unshift(event); } // TODO send delivery receipt if isLiveEvent // $broadcast this, as controllers may want to do funky things such as // scroll to the bottom, etc which cannot be expressed via simple $scope // updates. $rootScope.$broadcast(MSG_EVENT, event, isLiveEvent); }; var handleRoomMember = function(event, isLiveEvent, isStateEvent) { // add membership changes as if they were a room message if something interesting changed // Exception: Do not do this if the event is a room state event because such events already come // as room messages events. Moreover, when they come as room messages events, they are relatively ordered // with other other room messages if (!isStateEvent) { // could be a membership change, display name change, etc. // Find out which one. var memberChanges = undefined; if ((event.prev_content === undefined && event.content.membership) || (event.prev_content && (event.prev_content.membership !== event.content.membership))) { memberChanges = "membership"; } else if (event.prev_content && (event.prev_content.displayname !== event.content.displayname)) { memberChanges = "displayname"; } // mark the key which changed event.changedKey = memberChanges; // If there was a change we want to display, dump it in the message // list. if (memberChanges) { if (isLiveEvent) { $rootScope.events.rooms[event.room_id].messages.push(event); } else { $rootScope.events.rooms[event.room_id].messages.unshift(event); } } } // Use data from state event or the latest data from the stream. // Do not care of events that come when paginating back if (isStateEvent || isLiveEvent) { $rootScope.events.rooms[event.room_id].members[event.state_key] = event; } $rootScope.$broadcast(MEMBER_EVENT, event, isLiveEvent, isStateEvent); }; var handlePresence = function(event, isLiveEvent) { $rootScope.presence[event.content.user_id] = event; $rootScope.$broadcast(PRESENCE_EVENT, event, isLiveEvent); }; var handlePowerLevels = function(event, isLiveEvent) { // Keep the latest data. Do not care of events that come when paginating back if (!$rootScope.events.rooms[event.room_id][event.type] || isLiveEvent) { $rootScope.events.rooms[event.room_id][event.type] = event; $rootScope.$broadcast(POWERLEVEL_EVENT, event, isLiveEvent); } }; var handleRoomName = function(event, isLiveEvent, isStateEvent) { console.log("handleRoomName room_id: " + event.room_id + " - isLiveEvent: " + isLiveEvent + " - name: " + event.content.name); handleRoomDateEvent(event, isLiveEvent, !isStateEvent); $rootScope.$broadcast(NAME_EVENT, event, isLiveEvent); }; var handleRoomTopic = function(event, isLiveEvent, isStateEvent) { console.log("handleRoomTopic room_id: " + event.room_id + " - isLiveEvent: " + isLiveEvent + " - topic: " + event.content.topic); handleRoomDateEvent(event, isLiveEvent, !isStateEvent); $rootScope.$broadcast(TOPIC_EVENT, event, isLiveEvent); }; var handleCallEvent = function(event, isLiveEvent) { $rootScope.$broadcast(CALL_EVENT, event, isLiveEvent); if (event.type === 'm.call.invite') { $rootScope.events.rooms[event.room_id].messages.push(event); } }; var handleRedaction = function(event, isLiveEvent) { if (!isLiveEvent) { // we have nothing to remove, so just ignore it. console.log("Received redacted event: "+JSON.stringify(event)); return; } // we need to remove something possibly: do we know the redacted // event ID? if (eventMap[event.redacts]) { // remove event from list of messages in this room. var eventList = $rootScope.events.rooms[event.room_id].messages; for (var i=0; i 0; i--) { var message = room.messages[i]; if (event_id === message.event_id) { index = i; break; } } } return index; }; /** * Get the member object of a room member * @param {String} room_id the room id * @param {String} user_id the id of the user * @returns {undefined | Object} the member object of this user in this room if he is part of the room */ var getMember = function(room_id, user_id) { var member; var room = $rootScope.events.rooms[room_id]; if (room) { member = room.members[user_id]; } return member; }; /** * Return the display name of an user acccording to data already downloaded * @param {String} room_id the room id * @param {String} user_id the id of the user * @returns {String} the user displayname or user_id if not available */ var getUserDisplayName = function(room_id, user_id) { var displayName; // Get the user display name from the member list of the room var member = getMember(room_id, user_id); if (member && member.content.displayname) { // Do not consider null displayname displayName = member.content.displayname; // Disambiguate users who have the same displayname in the room if (user_id !== matrixService.config().user_id) { var room = $rootScope.events.rooms[room_id]; for (var member_id in room.members) { if (room.members.hasOwnProperty(member_id) && member_id !== user_id) { var member2 = room.members[member_id]; if (member2.content.displayname && member2.content.displayname === displayName) { displayName = displayName + " (" + user_id + ")"; break; } } } } } // The user may not have joined the room yet. So try to resolve display name from presence data // Note: This data may not be available if (undefined === displayName && user_id in $rootScope.presence) { displayName = $rootScope.presence[user_id].content.displayname; } if (undefined === displayName) { // By default, use the user ID displayName = user_id; } return displayName; }; return { ROOM_CREATE_EVENT: ROOM_CREATE_EVENT, MSG_EVENT: MSG_EVENT, MEMBER_EVENT: MEMBER_EVENT, PRESENCE_EVENT: PRESENCE_EVENT, POWERLEVEL_EVENT: POWERLEVEL_EVENT, CALL_EVENT: CALL_EVENT, NAME_EVENT: NAME_EVENT, TOPIC_EVENT: TOPIC_EVENT, RESET_EVENT: RESET_EVENT, reset: function() { reset(); $rootScope.$broadcast(RESET_EVENT); }, initRoom: function(room) { initRoom(room.room_id, room); }, handleEvent: function(event, isLiveEvent, isStateEvent) { // FIXME: /initialSync on a particular room is not yet available // So initRoom on a new room is not called. Make sure the room data is initialised here if (event.room_id) { initRoom(event.room_id); } // Avoid duplicated events // Needed for rooms where initialSync has not been done. // In this case, we do not know where to start pagination. So, it starts from the END // and we can have the same event (ex: joined, invitation) coming from the pagination // AND from the event stream. // FIXME: This workaround should be no more required when /initialSync on a particular room // will be available (as opposite to the global /initialSync done at startup) if (!isStateEvent) { // Do not consider state events if (event.event_id && eventMap[event.event_id]) { console.log("discarding duplicate event: " + JSON.stringify(event, undefined, 4)); return; } else { eventMap[event.event_id] = 1; } } if (event.type.indexOf('m.call.') === 0) { handleCallEvent(event, isLiveEvent); } else { switch(event.type) { case "m.room.create": handleRoomCreate(event, isLiveEvent); break; case "m.room.aliases": handleRoomAliases(event, isLiveEvent); break; case "m.room.message": handleMessage(event, isLiveEvent); break; case "m.room.member": handleRoomMember(event, isLiveEvent, isStateEvent); break; case "m.presence": handlePresence(event, isLiveEvent); break; case 'm.room.ops_levels': case 'm.room.send_event_level': case 'm.room.add_state_level': case 'm.room.join_rules': case 'm.room.power_levels': handlePowerLevels(event, isLiveEvent); break; case 'm.room.name': handleRoomName(event, isLiveEvent, isStateEvent); break; case 'm.room.topic': handleRoomTopic(event, isLiveEvent, isStateEvent); break; case 'm.room.redaction': handleRedaction(event, isLiveEvent); break; default: console.log("Unable to handle event type " + event.type); console.log(JSON.stringify(event, undefined, 4)); break; } } }, // isLiveEvents determines whether notifications should be shown, whether // messages get appended to the start/end of lists, etc. handleEvents: function(events, isLiveEvents, isStateEvents) { for (var i=0; i=0; i--) { this.handleEvent(events[i], isLiveEvents, isLiveEvents); } // Store where to start pagination $rootScope.events.rooms[room_id].pagination.earliest_token = messages.start; } }, handleInitialSyncDone: function(initialSyncData) { console.log("# handleInitialSyncDone"); initialSyncDeferred.resolve(initialSyncData); }, // Returns a promise that resolves when the initialSync request has been processed waitForInitialSyncCompletion: function() { return initialSyncDeferred.promise; }, resetRoomMessages: function(room_id) { resetRoomMessages(room_id); }, /** * Return the last message event of a room * @param {String} room_id the room id * @param {Boolean} filterFake true to not take into account fake messages * @returns {undefined | Event} the last message event if available */ getLastMessage: function(room_id, filterEcho) { var lastMessage; var room = $rootScope.events.rooms[room_id]; if (room) { for (var i = room.messages.length - 1; i >= 0; i--) { var message = room.messages[i]; if (!filterEcho || undefined === message.echo_msg_state) { lastMessage = message; break; } } } return lastMessage; }, /** * Compute the room users number, ie the number of members who has joined the room. * @param {String} room_id the room id * @returns {undefined | Number} the room users number if available */ getUsersCountInRoom: function(room_id) { var memberCount; var room = $rootScope.events.rooms[room_id]; if (room) { memberCount = 0; for (var i in room.members) { if (!room.members.hasOwnProperty(i)) continue; var member = room.members[i]; if ("join" === member.membership) { memberCount = memberCount + 1; } } } return memberCount; }, /** * Get the member object of a room member * @param {String} room_id the room id * @param {String} user_id the id of the user * @returns {undefined | Object} the member object of this user in this room if he is part of the room */ getMember: function(room_id, user_id) { return getMember(room_id, user_id); }, /** * Return the display name of an user acccording to data already downloaded * @param {String} room_id the room id * @param {String} user_id the id of the user * @returns {String} the user displayname or user_id if not available */ getUserDisplayName: function(room_id, user_id) { return getUserDisplayName(room_id, user_id); }, setRoomVisibility: function(room_id, visible) { if (!visible) { return; } initRoom(room_id); var room = $rootScope.events.rooms[room_id]; if (room) { room.visibility = visible; } } }; }]);