2015-09-18 19:39:16 +02:00
|
|
|
/*
|
2024-09-09 15:57:16 +02:00
|
|
|
Copyright 2024 New Vector Ltd.
|
|
|
|
Copyright 2015-2021 The Matrix.org Foundation C.I.C.
|
2015-09-18 19:39:16 +02:00
|
|
|
|
2024-09-09 15:57:16 +02:00
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
|
|
|
|
Please see LICENSE files in the repository root for full details.
|
2015-09-18 19:39:16 +02:00
|
|
|
*/
|
|
|
|
|
2023-08-04 09:36:16 +02:00
|
|
|
import { Room, EventType, RoomMember, MatrixClient } from "matrix-js-sdk/src/matrix";
|
2021-06-22 18:23:13 +02:00
|
|
|
|
2022-12-12 12:24:14 +01:00
|
|
|
import AliasCustomisations from "./customisations/Alias";
|
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;
|
2020-10-30 19:40:19 +01:00
|
|
|
*
|
|
|
|
* @param {Object} room The room object
|
|
|
|
* @returns {string} A display alias for the given room
|
2016-09-07 12:45:32 +02:00
|
|
|
*/
|
2023-02-13 18:01:43 +01:00
|
|
|
export function getDisplayAliasForRoom(room: Room): string | null {
|
2022-12-12 12:24:14 +01:00
|
|
|
return getDisplayAliasForAliasSet(room.getCanonicalAlias(), room.getAltAliases());
|
2021-07-09 18:04:37 +02:00
|
|
|
}
|
|
|
|
|
2021-07-12 12:56:47 +02:00
|
|
|
// The various display alias getters should all feed through this one path so
|
|
|
|
// there's a single place to change the logic.
|
2023-02-13 18:01:43 +01:00
|
|
|
export function getDisplayAliasForAliasSet(canonicalAlias: string | null, altAliases: string[]): string | null {
|
2021-07-09 18:11:17 +02:00
|
|
|
if (AliasCustomisations.getDisplayAliasForAliasSet) {
|
|
|
|
return AliasCustomisations.getDisplayAliasForAliasSet(canonicalAlias, altAliases);
|
|
|
|
}
|
2022-11-22 16:05:09 +01:00
|
|
|
return (canonicalAlias || altAliases?.[0]) ?? "";
|
2016-09-07 12:45:32 +02:00
|
|
|
}
|
2016-09-05 18:42:22 +02:00
|
|
|
|
2021-06-22 18:23:13 +02:00
|
|
|
export function guessAndSetDMRoom(room: Room, isDirect: boolean): Promise<void> {
|
2017-03-17 12:59:22 +01:00
|
|
|
let newTarget;
|
|
|
|
if (isDirect) {
|
2023-05-30 11:36:34 +02:00
|
|
|
const guessedUserId = guessDMRoomTargetId(room, room.client.getSafeUserId());
|
2018-07-25 14:54:10 +02:00
|
|
|
newTarget = guessedUserId;
|
2017-03-17 12:59:22 +01:00
|
|
|
} else {
|
|
|
|
newTarget = null;
|
|
|
|
}
|
|
|
|
|
2023-06-01 15:43:24 +02:00
|
|
|
return setDMRoom(room.client, 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.
|
2023-06-01 15:43:24 +02:00
|
|
|
* @param client the Matrix Client instance of the logged-in user
|
2016-09-07 18:46:45 +02:00
|
|
|
* @param {string} roomId The ID of the room to modify
|
2022-12-21 17:30:08 +01:00
|
|
|
* @param {string | null} userId The user ID of the desired DM room target user or
|
|
|
|
* null to un-mark this room as a DM room
|
2016-09-07 18:46:45 +02:00
|
|
|
* @returns {object} A promise
|
|
|
|
*/
|
2023-06-01 15:43:24 +02:00
|
|
|
export async function setDMRoom(client: MatrixClient, roomId: string, userId: string | null): Promise<void> {
|
|
|
|
if (client.isGuest()) return;
|
2016-09-16 17:15:25 +02:00
|
|
|
|
2023-06-01 15:43:24 +02:00
|
|
|
const mDirectEvent = client.getAccountData(EventType.Direct);
|
2022-12-21 17:30:08 +01:00
|
|
|
const currentContent = mDirectEvent?.getContent() || {};
|
2016-09-07 18:46:45 +02:00
|
|
|
|
2022-12-21 17:30:08 +01:00
|
|
|
const dmRoomMap = new Map(Object.entries(currentContent));
|
|
|
|
let modified = false;
|
2016-09-07 18:46:45 +02:00
|
|
|
|
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)
|
2022-12-21 17:30:08 +01:00
|
|
|
for (const thisUserId of dmRoomMap.keys()) {
|
|
|
|
const roomList = dmRoomMap.get(thisUserId) || [];
|
2016-09-07 18:46:45 +02:00
|
|
|
|
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);
|
2022-12-21 17:30:08 +01:00
|
|
|
modified = true;
|
2016-09-07 18:46:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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) {
|
2022-12-21 17:30:08 +01:00
|
|
|
const roomList = dmRoomMap.get(userId) || [];
|
2016-09-13 12:06:07 +02:00
|
|
|
if (roomList.indexOf(roomId) == -1) {
|
|
|
|
roomList.push(roomId);
|
2022-12-21 17:30:08 +01:00
|
|
|
modified = true;
|
2016-09-13 12:06:07 +02:00
|
|
|
}
|
2022-12-21 17:30:08 +01:00
|
|
|
dmRoomMap.set(userId, roomList);
|
2016-09-09 20:25:00 +02:00
|
|
|
}
|
|
|
|
|
2022-12-21 17:30:08 +01:00
|
|
|
// prevent unnecessary calls to setAccountData
|
|
|
|
if (!modified) return;
|
|
|
|
|
2023-06-01 15:43:24 +02:00
|
|
|
await client.setAccountData(EventType.Direct, Object.fromEntries(dmRoomMap));
|
2016-09-07 18:46:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
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
|
|
|
*/
|
2021-06-22 18:23:13 +02:00
|
|
|
function guessDMRoomTargetId(room: Room, myUserId: string): string {
|
2023-03-07 14:19:18 +01:00
|
|
|
let oldestTs: number | undefined;
|
|
|
|
let oldestUser: RoomMember | undefined;
|
2016-09-07 18:46:45 +02:00
|
|
|
|
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;
|
2023-02-13 18:01:43 +01:00
|
|
|
oldestTs = user.events.member?.getTs();
|
2017-06-13 18:35:09 +02:00
|
|
|
}
|
|
|
|
}
|
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;
|
2023-02-13 18:01:43 +01:00
|
|
|
oldestTs = user.events.member?.getTs();
|
2016-09-07 18:46:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|