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.
|
|
|
|
*/
|
|
|
|
|
2019-12-20 22:13:46 +01:00
|
|
|
import {MatrixClientPeg} from './MatrixClientPeg';
|
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) {
|
2020-02-21 13:01:36 +01:00
|
|
|
return room.getCanonicalAlias() || room.getAltAliases()[0];
|
2016-09-07 12:45:32 +02:00
|
|
|
}
|
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.
|
|
|
|
*/
|
2018-07-25 14:54:10 +02:00
|
|
|
export function getOnlyOtherMember(room, myUserId) {
|
|
|
|
if (room.currentState.getJoinedMemberCount() === 2) {
|
|
|
|
return room.getJoinedMembers().filter(function(m) {
|
|
|
|
return m.userId !== myUserId;
|
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-07-25 14:54:10 +02:00
|
|
|
function _isConfCallRoom(room, myUserId, conferenceHandler) {
|
2016-09-07 12:45:32 +02:00
|
|
|
if (!conferenceHandler) return false;
|
2016-09-05 19:35:32 +02:00
|
|
|
|
2018-08-14 11:47:05 +02:00
|
|
|
const myMembership = room.getMyMembership();
|
2018-07-25 14:54:10 +02:00
|
|
|
if (myMembership != "join") {
|
2016-09-07 12:45:32 +02:00
|
|
|
return false;
|
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2018-07-25 14:54:10 +02:00
|
|
|
const otherMember = getOnlyOtherMember(room, myUserId);
|
2018-08-17 12:23:58 +02:00
|
|
|
if (!otherMember) {
|
2016-09-07 12:45:32 +02:00
|
|
|
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
|
|
|
|
};
|
|
|
|
|
2018-07-25 14:54:10 +02:00
|
|
|
export function isConfCallRoom(room, myUserId, conferenceHandler) {
|
2018-01-05 15:53:04 +01:00
|
|
|
if (isConfCallRoomCache[room.roomId] !== undefined) {
|
|
|
|
return isConfCallRoomCache[room.roomId];
|
|
|
|
}
|
|
|
|
|
2018-07-25 14:54:10 +02:00
|
|
|
const result = _isConfCallRoom(room, myUserId, conferenceHandler);
|
2018-01-05 15:53:04 +01:00
|
|
|
|
|
|
|
isConfCallRoomCache[room.roomId] = result;
|
|
|
|
|
|
|
|
return result;
|
2016-09-07 12:45:32 +02:00
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2018-07-25 14:54:10 +02:00
|
|
|
export function looksLikeDirectMessageRoom(room, myUserId) {
|
2018-08-14 11:47:05 +02:00
|
|
|
const myMembership = room.getMyMembership();
|
2018-07-25 14:54:10 +02:00
|
|
|
const me = room.getMember(myUserId);
|
|
|
|
|
|
|
|
if (myMembership == "join" || myMembership === "ban" || (me && me.isKicked())) {
|
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
|
|
|
|
// Show 1:1 chats in seperate "Direct Messages" section as long as they haven't
|
|
|
|
// been moved to a different tag section
|
2018-08-14 11:43:03 +02:00
|
|
|
const totalMemberCount = room.currentState.getJoinedMemberCount() +
|
|
|
|
room.currentState.getInvitedMemberCount();
|
|
|
|
if (totalMemberCount === 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) {
|
2018-07-25 14:54:10 +02:00
|
|
|
const guessedUserId = guessDMRoomTargetId(
|
2018-10-27 05:50:35 +02:00
|
|
|
room, MatrixClientPeg.get().getUserId(),
|
2017-03-17 12:59:22 +01:00
|
|
|
);
|
2018-07-25 14:54:10 +02:00
|
|
|
newTarget = guessedUserId;
|
2017-03-17 12:59:22 +01:00
|
|
|
} 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.
|
2019-01-21 17:02:25 +01:00
|
|
|
*
|
|
|
|
* @param {Object} room Target room
|
|
|
|
* @param {string} myUserId User ID of the current user
|
|
|
|
* @returns {string} User ID of the user that the room is probably a DM with
|
2016-09-07 18:46:45 +02:00
|
|
|
*/
|
2018-07-25 14:54:10 +02:00
|
|
|
function guessDMRoomTargetId(room, myUserId) {
|
2016-09-07 18:46:45 +02:00
|
|
|
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()) {
|
2018-07-25 14:54:10 +02:00
|
|
|
if (user.userId == myUserId) continue;
|
2017-06-13 18:35:09 +02:00
|
|
|
|
2018-07-13 15:46:46 +02:00
|
|
|
if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
|
2017-06-13 18:35:09 +02:00
|
|
|
oldestUser = user;
|
|
|
|
oldestTs = user.events.member.getTs();
|
|
|
|
}
|
|
|
|
}
|
2019-01-21 17:02:25 +01:00
|
|
|
if (oldestUser) return oldestUser.userId;
|
2017-06-13 18:35:09 +02:00
|
|
|
|
|
|
|
// 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()) {
|
2018-07-25 14:54:10 +02:00
|
|
|
if (user.userId == myUserId) continue;
|
2016-09-07 18:46:45 +02:00
|
|
|
|
2018-07-13 15:46:46 +02:00
|
|
|
if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
|
2016-09-07 18:46:45 +02:00
|
|
|
oldestUser = user;
|
|
|
|
oldestTs = user.events.member.getTs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 14:54:10 +02:00
|
|
|
if (oldestUser === undefined) return myUserId;
|
2019-01-21 17:02:25 +01:00
|
|
|
return oldestUser.userId;
|
2016-09-07 18:46:45 +02:00
|
|
|
}
|