2015-09-19 01:39:16 +08:00
|
|
|
/*
|
2016-01-07 12:06:39 +08:00
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2015-09-19 01:39:16 +08: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-08 00:46:45 +08:00
|
|
|
import MatrixClientPeg from './MatrixClientPeg';
|
2016-08-12 00:32:39 +08:00
|
|
|
|
2016-09-07 18:45:32 +08: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-06 00:42:22 +08:00
|
|
|
|
2016-09-07 18:45:32 +08:00
|
|
|
/**
|
|
|
|
* If the room contains only two members including the logged-in user,
|
|
|
|
* return the other one. Otherwise, return null.
|
|
|
|
*/
|
2018-07-25 20:54:10 +08:00
|
|
|
export function getOnlyOtherMember(room, myUserId) {
|
|
|
|
if (room.currentState.getJoinedMemberCount() === 2) {
|
|
|
|
return room.getJoinedMembers().filter(function(m) {
|
|
|
|
return m.userId !== myUserId;
|
2016-09-07 18:45:32 +08:00
|
|
|
})[0];
|
|
|
|
}
|
2016-09-06 00:42:22 +08:00
|
|
|
|
2016-09-07 18:45:32 +08:00
|
|
|
return null;
|
|
|
|
}
|
2016-09-06 00:42:22 +08:00
|
|
|
|
2018-07-25 20:54:10 +08:00
|
|
|
function _isConfCallRoom(room, myUserId, conferenceHandler) {
|
2016-09-07 18:45:32 +08:00
|
|
|
if (!conferenceHandler) return false;
|
2016-09-06 01:35:32 +08:00
|
|
|
|
2018-08-14 17:47:05 +08:00
|
|
|
const myMembership = room.getMyMembership();
|
2018-07-25 20:54:10 +08:00
|
|
|
if (myMembership != "join") {
|
2016-09-07 18:45:32 +08:00
|
|
|
return false;
|
|
|
|
}
|
2016-09-06 00:42:22 +08:00
|
|
|
|
2018-07-25 20:54:10 +08:00
|
|
|
const otherMember = getOnlyOtherMember(room, myUserId);
|
2018-08-17 18:23:58 +08:00
|
|
|
if (!otherMember) {
|
2016-09-07 18:45:32 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (conferenceHandler.isConferenceUser(otherMember.userId)) {
|
|
|
|
return true;
|
|
|
|
}
|
2018-01-05 22:53:04 +08: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 20:54:10 +08:00
|
|
|
export function isConfCallRoom(room, myUserId, conferenceHandler) {
|
2018-01-05 22:53:04 +08:00
|
|
|
if (isConfCallRoomCache[room.roomId] !== undefined) {
|
|
|
|
return isConfCallRoomCache[room.roomId];
|
|
|
|
}
|
|
|
|
|
2018-07-25 20:54:10 +08:00
|
|
|
const result = _isConfCallRoom(room, myUserId, conferenceHandler);
|
2018-01-05 22:53:04 +08:00
|
|
|
|
|
|
|
isConfCallRoomCache[room.roomId] = result;
|
|
|
|
|
|
|
|
return result;
|
2016-09-07 18:45:32 +08:00
|
|
|
}
|
2016-09-06 00:42:22 +08:00
|
|
|
|
2018-07-25 20:54:10 +08:00
|
|
|
export function looksLikeDirectMessageRoom(room, myUserId) {
|
2018-08-14 17:47:05 +08:00
|
|
|
const myMembership = room.getMyMembership();
|
2018-07-25 20:54:10 +08:00
|
|
|
const me = room.getMember(myUserId);
|
|
|
|
|
|
|
|
if (myMembership == "join" || myMembership === "ban" || (me && me.isKicked())) {
|
2016-09-07 18:45:32 +08: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 17:43:03 +08:00
|
|
|
const totalMemberCount = room.currentState.getJoinedMemberCount() +
|
|
|
|
room.currentState.getInvitedMemberCount();
|
|
|
|
if (totalMemberCount === 2 && !tagNames.length) {
|
2016-09-07 18:45:32 +08:00
|
|
|
return true;
|
2016-08-12 00:32:39 +08:00
|
|
|
}
|
2016-09-07 18:45:32 +08:00
|
|
|
}
|
|
|
|
return false;
|
2015-09-19 01:39:16 +08:00
|
|
|
}
|
2016-09-08 00:46:45 +08:00
|
|
|
|
2017-03-17 19:59:22 +08:00
|
|
|
export function guessAndSetDMRoom(room, isDirect) {
|
|
|
|
let newTarget;
|
|
|
|
if (isDirect) {
|
2018-07-25 20:54:10 +08:00
|
|
|
const guessedUserId = guessDMRoomTargetId(
|
2018-10-27 11:50:35 +08:00
|
|
|
room, MatrixClientPeg.get().getUserId(),
|
2017-03-17 19:59:22 +08:00
|
|
|
);
|
2018-07-25 20:54:10 +08:00
|
|
|
newTarget = guessedUserId;
|
2017-03-17 19:59:22 +08:00
|
|
|
} else {
|
|
|
|
newTarget = null;
|
|
|
|
}
|
|
|
|
|
2017-03-27 16:42:04 +08:00
|
|
|
return setDMRoom(room.roomId, newTarget);
|
2017-03-17 19:59:22 +08:00
|
|
|
}
|
|
|
|
|
2016-09-08 00:46:45 +08: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 23:15:25 +08:00
|
|
|
if (MatrixClientPeg.get().isGuest()) {
|
2017-07-12 21:02:00 +08:00
|
|
|
return Promise.resolve();
|
2016-09-16 23:15:25 +08:00
|
|
|
}
|
|
|
|
|
2016-09-08 00:46:45 +08:00
|
|
|
const mDirectEvent = MatrixClientPeg.get().getAccountData('m.direct');
|
|
|
|
let dmRoomMap = {};
|
|
|
|
|
|
|
|
if (mDirectEvent !== undefined) dmRoomMap = mDirectEvent.getContent();
|
|
|
|
|
2016-09-10 02:25:00 +08:00
|
|
|
// remove it from the lists of any others users
|
|
|
|
// (it can only be a DM room for one person)
|
2016-09-08 00:46:45 +08:00
|
|
|
for (const thisUserId of Object.keys(dmRoomMap)) {
|
|
|
|
const roomList = dmRoomMap[thisUserId];
|
|
|
|
|
2016-09-10 02:25:00 +08:00
|
|
|
if (thisUserId != userId) {
|
2016-09-08 00:46:45 +08:00
|
|
|
const indexOfRoom = roomList.indexOf(roomId);
|
|
|
|
if (indexOfRoom > -1) {
|
|
|
|
roomList.splice(indexOfRoom, 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-10 02:25:00 +08:00
|
|
|
// now add it, if it's not already there
|
2016-09-13 18:06:07 +08:00
|
|
|
if (userId) {
|
|
|
|
const roomList = dmRoomMap[userId] || [];
|
|
|
|
if (roomList.indexOf(roomId) == -1) {
|
|
|
|
roomList.push(roomId);
|
|
|
|
}
|
|
|
|
dmRoomMap[userId] = roomList;
|
2016-09-10 02:25:00 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-08 00:46:45 +08:00
|
|
|
return MatrixClientPeg.get().setAccountData('m.direct', dmRoomMap);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-09-08 20:56:45 +08:00
|
|
|
* Given a room, estimate which of its members is likely to
|
2016-09-08 00:46:45 +08:00
|
|
|
* be the target if the room were a DM room and return that user.
|
2019-01-22 00:02:25 +08: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-08 00:46:45 +08:00
|
|
|
*/
|
2018-07-25 20:54:10 +08:00
|
|
|
function guessDMRoomTargetId(room, myUserId) {
|
2016-09-08 00:46:45 +08:00
|
|
|
let oldestTs;
|
|
|
|
let oldestUser;
|
|
|
|
|
2017-06-14 00:35:09 +08:00
|
|
|
// Pick the joined user who's been here longest (and isn't us),
|
|
|
|
for (const user of room.getJoinedMembers()) {
|
2018-07-25 20:54:10 +08:00
|
|
|
if (user.userId == myUserId) continue;
|
2017-06-14 00:35:09 +08:00
|
|
|
|
2018-07-13 21:46:46 +08:00
|
|
|
if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
|
2017-06-14 00:35:09 +08:00
|
|
|
oldestUser = user;
|
|
|
|
oldestTs = user.events.member.getTs();
|
|
|
|
}
|
|
|
|
}
|
2019-01-22 00:02:25 +08:00
|
|
|
if (oldestUser) return oldestUser.userId;
|
2017-06-14 00:35:09 +08:00
|
|
|
|
|
|
|
// if there are no joined members other than us, use the oldest member
|
2016-09-08 00:46:45 +08:00
|
|
|
for (const user of room.currentState.getMembers()) {
|
2018-07-25 20:54:10 +08:00
|
|
|
if (user.userId == myUserId) continue;
|
2016-09-08 00:46:45 +08:00
|
|
|
|
2018-07-13 21:46:46 +08:00
|
|
|
if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
|
2016-09-08 00:46:45 +08:00
|
|
|
oldestUser = user;
|
|
|
|
oldestTs = user.events.member.getTs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-25 20:54:10 +08:00
|
|
|
if (oldestUser === undefined) return myUserId;
|
2019-01-22 00:02:25 +08:00
|
|
|
return oldestUser.userId;
|
2016-09-08 00:46:45 +08:00
|
|
|
}
|