2014-08-15 06:31:13 -04:00
|
|
|
/*
|
2014-09-03 12:29:13 -04:00
|
|
|
Copyright 2014 OpenMarket Ltd
|
2014-08-15 06:31:13 -04:00
|
|
|
|
|
|
|
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', [])
|
2014-10-31 10:26:51 -04:00
|
|
|
.factory('eventHandlerService', ['matrixService', '$rootScope', '$q', '$timeout', 'mPresence', 'notificationService', 'modelService',
|
|
|
|
function(matrixService, $rootScope, $q, $timeout, mPresence, notificationService, modelService) {
|
2014-09-02 07:55:14 -04:00
|
|
|
var ROOM_CREATE_EVENT = "ROOM_CREATE_EVENT";
|
2014-08-15 06:31:13 -04:00
|
|
|
var MSG_EVENT = "MSG_EVENT";
|
|
|
|
var MEMBER_EVENT = "MEMBER_EVENT";
|
|
|
|
var PRESENCE_EVENT = "PRESENCE_EVENT";
|
2014-09-03 08:12:56 -04:00
|
|
|
var POWERLEVEL_EVENT = "POWERLEVEL_EVENT";
|
2014-08-29 08:23:01 -04:00
|
|
|
var CALL_EVENT = "CALL_EVENT";
|
2014-09-03 09:40:54 -04:00
|
|
|
var NAME_EVENT = "NAME_EVENT";
|
2014-09-11 05:31:24 -04:00
|
|
|
var TOPIC_EVENT = "TOPIC_EVENT";
|
2014-09-12 11:43:25 -04:00
|
|
|
var RESET_EVENT = "RESET_EVENT"; // eventHandlerService has been resetted
|
|
|
|
|
2014-09-15 04:39:30 -04:00
|
|
|
// 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 = {};
|
2014-10-29 13:43:18 -04:00
|
|
|
|
2014-09-12 11:43:25 -04:00
|
|
|
var initialSyncDeferred;
|
|
|
|
|
|
|
|
var reset = function() {
|
|
|
|
initialSyncDeferred = $q.defer();
|
|
|
|
|
|
|
|
$rootScope.events = {
|
|
|
|
rooms: {} // will contain roomId: { messages:[], members:{userid1: event} }
|
|
|
|
};
|
2014-08-28 10:22:35 -04:00
|
|
|
|
2014-09-15 04:39:30 -04:00
|
|
|
$rootScope.presence = {};
|
|
|
|
|
|
|
|
eventMap = {};
|
|
|
|
};
|
|
|
|
reset();
|
2014-08-22 05:50:38 -04:00
|
|
|
|
2014-09-18 11:34:13 -04:00
|
|
|
var initRoom = function(room_id, room) {
|
2014-08-15 10:40:37 -04:00
|
|
|
if (!(room_id in $rootScope.events.rooms)) {
|
2014-09-23 15:00:56 -04:00
|
|
|
console.log("Creating new rooms entry for " + room_id);
|
2014-09-15 10:31:59 -04:00
|
|
|
$rootScope.events.rooms[room_id] = {
|
|
|
|
room_id: room_id,
|
|
|
|
messages: [],
|
|
|
|
members: {},
|
|
|
|
// Pagination information
|
|
|
|
pagination: {
|
|
|
|
earliest_token: "END" // how far back we've paginated
|
|
|
|
}
|
2014-09-12 04:12:56 -04:00
|
|
|
};
|
2014-08-15 10:40:37 -04:00
|
|
|
}
|
2014-09-18 11:34:13 -04:00
|
|
|
|
2014-09-23 15:00:56 -04:00
|
|
|
if (room) { // we got an existing room object from initialsync, seemingly.
|
2014-09-18 11:34:13 -04:00
|
|
|
// Report all other metadata of the room object (membership, inviter, visibility, ...)
|
|
|
|
for (var field in room) {
|
2014-09-24 06:22:40 -04:00
|
|
|
if (!room.hasOwnProperty(field)) continue;
|
|
|
|
|
2014-09-23 15:00:56 -04:00
|
|
|
if (-1 === ["room_id", "messages", "state"].indexOf(field)) { // why indexOf - why not ===? --Matthew
|
2014-09-18 11:34:13 -04:00
|
|
|
$rootScope.events.rooms[room_id][field] = room[field];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
$rootScope.events.rooms[room_id].membership = room.membership;
|
|
|
|
}
|
2014-10-31 10:50:31 -04:00
|
|
|
|
|
|
|
// =========================================
|
|
|
|
var __room = modelService.getRoom(room_id);
|
|
|
|
if (room) { // /initialSync data
|
|
|
|
__room.current_room_state.storeStateEvents(room.state);
|
|
|
|
__room.current_room_state.pagination_token = room.messages.end;
|
|
|
|
|
|
|
|
__room.old_room_state.storeStateEvents(room.state);
|
|
|
|
__room.old_room_state.pagination_token = room.messages.start;
|
|
|
|
|
2014-10-31 12:22:15 -04:00
|
|
|
$rootScope["debug_"+room_id] = __room;
|
2014-10-31 10:50:31 -04:00
|
|
|
}
|
2014-09-02 07:55:14 -04:00
|
|
|
};
|
2014-08-22 05:50:10 -04:00
|
|
|
|
2014-08-28 10:22:35 -04:00
|
|
|
var resetRoomMessages = function(room_id) {
|
|
|
|
if ($rootScope.events.rooms[room_id]) {
|
|
|
|
$rootScope.events.rooms[room_id].messages = [];
|
|
|
|
}
|
|
|
|
};
|
2014-08-15 07:51:20 -04:00
|
|
|
|
2014-09-12 04:12:56 -04:00
|
|
|
// Generic method to handle events data
|
|
|
|
var handleRoomDateEvent = function(event, isLiveEvent, addToRoomMessages) {
|
2014-10-31 11:16:43 -04:00
|
|
|
var __room = modelService.getRoom(event.room_id);
|
|
|
|
if (addToRoomMessages) {
|
|
|
|
__room.addMessageEvent(event, !isLiveEvent);
|
|
|
|
}
|
|
|
|
if (isLiveEvent) {
|
|
|
|
__room.current_room_state.storeStateEvent(event);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
var eventTs = event.origin_server_ts;
|
|
|
|
var storedEvent = __room.current_room_state.getStateEvent(event.type, event.state_key);
|
|
|
|
if (storedEvent) {
|
|
|
|
if (storedEvent.origin_server_ts < eventTs) {
|
|
|
|
// the incoming event is newer, use it.
|
|
|
|
__room.current_room_state.storeStateEvent(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// =====================================
|
|
|
|
|
2014-09-12 04:12:56 -04:00
|
|
|
// Add topic changes as if they were a room message
|
|
|
|
if (addToRoomMessages) {
|
|
|
|
if (isLiveEvent) {
|
|
|
|
$rootScope.events.rooms[event.room_id].messages.push(event);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
$rootScope.events.rooms[event.room_id].messages.unshift(event);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// live events always update, but non-live events only update if the
|
|
|
|
// ts is later.
|
|
|
|
var latestData = true;
|
|
|
|
if (!isLiveEvent) {
|
2014-10-17 18:53:24 -04:00
|
|
|
var eventTs = event.origin_server_ts;
|
2014-09-12 04:12:56 -04:00
|
|
|
var storedEvent = $rootScope.events.rooms[event.room_id][event.type];
|
|
|
|
if (storedEvent) {
|
2014-10-17 18:53:24 -04:00
|
|
|
if (storedEvent.origin_server_ts > eventTs) {
|
2014-09-12 04:12:56 -04:00
|
|
|
// ignore it, we have a newer one already.
|
|
|
|
latestData = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (latestData) {
|
|
|
|
$rootScope.events.rooms[event.room_id][event.type] = event;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-09-02 07:55:14 -04:00
|
|
|
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);
|
|
|
|
};
|
|
|
|
|
2014-09-06 03:31:57 -04:00
|
|
|
var handleRoomAliases = function(event, isLiveEvent) {
|
|
|
|
matrixService.createRoomIdToAliasMapping(event.room_id, event.content.aliases[0]);
|
|
|
|
};
|
2014-10-31 11:16:43 -04:00
|
|
|
|
|
|
|
var displayNotification = function(event) {
|
|
|
|
if (window.Notification && event.user_id != matrixService.config().user_id) {
|
|
|
|
var shouldBing = notificationService.containsBingWord(
|
|
|
|
matrixService.config().user_id,
|
|
|
|
matrixService.config().display_name,
|
|
|
|
matrixService.config().bingWords,
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
else if (event.content.msgtype === "m.image") {
|
|
|
|
message = displayname + " sent an image.";
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
notificationService.showNotification(
|
|
|
|
displayname + " (" + roomTitle + ")",
|
|
|
|
message,
|
|
|
|
member ? member.avatar_url : undefined,
|
|
|
|
function() {
|
|
|
|
console.log("notification.onclick() room=" + event.room_id);
|
|
|
|
$rootScope.goToPage('room/' + event.room_id);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2014-09-06 03:31:57 -04:00
|
|
|
|
2014-08-15 06:31:13 -04:00
|
|
|
var handleMessage = function(event, isLiveEvent) {
|
2014-10-29 11:48:41 -04:00
|
|
|
// 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;
|
|
|
|
}
|
2014-10-31 11:16:43 -04:00
|
|
|
|
|
|
|
// =======================
|
|
|
|
|
|
|
|
var __room = modelService.getRoom(event.room_id);
|
|
|
|
|
|
|
|
if (event.user_id !== matrixService.config().user_id) {
|
|
|
|
__room.addMessageEvent(event, !isLiveEvent);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// we may have locally echoed this, so we should replace the event
|
|
|
|
// instead of just adding.
|
|
|
|
__room.addOrReplaceMessageEvent(event, !isLiveEvent);
|
|
|
|
}
|
|
|
|
|
|
|
|
// =======================
|
2014-10-29 11:48:41 -04:00
|
|
|
|
2014-08-15 07:51:20 -04:00
|
|
|
if (isLiveEvent) {
|
2014-09-06 13:13:38 -04:00
|
|
|
if (event.user_id === matrixService.config().user_id &&
|
|
|
|
(event.content.msgtype === "m.text" || event.content.msgtype === "m.emote") ) {
|
2014-09-10 12:24:03 -04:00
|
|
|
// 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);
|
|
|
|
}
|
2014-09-06 03:31:57 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
$rootScope.events.rooms[event.room_id].messages.push(event);
|
2014-10-31 11:16:43 -04:00
|
|
|
displayNotification(event);
|
2014-09-06 03:31:57 -04:00
|
|
|
}
|
2014-09-16 09:32:50 -04:00
|
|
|
|
2014-10-31 11:16:43 -04:00
|
|
|
|
2014-08-15 07:51:20 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
$rootScope.events.rooms[event.room_id].messages.unshift(event);
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO send delivery receipt if isLiveEvent
|
|
|
|
|
2014-08-15 06:31:13 -04:00
|
|
|
// $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);
|
|
|
|
};
|
|
|
|
|
2014-09-10 08:36:30 -04:00
|
|
|
var handleRoomMember = function(event, isLiveEvent, isStateEvent) {
|
2014-10-31 13:13:27 -04:00
|
|
|
var __room = modelService.getRoom(event.room_id);
|
|
|
|
|
2014-09-06 03:31:57 -04:00
|
|
|
|
2014-08-21 14:02:00 -04:00
|
|
|
// add membership changes as if they were a room message if something interesting changed
|
2014-09-10 08:36:30 -04:00
|
|
|
// 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
|
2014-09-18 09:19:35 -04:00
|
|
|
// with other other room messages
|
2014-09-22 12:42:53 -04:00
|
|
|
if (!isStateEvent) {
|
|
|
|
// could be a membership change, display name change, etc.
|
|
|
|
// Find out which one.
|
|
|
|
var memberChanges = undefined;
|
2014-10-27 09:16:59 -04:00
|
|
|
if ((event.prev_content === undefined && event.content.membership) || (event.prev_content && (event.prev_content.membership !== event.content.membership))) {
|
2014-09-22 12:42:53 -04:00
|
|
|
memberChanges = "membership";
|
2014-08-21 14:02:00 -04:00
|
|
|
}
|
2014-09-23 08:01:12 -04:00
|
|
|
else if (event.prev_content && (event.prev_content.displayname !== event.content.displayname)) {
|
2014-09-22 12:42:53 -04:00
|
|
|
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);
|
|
|
|
}
|
2014-10-31 13:13:27 -04:00
|
|
|
// ============
|
|
|
|
|
|
|
|
__room.addMessageEvent(event, !isLiveEvent);
|
2014-08-21 14:02:00 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-09-10 10:26:11 -04:00
|
|
|
// 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;
|
2014-10-31 13:13:27 -04:00
|
|
|
__room.current_room_state.members[event.state_key] = event;
|
2014-09-10 10:26:11 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
$rootScope.$broadcast(MEMBER_EVENT, event, isLiveEvent, isStateEvent);
|
2014-08-15 06:31:13 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
var handlePresence = function(event, isLiveEvent) {
|
2014-08-22 05:50:38 -04:00
|
|
|
$rootScope.presence[event.content.user_id] = event;
|
2014-08-15 06:31:13 -04:00
|
|
|
$rootScope.$broadcast(PRESENCE_EVENT, event, isLiveEvent);
|
|
|
|
};
|
2014-09-02 05:54:11 -04:00
|
|
|
|
|
|
|
var handlePowerLevels = function(event, isLiveEvent) {
|
2014-09-03 09:00:04 -04:00
|
|
|
// 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);
|
|
|
|
}
|
2014-09-02 05:54:11 -04:00
|
|
|
};
|
2014-08-29 08:23:01 -04:00
|
|
|
|
2014-09-12 04:12:56 -04:00
|
|
|
var handleRoomName = function(event, isLiveEvent, isStateEvent) {
|
|
|
|
console.log("handleRoomName room_id: " + event.room_id + " - isLiveEvent: " + isLiveEvent + " - name: " + event.content.name);
|
|
|
|
handleRoomDateEvent(event, isLiveEvent, !isStateEvent);
|
2014-09-03 09:40:54 -04:00
|
|
|
$rootScope.$broadcast(NAME_EVENT, event, isLiveEvent);
|
|
|
|
};
|
2014-09-08 18:36:52 -04:00
|
|
|
|
2014-09-11 05:31:24 -04:00
|
|
|
|
2014-09-12 04:12:56 -04:00
|
|
|
var handleRoomTopic = function(event, isLiveEvent, isStateEvent) {
|
|
|
|
console.log("handleRoomTopic room_id: " + event.room_id + " - isLiveEvent: " + isLiveEvent + " - topic: " + event.content.topic);
|
|
|
|
handleRoomDateEvent(event, isLiveEvent, !isStateEvent);
|
2014-09-11 05:31:24 -04:00
|
|
|
$rootScope.$broadcast(TOPIC_EVENT, event, isLiveEvent);
|
2014-09-08 18:36:52 -04:00
|
|
|
};
|
2014-09-03 09:40:54 -04:00
|
|
|
|
2014-08-29 08:23:01 -04:00
|
|
|
var handleCallEvent = function(event, isLiveEvent) {
|
|
|
|
$rootScope.$broadcast(CALL_EVENT, event, isLiveEvent);
|
2014-09-15 04:39:30 -04:00
|
|
|
if (event.type === 'm.call.invite') {
|
2014-09-09 06:45:36 -04:00
|
|
|
$rootScope.events.rooms[event.room_id].messages.push(event);
|
|
|
|
}
|
2014-08-29 08:23:01 -04:00
|
|
|
};
|
2014-10-29 11:48:41 -04:00
|
|
|
|
|
|
|
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<eventList.length; i++) {
|
|
|
|
if (eventList[i].event_id === event.redacts) {
|
|
|
|
console.log("Removing event " + event.redacts);
|
|
|
|
eventList.splice(i, 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// broadcast the redaction so controllers can nuke this
|
|
|
|
console.log("Redacted an event.");
|
|
|
|
}
|
|
|
|
}
|
2014-08-15 06:31:13 -04:00
|
|
|
|
2014-09-10 12:24:03 -04:00
|
|
|
/**
|
|
|
|
* Get the index of the event in $rootScope.events.rooms[room_id].messages
|
|
|
|
* @param {type} room_id the room id
|
|
|
|
* @param {type} event_id the event id to look for
|
|
|
|
* @returns {Number | undefined} the index. undefined if not found.
|
|
|
|
*/
|
|
|
|
var getRoomEventIndex = function(room_id, event_id) {
|
|
|
|
var index;
|
2014-09-11 03:11:24 -04:00
|
|
|
|
2014-09-10 12:24:03 -04:00
|
|
|
var room = $rootScope.events.rooms[room_id];
|
|
|
|
if (room) {
|
2014-09-11 03:11:24 -04:00
|
|
|
// Start looking from the tail since the first goal of this function
|
|
|
|
// is to find a messaged among the latest ones
|
|
|
|
for (var i = room.messages.length - 1; i > 0; i--) {
|
2014-09-10 12:24:03 -04:00
|
|
|
var message = room.messages[i];
|
|
|
|
if (event_id === message.event_id) {
|
|
|
|
index = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return index;
|
2014-09-15 04:39:30 -04:00
|
|
|
};
|
2014-09-10 12:24:03 -04:00
|
|
|
|
2014-09-24 10:52:48 -04:00
|
|
|
/**
|
|
|
|
* 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);
|
2014-09-24 11:42:21 -04:00
|
|
|
if (member && member.content.displayname) { // Do not consider null displayname
|
2014-09-24 10:52:48 -04:00
|
|
|
displayName = member.content.displayname;
|
2014-09-25 05:45:01 -04:00
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-09-24 10:52:48 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
};
|
|
|
|
|
2014-08-15 06:31:13 -04:00
|
|
|
return {
|
2014-09-02 07:55:14 -04:00
|
|
|
ROOM_CREATE_EVENT: ROOM_CREATE_EVENT,
|
2014-08-15 06:31:13 -04:00
|
|
|
MSG_EVENT: MSG_EVENT,
|
|
|
|
MEMBER_EVENT: MEMBER_EVENT,
|
|
|
|
PRESENCE_EVENT: PRESENCE_EVENT,
|
2014-09-03 08:12:56 -04:00
|
|
|
POWERLEVEL_EVENT: POWERLEVEL_EVENT,
|
2014-08-29 08:23:01 -04:00
|
|
|
CALL_EVENT: CALL_EVENT,
|
2014-09-03 09:40:54 -04:00
|
|
|
NAME_EVENT: NAME_EVENT,
|
2014-09-11 05:31:24 -04:00
|
|
|
TOPIC_EVENT: TOPIC_EVENT,
|
2014-09-12 11:43:25 -04:00
|
|
|
RESET_EVENT: RESET_EVENT,
|
|
|
|
|
|
|
|
reset: function() {
|
|
|
|
reset();
|
|
|
|
$rootScope.$broadcast(RESET_EVENT);
|
|
|
|
},
|
2014-09-18 11:34:13 -04:00
|
|
|
|
|
|
|
initRoom: function(room) {
|
|
|
|
initRoom(room.room_id, room);
|
|
|
|
},
|
2014-08-15 06:31:13 -04:00
|
|
|
|
2014-09-10 08:36:30 -04:00
|
|
|
handleEvent: function(event, isLiveEvent, isStateEvent) {
|
2014-09-12 02:54:18 -04:00
|
|
|
|
|
|
|
// 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
|
2014-09-15 10:31:59 -04:00
|
|
|
if (event.room_id) {
|
|
|
|
initRoom(event.room_id);
|
|
|
|
}
|
2014-09-12 02:54:18 -04:00
|
|
|
|
2014-09-10 06:01:00 -04:00
|
|
|
// 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)
|
2014-09-10 08:45:32 -04:00
|
|
|
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;
|
|
|
|
}
|
2014-09-06 13:13:38 -04:00
|
|
|
}
|
2014-09-10 08:45:32 -04:00
|
|
|
|
2014-09-02 07:55:14 -04:00
|
|
|
if (event.type.indexOf('m.call.') === 0) {
|
2014-08-29 08:23:01 -04:00
|
|
|
handleCallEvent(event, isLiveEvent);
|
|
|
|
}
|
2014-09-06 03:36:55 -04:00
|
|
|
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":
|
2014-09-10 08:36:30 -04:00
|
|
|
handleRoomMember(event, isLiveEvent, isStateEvent);
|
2014-09-06 03:36:55 -04:00
|
|
|
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':
|
2014-09-12 04:12:56 -04:00
|
|
|
handleRoomName(event, isLiveEvent, isStateEvent);
|
2014-09-06 03:36:55 -04:00
|
|
|
break;
|
2014-09-08 18:36:52 -04:00
|
|
|
case 'm.room.topic':
|
2014-09-11 05:31:24 -04:00
|
|
|
handleRoomTopic(event, isLiveEvent, isStateEvent);
|
2014-09-08 18:36:52 -04:00
|
|
|
break;
|
2014-10-29 11:48:41 -04:00
|
|
|
case 'm.room.redaction':
|
|
|
|
handleRedaction(event, isLiveEvent);
|
|
|
|
break;
|
2014-09-06 03:36:55 -04:00
|
|
|
default:
|
2014-10-30 07:22:47 -04:00
|
|
|
// if it is a state event, then just add it in so it
|
|
|
|
// displays on the Room Info screen.
|
|
|
|
if (typeof(event.state_key) === "string") { // incls. 0-len strings
|
|
|
|
if (event.room_id) {
|
|
|
|
handleRoomDateEvent(event, isLiveEvent, false);
|
|
|
|
}
|
|
|
|
}
|
2014-09-06 03:36:55 -04:00
|
|
|
console.log("Unable to handle event type " + event.type);
|
|
|
|
console.log(JSON.stringify(event, undefined, 4));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-08-15 06:31:13 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
// isLiveEvents determines whether notifications should be shown, whether
|
|
|
|
// messages get appended to the start/end of lists, etc.
|
2014-09-10 08:36:30 -04:00
|
|
|
handleEvents: function(events, isLiveEvents, isStateEvents) {
|
2014-08-15 06:31:13 -04:00
|
|
|
for (var i=0; i<events.length; i++) {
|
2014-09-10 08:36:30 -04:00
|
|
|
this.handleEvent(events[i], isLiveEvents, isStateEvents);
|
2014-08-15 06:31:13 -04:00
|
|
|
}
|
2014-08-22 05:50:10 -04:00
|
|
|
},
|
|
|
|
|
2014-09-10 06:01:00 -04:00
|
|
|
// Handle messages from /initialSync or /messages
|
2014-09-16 09:03:07 -04:00
|
|
|
handleRoomMessages: function(room_id, messages, isLiveEvents, dir) {
|
2014-09-11 12:40:38 -04:00
|
|
|
initRoom(room_id);
|
2014-09-10 06:01:00 -04:00
|
|
|
|
2014-09-16 09:03:07 -04:00
|
|
|
var events = messages.chunk;
|
|
|
|
|
|
|
|
// Handles messages according to their time order
|
|
|
|
if (dir && 'b' === dir) {
|
|
|
|
// paginateBackMessages requests messages to be in reverse chronological order
|
|
|
|
for (var i=0; i<events.length; i++) {
|
|
|
|
this.handleEvent(events[i], isLiveEvents, isLiveEvents);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Store how far back we've paginated
|
|
|
|
$rootScope.events.rooms[room_id].pagination.earliest_token = messages.end;
|
2014-10-31 12:22:15 -04:00
|
|
|
|
|
|
|
var __room = modelService.getRoom(room_id);
|
|
|
|
__room.old_room_state.pagination_token = messages.end;
|
|
|
|
|
2014-09-16 09:03:07 -04:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
// InitialSync returns messages in chronological order
|
|
|
|
for (var i=events.length - 1; i>=0; i--) {
|
|
|
|
this.handleEvent(events[i], isLiveEvents, isLiveEvents);
|
|
|
|
}
|
|
|
|
// Store where to start pagination
|
|
|
|
$rootScope.events.rooms[room_id].pagination.earliest_token = messages.start;
|
2014-10-31 12:22:15 -04:00
|
|
|
|
|
|
|
var __room = modelService.getRoom(room_id);
|
|
|
|
__room.old_room_state.pagination_token = messages.start;
|
2014-09-16 09:03:07 -04:00
|
|
|
}
|
2014-09-10 06:01:00 -04:00
|
|
|
},
|
|
|
|
|
2014-11-03 04:44:20 -05:00
|
|
|
handleInitialSyncDone: function(response) {
|
2014-08-28 10:22:35 -04:00
|
|
|
console.log("# handleInitialSyncDone");
|
2014-11-03 04:44:20 -05:00
|
|
|
|
|
|
|
var rooms = response.data.rooms;
|
|
|
|
for (var i = 0; i < rooms.length; ++i) {
|
|
|
|
var room = rooms[i];
|
|
|
|
|
|
|
|
this.initRoom(room);
|
|
|
|
|
|
|
|
if ("messages" in room) {
|
|
|
|
this.handleRoomMessages(room.room_id, room.messages, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ("state" in room) {
|
|
|
|
this.handleEvents(room.state, false, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
var presence = response.data.presence;
|
|
|
|
this.handleEvents(presence, false);
|
|
|
|
|
|
|
|
initialSyncDeferred.resolve(response);
|
2014-08-22 05:50:10 -04:00
|
|
|
},
|
2014-08-28 10:22:35 -04:00
|
|
|
|
|
|
|
// Returns a promise that resolves when the initialSync request has been processed
|
|
|
|
waitForInitialSyncCompletion: function() {
|
2014-09-08 12:13:22 -04:00
|
|
|
return initialSyncDeferred.promise;
|
2014-08-28 10:22:35 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
resetRoomMessages: function(room_id) {
|
|
|
|
resetRoomMessages(room_id);
|
2014-09-15 10:31:59 -04:00
|
|
|
},
|
|
|
|
|
2014-09-15 11:31:07 -04:00
|
|
|
/**
|
|
|
|
* 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;
|
|
|
|
},
|
|
|
|
|
2014-09-15 10:31:59 -04:00
|
|
|
/**
|
|
|
|
* 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) {
|
2014-09-24 06:22:40 -04:00
|
|
|
if (!room.members.hasOwnProperty(i)) continue;
|
|
|
|
|
2014-09-15 10:31:59 -04:00
|
|
|
var member = room.members[i];
|
|
|
|
|
|
|
|
if ("join" === member.membership) {
|
|
|
|
memberCount = memberCount + 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return memberCount;
|
2014-09-15 12:22:38 -04:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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) {
|
2014-09-24 10:52:48 -04:00
|
|
|
return getMember(room_id, user_id);
|
2014-09-17 11:29:21 -04:00
|
|
|
},
|
|
|
|
|
2014-11-03 05:23:14 -05:00
|
|
|
/**
|
|
|
|
* Return the power level of an user in a particular room
|
|
|
|
* @param {String} room_id the room id
|
|
|
|
* @param {String} user_id the user id
|
|
|
|
* @returns {Number} a value between 0 and 10
|
|
|
|
*/
|
|
|
|
getUserPowerLevel: function(room_id, user_id) {
|
|
|
|
var powerLevel = 0;
|
|
|
|
var room = $rootScope.events.rooms[room_id];
|
|
|
|
if (room && room["m.room.power_levels"]) {
|
|
|
|
if (user_id in room["m.room.power_levels"].content) {
|
|
|
|
powerLevel = room["m.room.power_levels"].content[user_id];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Use the room default user power
|
|
|
|
powerLevel = room["m.room.power_levels"].content["default"];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return powerLevel;
|
|
|
|
},
|
|
|
|
|
2014-09-24 05:04:27 -04:00
|
|
|
/**
|
|
|
|
* 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) {
|
2014-09-24 10:52:48 -04:00
|
|
|
return getUserDisplayName(room_id, user_id);
|
2014-09-24 05:04:27 -04:00
|
|
|
},
|
|
|
|
|
2014-09-17 11:29:21 -04:00
|
|
|
setRoomVisibility: function(room_id, visible) {
|
|
|
|
if (!visible) {
|
|
|
|
return;
|
|
|
|
}
|
2014-09-17 11:38:32 -04:00
|
|
|
initRoom(room_id);
|
2014-09-17 11:29:21 -04:00
|
|
|
|
|
|
|
var room = $rootScope.events.rooms[room_id];
|
|
|
|
if (room) {
|
|
|
|
room.visibility = visible;
|
|
|
|
}
|
2014-08-28 10:22:35 -04:00
|
|
|
}
|
2014-08-15 06:31:13 -04:00
|
|
|
};
|
|
|
|
}]);
|