138 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
			
		
		
	
	
			138 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			TypeScript
		
	
	
/*
 | 
						|
Copyright 2015 - 2021 The Matrix.org Foundation C.I.C.
 | 
						|
 | 
						|
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.
 | 
						|
*/
 | 
						|
 | 
						|
import { Room, EventType, RoomMember, MatrixClient } from "matrix-js-sdk/src/matrix";
 | 
						|
 | 
						|
import AliasCustomisations from "./customisations/Alias";
 | 
						|
 | 
						|
/**
 | 
						|
 * 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;
 | 
						|
 *
 | 
						|
 * @param {Object} room The room object
 | 
						|
 * @returns {string} A display alias for the given room
 | 
						|
 */
 | 
						|
export function getDisplayAliasForRoom(room: Room): string | null {
 | 
						|
    return getDisplayAliasForAliasSet(room.getCanonicalAlias(), room.getAltAliases());
 | 
						|
}
 | 
						|
 | 
						|
// The various display alias getters should all feed through this one path so
 | 
						|
// there's a single place to change the logic.
 | 
						|
export function getDisplayAliasForAliasSet(canonicalAlias: string | null, altAliases: string[]): string | null {
 | 
						|
    if (AliasCustomisations.getDisplayAliasForAliasSet) {
 | 
						|
        return AliasCustomisations.getDisplayAliasForAliasSet(canonicalAlias, altAliases);
 | 
						|
    }
 | 
						|
    return (canonicalAlias || altAliases?.[0]) ?? "";
 | 
						|
}
 | 
						|
 | 
						|
export function guessAndSetDMRoom(room: Room, isDirect: boolean): Promise<void> {
 | 
						|
    let newTarget;
 | 
						|
    if (isDirect) {
 | 
						|
        const guessedUserId = guessDMRoomTargetId(room, room.client.getSafeUserId());
 | 
						|
        newTarget = guessedUserId;
 | 
						|
    } else {
 | 
						|
        newTarget = null;
 | 
						|
    }
 | 
						|
 | 
						|
    return setDMRoom(room.client, room.roomId, newTarget);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Marks or unmarks the given room as being as a DM room.
 | 
						|
 * @param client the Matrix Client instance of the logged-in user
 | 
						|
 * @param {string} roomId The ID of the room to modify
 | 
						|
 * @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
 | 
						|
 * @returns {object} A promise
 | 
						|
 */
 | 
						|
export async function setDMRoom(client: MatrixClient, roomId: string, userId: string | null): Promise<void> {
 | 
						|
    if (client.isGuest()) return;
 | 
						|
 | 
						|
    const mDirectEvent = client.getAccountData(EventType.Direct);
 | 
						|
    const currentContent = mDirectEvent?.getContent() || {};
 | 
						|
 | 
						|
    const dmRoomMap = new Map(Object.entries(currentContent));
 | 
						|
    let modified = false;
 | 
						|
 | 
						|
    // remove it from the lists of any others users
 | 
						|
    // (it can only be a DM room for one person)
 | 
						|
    for (const thisUserId of dmRoomMap.keys()) {
 | 
						|
        const roomList = dmRoomMap.get(thisUserId) || [];
 | 
						|
 | 
						|
        if (thisUserId != userId) {
 | 
						|
            const indexOfRoom = roomList.indexOf(roomId);
 | 
						|
            if (indexOfRoom > -1) {
 | 
						|
                roomList.splice(indexOfRoom, 1);
 | 
						|
                modified = true;
 | 
						|
            }
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    // now add it, if it's not already there
 | 
						|
    if (userId) {
 | 
						|
        const roomList = dmRoomMap.get(userId) || [];
 | 
						|
        if (roomList.indexOf(roomId) == -1) {
 | 
						|
            roomList.push(roomId);
 | 
						|
            modified = true;
 | 
						|
        }
 | 
						|
        dmRoomMap.set(userId, roomList);
 | 
						|
    }
 | 
						|
 | 
						|
    // prevent unnecessary calls to setAccountData
 | 
						|
    if (!modified) return;
 | 
						|
 | 
						|
    await client.setAccountData(EventType.Direct, Object.fromEntries(dmRoomMap));
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Given a room, estimate which of its members is likely to
 | 
						|
 * be the target if the room were a DM room and return that user.
 | 
						|
 *
 | 
						|
 * @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
 | 
						|
 */
 | 
						|
function guessDMRoomTargetId(room: Room, myUserId: string): string {
 | 
						|
    let oldestTs: number | undefined;
 | 
						|
    let oldestUser: RoomMember | undefined;
 | 
						|
 | 
						|
    // Pick the joined user who's been here longest (and isn't us),
 | 
						|
    for (const user of room.getJoinedMembers()) {
 | 
						|
        if (user.userId == myUserId) continue;
 | 
						|
 | 
						|
        if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
 | 
						|
            oldestUser = user;
 | 
						|
            oldestTs = user.events.member?.getTs();
 | 
						|
        }
 | 
						|
    }
 | 
						|
    if (oldestUser) return oldestUser.userId;
 | 
						|
 | 
						|
    // if there are no joined members other than us, use the oldest member
 | 
						|
    for (const user of room.currentState.getMembers()) {
 | 
						|
        if (user.userId == myUserId) continue;
 | 
						|
 | 
						|
        if (oldestTs === undefined || (user.events.member && user.events.member.getTs() < oldestTs)) {
 | 
						|
            oldestUser = user;
 | 
						|
            oldestTs = user.events.member?.getTs();
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    if (oldestUser === undefined) return myUserId;
 | 
						|
    return oldestUser.userId;
 | 
						|
}
 |