2015-09-18 19:39:16 +02:00
|
|
|
/*
|
2016-01-07 05:06:39 +01:00
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2015-09-18 19:39:16 +02: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.
|
|
|
|
*/
|
|
|
|
|
2016-09-07 18:46:45 +02:00
|
|
|
import MatrixClientPeg from './MatrixClientPeg';
|
2017-07-12 14:58:14 +02:00
|
|
|
import Promise from 'bluebird';
|
2016-08-11 18:32:39 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
/**
|
|
|
|
* Given a room object, return the alias we should use for it,
|
|
|
|
* if any. This could be the canonical alias if one exists, otherwise
|
|
|
|
* an alias selected arbitrarily but deterministically from the list
|
|
|
|
* of aliases. Otherwise return null;
|
|
|
|
*/
|
|
|
|
export function getDisplayAliasForRoom(room) {
|
|
|
|
return room.getCanonicalAlias() || room.getAliases()[0];
|
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
/**
|
|
|
|
* If the room contains only two members including the logged-in user,
|
|
|
|
* return the other one. Otherwise, return null.
|
|
|
|
*/
|
|
|
|
export function getOnlyOtherMember(room, me) {
|
|
|
|
const joinedMembers = room.getJoinedMembers();
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
if (joinedMembers.length === 2) {
|
|
|
|
return joinedMembers.filter(function(m) {
|
2017-01-20 15:22:27 +01:00
|
|
|
return m.userId !== me.userId;
|
2016-09-07 12:45:32 +02:00
|
|
|
})[0];
|
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
return null;
|
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2018-01-05 15:53:04 +01:00
|
|
|
function _isConfCallRoom(room, me, conferenceHandler) {
|
2016-09-07 12:45:32 +02:00
|
|
|
if (!conferenceHandler) return false;
|
2016-09-05 19:35:32 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
if (me.membership != "join") {
|
|
|
|
return false;
|
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
const otherMember = getOnlyOtherMember(room, me);
|
|
|
|
if (otherMember === null) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (conferenceHandler.isConferenceUser(otherMember.userId)) {
|
|
|
|
return true;
|
|
|
|
}
|
2018-01-05 15:53:04 +01:00
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cache whether a room is a conference call. Assumes that rooms will always
|
|
|
|
// either will or will not be a conference call room.
|
|
|
|
const isConfCallRoomCache = {
|
|
|
|
// $roomId: bool
|
|
|
|
};
|
|
|
|
|
|
|
|
export function isConfCallRoom(room, me, conferenceHandler) {
|
|
|
|
if (isConfCallRoomCache[room.roomId] !== undefined) {
|
|
|
|
return isConfCallRoomCache[room.roomId];
|
|
|
|
}
|
|
|
|
|
|
|
|
const result = _isConfCallRoom(room, me, conferenceHandler);
|
|
|
|
|
|
|
|
isConfCallRoomCache[room.roomId] = result;
|
|
|
|
|
|
|
|
return result;
|
2016-09-07 12:45:32 +02:00
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
export function looksLikeDirectMessageRoom(room, me) {
|
|
|
|
if (me.membership == "join" || me.membership === "ban" ||
|
2017-10-11 18:56:17 +02:00
|
|
|
(me.membership === "leave" && me.events.member.getSender() !== me.events.member.getStateKey())) {
|
2016-09-07 12:45:32 +02:00
|
|
|
// Used to split rooms via tags
|
|
|
|
const tagNames = Object.keys(room.tags);
|
|
|
|
// Used for 1:1 direct chats
|
2016-09-21 10:57:17 +02:00
|
|
|
const members = room.currentState.getMembers();
|
2016-08-11 18:32:39 +02:00
|
|
|
|
2016-09-07 12:45:32 +02:00
|
|
|
// Show 1:1 chats in seperate "Direct Messages" section as long as they haven't
|
|
|
|
// been moved to a different tag section
|
2016-09-21 10:57:17 +02:00
|
|
|
if (members.length === 2 && !tagNames.length) {
|
2016-09-07 12:45:32 +02:00
|
|
|
return true;
|
2016-08-11 18:32:39 +02:00
|
|
|
}
|
2016-09-07 12:45:32 +02:00
|
|
|
}
|
|
|
|
return false;
|
2015-09-18 19:39:16 +02:00
|
|
|
}
|
2016-09-07 18:46:45 +02:00
|
|
|
|
2017-03-17 12:59:22 +01:00
|
|
|
export function guessAndSetDMRoom(room, isDirect) {
|
|
|
|
let newTarget;
|
|
|
|
if (isDirect) {
|
|
|
|
const guessedTarget = guessDMRoomTarget(
|
|
|
|
room, room.getMember(MatrixClientPeg.get().credentials.userId),
|
|
|
|
);
|
|
|
|
newTarget = guessedTarget.userId;
|
|
|
|
} else {
|
|
|
|
newTarget = null;
|
|
|
|
}
|
|
|
|
|
2017-03-27 10:42:04 +02:00
|
|
|
return setDMRoom(room.roomId, newTarget);
|
2017-03-17 12:59:22 +01:00
|
|
|
}
|
|
|
|
|
2016-09-07 18:46:45 +02:00
|
|
|
/**
|
|
|
|
* Marks or unmarks the given room as being as a DM room.
|
|
|
|
* @param {string} roomId The ID of the room to modify
|
|
|
|
* @param {string} userId The user ID of the desired DM
|
|
|
|
room target user or null to un-mark
|
|
|
|
this room as a DM room
|
|
|
|
* @returns {object} A promise
|
|
|
|
*/
|
|
|
|
export function setDMRoom(roomId, userId) {
|
2016-09-16 17:15:25 +02:00
|
|
|
if (MatrixClientPeg.get().isGuest()) {
|
2017-07-12 15:02:00 +02:00
|
|
|
return Promise.resolve();
|
2016-09-16 17:15:25 +02:00
|
|
|
}
|
|
|
|
|
2016-09-07 18:46:45 +02:00
|
|
|
const mDirectEvent = MatrixClientPeg.get().getAccountData('m.direct');
|
|
|
|
let dmRoomMap = {};
|
|
|
|
|
|
|
|
if (mDirectEvent !== undefined) dmRoomMap = mDirectEvent.getContent();
|
|
|
|
|
2016-09-09 20:25:00 +02:00
|
|
|
// remove it from the lists of any others users
|
|
|
|
// (it can only be a DM room for one person)
|
2016-09-07 18:46:45 +02:00
|
|
|
for (const thisUserId of Object.keys(dmRoomMap)) {
|
|
|
|
const roomList = dmRoomMap[thisUserId];
|
|
|
|
|
2016-09-09 20:25:00 +02:00
|
|
|
if (thisUserId != userId) {
|
2016-09-07 18:46:45 +02:00
|
|
|
const indexOfRoom = roomList.indexOf(roomId);
|
|
|
|
if (indexOfRoom > -1) {
|
|
|
|
roomList.splice(indexOfRoom, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-09 20:25:00 +02:00
|
|
|
// now add it, if it's not already there
|
2016-09-13 12:06:07 +02:00
|
|
|
if (userId) {
|
|
|
|
const roomList = dmRoomMap[userId] || [];
|
|
|
|
if (roomList.indexOf(roomId) == -1) {
|
|
|
|
roomList.push(roomId);
|
|
|
|
}
|
|
|
|
dmRoomMap[userId] = roomList;
|
2016-09-09 20:25:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-07 18:46:45 +02:00
|
|
|
return MatrixClientPeg.get().setAccountData('m.direct', dmRoomMap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-09-08 14:56:45 +02:00
|
|
|
* Given a room, estimate which of its members is likely to
|
2016-09-07 18:46:45 +02:00
|
|
|
* be the target if the room were a DM room and return that user.
|
|
|
|
*/
|
|
|
|
export function guessDMRoomTarget(room, me) {
|
|
|
|
let oldestTs;
|
|
|
|
let oldestUser;
|
|
|
|
|
2017-06-13 18:35:09 +02:00
|
|
|
// Pick the joined user who's been here longest (and isn't us),
|
|
|
|
for (const user of room.getJoinedMembers()) {
|
|
|
|
if (user.userId == me.userId) continue;
|
|
|
|
|
|
|
|
if (oldestTs === undefined || user.events.member.getTs() < oldestTs) {
|
|
|
|
oldestUser = user;
|
|
|
|
oldestTs = user.events.member.getTs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (oldestUser) return oldestUser;
|
|
|
|
|
|
|
|
// if there are no joined members other than us, use the oldest member
|
2016-09-07 18:46:45 +02:00
|
|
|
for (const user of room.currentState.getMembers()) {
|
|
|
|
if (user.userId == me.userId) continue;
|
|
|
|
|
|
|
|
if (oldestTs === undefined || user.events.member.getTs() < oldestTs) {
|
|
|
|
oldestUser = user;
|
|
|
|
oldestTs = user.events.member.getTs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (oldestUser === undefined) return me;
|
|
|
|
return oldestUser;
|
|
|
|
}
|