Files
element-call/src/state/remoteMembers/displayname.ts

93 lines
3.1 KiB
TypeScript
Raw Normal View History

2025-10-28 21:18:47 +01:00
/*
Copyright 2025 New Vector Ltd.
SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial
Please see LICENSE in the repository root for full details.
*/
import { type RoomMember, RoomStateEvent } from "matrix-js-sdk";
2025-11-04 17:13:28 +01:00
import {
combineLatest,
fromEvent,
map,
type Observable,
startWith,
} from "rxjs";
2025-10-28 21:18:47 +01:00
import { type CallMembership } from "matrix-js-sdk/lib/matrixrtc";
import { logger } from "matrix-js-sdk/lib/logger";
import { type Room as MatrixRoom } from "matrix-js-sdk/lib/matrix";
// eslint-disable-next-line rxjs/no-internal
import { type HasEventTargetAddRemove } from "rxjs/internal/observable/fromEvent";
2025-10-28 21:18:47 +01:00
import { type ObservableScope } from "../ObservableScope";
2025-10-30 00:09:07 +01:00
import {
calculateDisplayName,
shouldDisambiguate,
} from "../../utils/displayname";
import { type Behavior } from "../Behavior";
2025-10-28 21:18:47 +01:00
/**
* Displayname for each member of the call. This will disambiguate
* any displayname that clashes with another member. Only members
* joined to the call are considered here.
*
* @returns Map<member.id, displayname> uses the rtc member idenitfier as the key.
2025-10-28 21:18:47 +01:00
*/
// don't do this work more times than we need to. This is achieved by converting to a behavior:
export const memberDisplaynames$ = (
2025-10-30 00:09:07 +01:00
scope: ObservableScope,
matrixRoom: Pick<MatrixRoom, "getMember"> & HasEventTargetAddRemove<unknown>,
2025-10-28 21:18:47 +01:00
memberships$: Observable<CallMembership[]>,
userId: string,
deviceId: string,
2025-10-30 00:09:07 +01:00
): Behavior<Map<string, string>> =>
2025-10-28 21:18:47 +01:00
scope.behavior(
2025-11-04 17:13:28 +01:00
combineLatest([
// Handle call membership changes
memberships$,
// Additionally handle display name changes (implicitly reacting to them)
fromEvent(matrixRoom, RoomStateEvent.Members).pipe(startWith(null)),
// TODO: do we need: pauseWhen(this.pretendToBeDisconnected$),
]).pipe(
map((memberships, _displaynames) => {
2025-10-28 21:18:47 +01:00
const displaynameMap = new Map<string, string>([
[
`${userId}:${deviceId}`,
matrixRoom.getMember(userId)?.rawDisplayName ?? userId,
],
]);
const room = matrixRoom;
// We only consider RTC members for disambiguation as they are the only visible members.
for (const rtcMember of memberships) {
const matrixIdentifier = `${rtcMember.userId}:${rtcMember.deviceId}`;
const { member } = getRoomMemberFromRtcMember(rtcMember, room);
if (!member) {
logger.error(
"Could not find member for media id:",
matrixIdentifier,
);
continue;
}
const disambiguate = shouldDisambiguate(member, memberships, room);
displaynameMap.set(
matrixIdentifier,
calculateDisplayName(member, disambiguate),
);
}
return displaynameMap;
2025-11-04 17:13:28 +01:00
}),
2025-10-28 21:18:47 +01:00
),
2025-11-04 17:13:28 +01:00
new Map<string, string>(),
2025-10-28 21:18:47 +01:00
);
export function getRoomMemberFromRtcMember(
rtcMember: CallMembership,
room: Pick<MatrixRoom, "getMember">,
2025-10-28 21:18:47 +01:00
): { id: string; member: RoomMember | undefined } {
return {
id: rtcMember.userId + ":" + rtcMember.deviceId,
member: room.getMember(rtcMember.userId) ?? undefined,
};
}