Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make video tiles be based on MatrixRTC member not LiveKit participants #2701

Merged
merged 99 commits into from
Dec 6, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
99 commits
Select commit Hold shift + click to select a range
5f8809c
make tiles based on rtc member
toger5 Oct 30, 2024
3f233a6
display missing lk participant + fix tile multiplier
toger5 Oct 30, 2024
34e9e8a
add show_non_member_participants config option
toger5 Oct 30, 2024
e1e202d
per member tiles
toger5 Nov 4, 2024
2a5dc60
merge fixes
toger5 Nov 4, 2024
1ab4d50
linter
toger5 Nov 4, 2024
14919ca
linter and tests
toger5 Nov 4, 2024
b5208ff
tests
toger5 Nov 5, 2024
e64204d
adapt tests (wip)
toger5 Nov 5, 2024
bb0febf
Remove unused keys
hughns Nov 5, 2024
e7dbddb
Fix optionality of nonMemberItemCount
hughns Nov 6, 2024
0d1b54e
video is optional
hughns Nov 6, 2024
fecd789
Mock RTC members
hughns Nov 6, 2024
733b685
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 6, 2024
c829f2f
Lint
hughns Nov 6, 2024
73107e4
Merge fixes
hughns Nov 6, 2024
bcbdb59
Fix user id
hughns Nov 6, 2024
b0e0e0e
Add explicit types for public fields
hughns Nov 6, 2024
a1083f3
isRTCParticipantAvailable => isLiveKitParticipantAvailable
hughns Nov 6, 2024
5213636
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 6, 2024
2c3d733
isLiveKitParticipantAvailable
hughns Nov 6, 2024
d8d4e89
Readonly
hughns Nov 6, 2024
efee27a
More keys removal
hughns Nov 6, 2024
eec4470
Make local field based on view model class not observable
hughns Nov 6, 2024
b1798e7
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 6, 2024
462f034
Wording
hughns Nov 6, 2024
efe9c9b
Fix RTC members in tes
hughns Nov 6, 2024
4f6b1b0
Tests again
hughns Nov 6, 2024
59e49d5
Lint
hughns Nov 6, 2024
f9f8f37
Disable showing non-member tiles by default
hughns Nov 6, 2024
dfd7273
Duplicate screen sharing tiles like we used to
hughns Nov 7, 2024
e515359
Lint
hughns Nov 7, 2024
9bfd2e3
Revert function reordering
hughns Nov 7, 2024
0760796
Remove throttleTime from bad merge
hughns Nov 7, 2024
95effe0
Cleanup
hughns Nov 7, 2024
6bda895
Tidy config of show non-member settings
hughns Nov 7, 2024
5e8a947
tidy up handling of local rtc member in tests
hughns Nov 7, 2024
bb56f42
tidy up test init
hughns Nov 7, 2024
5da642b
Fix mocks
hughns Nov 7, 2024
bf41cfc
Cleanup
hughns Nov 7, 2024
0c4cddb
Apply local override where participant not yet known
hughns Nov 7, 2024
be250e2
Handle no visible media id
hughns Nov 7, 2024
5009f1f
Assertions for one-on-one view
hughns Nov 7, 2024
238797a
Remove isLiveKitParticipantAvailable and show via encryption status
hughns Nov 7, 2024
c37c2e5
Handle no local media (yet)
hughns Nov 7, 2024
190ac9b
Remove unused effect for setting
hughns Nov 7, 2024
cf3893b
Tidy settings
hughns Nov 7, 2024
886dc2c
Avoid case of one-to-one layout with missing local or remote
hughns Nov 8, 2024
02b96fa
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 11, 2024
dbbf774
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 13, 2024
8351421
Iterate
hughns Nov 13, 2024
44935ee
Remove option to show non-member tiles to simplify code review
hughns Nov 13, 2024
b3e725f
Remove unused code
hughns Nov 13, 2024
87c3793
Remove more remnants of show-non-member-tiles
hughns Nov 14, 2024
fa3c9fd
iterate
hughns Nov 14, 2024
eafd8fd
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 14, 2024
f6a641b
back
hughns Nov 14, 2024
5cae849
Fix unit test
hughns Nov 18, 2024
09de976
Refactor
hughns Nov 18, 2024
23579b2
Expose TestScheduler as global
hughns Nov 18, 2024
8f59f08
Fix incorrect type assertion
hughns Nov 20, 2024
256a65a
Simplify speaking observer
hughns Nov 20, 2024
67a9d58
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 20, 2024
edb53ae
Fix
hughns Nov 20, 2024
3cb5ecc
Whitespace
hughns Nov 20, 2024
138a6c7
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 20, 2024
7677514
Make it clear that we are mocking MatrixRTC memberships
hughns Nov 20, 2024
b7ecd39
Test case for only showing tiles for MatrixRTC session members
hughns Nov 20, 2024
18e7ca5
Simplify diff
hughns Nov 20, 2024
a63d44a
Simplify diff
hughns Nov 20, 2024
1c04d22
.
hughns Nov 20, 2024
3f11f51
Whitespaces
hughns Nov 20, 2024
bc90deb
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 20, 2024
8f62cb6
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 21, 2024
ee8038f
Use asObservable when exposing subject
hughns Nov 21, 2024
e0651c5
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 23, 2024
b819a72
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Nov 25, 2024
4f2591f
Show "waiting for media..." when no participant
hughns Nov 25, 2024
16666b8
Additional test case
hughns Nov 25, 2024
8f9bee7
Don't show "waiting for media..." in case of local participant
hughns Nov 25, 2024
765f7b4
Make the loading state more subtle
toger5 Nov 26, 2024
52cb393
Use correct key for matrix rtc foci in code comment. (#2838)
toger5 Nov 27, 2024
cc1f0d5
Update src/tile/SpotlightTile.tsx
hughns Nov 27, 2024
133dc26
Update src/state/CallViewModel.ts
hughns Nov 27, 2024
743638d
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Dec 2, 2024
3d0bfae
Make the purpose of BaseMediaViewModel.local explicit
hughns Dec 2, 2024
968de5e
Use named object instead of unnamed array for spotlightAndPip
hughns Dec 2, 2024
33f398d
Refactor spotlightAndPip into spotlight and pip
hughns Dec 2, 2024
7fdcbb3
Use if statement instead of ternary for readability in spotlight and …
hughns Dec 2, 2024
370d8bd
Review feedback
hughns Dec 2, 2024
a37e8fd
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Dec 2, 2024
70deaf0
Fix tests for CallEventAudioRenderer
hughns Dec 2, 2024
79ea8d9
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Dec 2, 2024
e4087b2
Lint
hughns Dec 2, 2024
2c923bc
Revert "Make the loading state more subtle"
hughns Dec 2, 2024
63b2fe6
Update src/state/CallViewModel.ts
hughns Dec 4, 2024
00156a4
Fix spelling
hughns Dec 4, 2024
4a783e8
Merge branch 'livekit' into toger5/tiles_based_on_rtc_member
hughns Dec 4, 2024
0edfec7
Remove a non-null assertion that failed at runtime
hughns Dec 4, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion locales/en/app.json
Original file line number Diff line number Diff line change
Expand Up @@ -194,6 +194,7 @@
"expand": "Expand",
"mute_for_me": "Mute for me",
"muted_for_me": "Muted for me",
"volume": "Volume"
"volume": "Volume",
"waiting_for_media": "Waiting for media..."
}
}
2 changes: 1 addition & 1 deletion src/config/ConfigOptions.ts
Original file line number Diff line number Diff line change
Expand Up @@ -51,7 +51,7 @@ export interface ConfigOptions {
// a livekit service url in the client well-known.
// The well known needs to be formatted like so:
// {"type":"livekit", "livekit_service_url":"https://livekit.example.com"}
// and stored under the key: "livekit_focus"
// and stored under the key: "org.matrix.msc4143.rtc_foci"
livekit_service_url: string;
};

Expand Down
182 changes: 126 additions & 56 deletions src/room/CallEventAudioRenderer.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -8,10 +8,14 @@ Please see LICENSE in the repository root for full details.
import { render } from "@testing-library/react";
import { beforeEach, expect, test } from "vitest";
import { MatrixClient } from "matrix-js-sdk/src/client";
import { ConnectionState, RemoteParticipant, Room } from "livekit-client";
import { of } from "rxjs";
import { ConnectionState, Room } from "livekit-client";
import { BehaviorSubject, of } from "rxjs";
import { afterEach } from "node:test";
import { act } from "react";
import {
CallMembership,
type MatrixRTCSession,
} from "matrix-js-sdk/src/matrixrtc";

import { soundEffectVolumeSetting } from "../settings/settings";
import {
Expand All @@ -22,6 +26,8 @@ import {
mockMatrixRoomMember,
mockMediaPlay,
mockRemoteParticipant,
mockRtcMembership,
MockRTCSession,
} from "../utils/test";
import { E2eeType } from "../e2ee/e2eeType";
import { CallViewModel } from "../state/CallViewModel";
Expand All @@ -30,11 +36,15 @@ import {
MAX_PARTICIPANT_COUNT_FOR_SOUND,
} from "./CallEventAudioRenderer";

const alice = mockMatrixRoomMember({ userId: "@alice:example.org" });
const bob = mockMatrixRoomMember({ userId: "@bob:example.org" });
const aliceId = `${alice.userId}:AAAA`;
const bobId = `${bob.userId}:BBBB`;
const localRtcMember = mockRtcMembership("@carol:example.org", "CCCC");
const local = mockMatrixRoomMember(localRtcMember);
const aliceRtcMember = mockRtcMembership("@alice:example.org", "AAAA");
const alice = mockMatrixRoomMember(aliceRtcMember);
const bobRtcMember = mockRtcMembership("@bob:example.org", "BBBB");
const bob = mockMatrixRoomMember(bobRtcMember);
const localParticipant = mockLocalParticipant({ identity: "" });
const aliceId = `${alice.userId}:${aliceRtcMember.deviceId}`;
const bobId = `${bob.userId}:${bobRtcMember.deviceId}`;
const aliceParticipant = mockRemoteParticipant({ identity: aliceId });
const bobParticipant = mockRemoteParticipant({ identity: bobId });

Expand All @@ -53,20 +63,28 @@ afterEach(() => {

test("plays a sound when entering a call", () => {
const audioIsPlaying: string[] = mockMediaPlay();
const members = new Map([alice, bob].map((p) => [p.userId, p]));
const matrixRoomMembers = new Map(
[local, alice, bob].map((p) => [p.userId, p]),
);
const remoteParticipants = of([aliceParticipant]);
const liveKitRoom = mockLivekitRoom(
{ localParticipant },
{ remoteParticipants },
);
const matrixRoom = mockMatrixRoom({
client: {
getUserId: () => localRtcMember.sender,
getDeviceId: () => localRtcMember.deviceId,
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => matrixRoomMembers.get(userId) ?? null,
});

const session = new MockRTCSession(matrixRoom, localRtcMember, [
aliceRtcMember,
]) as unknown as MatrixRTCSession;

const vm = new CallViewModel(
mockMatrixRoom({
client: {
getUserId: () => "@carol:example.org",
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => members.get(userId) ?? null,
}),
session,
liveKitRoom,
{
kind: E2eeType.PER_PARTICIPANT,
Expand All @@ -84,20 +102,29 @@ test("plays a sound when entering a call", () => {
test("plays no sound when muted", () => {
soundEffectVolumeSetting.setValue(0);
const audioIsPlaying: string[] = mockMediaPlay();
const members = new Map([alice, bob].map((p) => [p.userId, p]));
const matrixRoomMembers = new Map(
[local, alice, bob].map((p) => [p.userId, p]),
);
const remoteParticipants = of([aliceParticipant, bobParticipant]);
const liveKitRoom = mockLivekitRoom(
{ localParticipant },
{ remoteParticipants },
);

const matrixRoom = mockMatrixRoom({
client: {
getUserId: () => localRtcMember.sender,
getDeviceId: () => localRtcMember.deviceId,
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => matrixRoomMembers.get(userId) ?? null,
});

const session = new MockRTCSession(matrixRoom, localRtcMember, [
aliceRtcMember,
]) as unknown as MatrixRTCSession;

const vm = new CallViewModel(
mockMatrixRoom({
client: {
getUserId: () => "@carol:example.org",
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => members.get(userId) ?? null,
}),
session,
liveKitRoom,
{
kind: E2eeType.PER_PARTICIPANT,
Expand All @@ -112,7 +139,7 @@ test("plays no sound when muted", () => {

test("plays a sound when a user joins", () => {
const audioIsPlaying: string[] = mockMediaPlay();
const members = new Map([alice].map((p) => [p.userId, p]));
const matrixRoomMembers = new Map([local, alice].map((p) => [p.userId, p]));
const remoteParticipants = new Map(
[aliceParticipant].map((p) => [p.identity, p]),
);
Expand All @@ -121,13 +148,27 @@ test("plays a sound when a user joins", () => {
remoteParticipants,
});

const matrixRoom = mockMatrixRoom({
client: {
getUserId: () => localRtcMember.sender,
getDeviceId: () => localRtcMember.deviceId,
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => matrixRoomMembers.get(userId) ?? null,
});

const remoteRtcMemberships = new BehaviorSubject<CallMembership[]>([
aliceRtcMember,
]);
// we give Bob an RTC session now, but no participant yet
const session = new MockRTCSession(
matrixRoom,
localRtcMember,
).withMemberships(
remoteRtcMemberships.asObservable(),
) as unknown as MatrixRTCSession;

const vm = new CallViewModel(
mockMatrixRoom({
client: {
getUserId: () => "@carol:example.org",
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => members.get(userId) ?? null,
}),
session,
liveKitRoom as unknown as Room,
{
kind: E2eeType.PER_PARTICIPANT,
Expand All @@ -137,20 +178,20 @@ test("plays a sound when a user joins", () => {
render(<CallEventAudioRenderer vm={vm} />);

act(() => {
liveKitRoom.addParticipant(bobParticipant);
remoteRtcMemberships.next([aliceRtcMember, bobRtcMember]);
});
// Play a sound when joining a call.
expect(audioIsPlaying).toEqual([
// Joining the call
enterSound,
// Bob leaves
// Bob joins
enterSound,
]);
});

test("plays a sound when a user leaves", () => {
const audioIsPlaying: string[] = mockMediaPlay();
const members = new Map([alice].map((p) => [p.userId, p]));
const matrixRoomMembers = new Map([local, alice].map((p) => [p.userId, p]));
const remoteParticipants = new Map(
[aliceParticipant].map((p) => [p.identity, p]),
);
Expand All @@ -159,13 +200,25 @@ test("plays a sound when a user leaves", () => {
remoteParticipants,
});

const matrixRoom = mockMatrixRoom({
client: {
getUserId: () => localRtcMember.sender,
getDeviceId: () => localRtcMember.deviceId,
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => matrixRoomMembers.get(userId) ?? null,
});

const remoteRtcMemberships = new BehaviorSubject<CallMembership[]>([
aliceRtcMember,
]);

const session = new MockRTCSession(
matrixRoom,
localRtcMember,
).withMemberships(remoteRtcMemberships) as unknown as MatrixRTCSession;

const vm = new CallViewModel(
mockMatrixRoom({
client: {
getUserId: () => "@carol:example.org",
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => members.get(userId) ?? null,
}),
session,
liveKitRoom as unknown as Room,
{
kind: E2eeType.PER_PARTICIPANT,
Expand All @@ -175,7 +228,7 @@ test("plays a sound when a user leaves", () => {
render(<CallEventAudioRenderer vm={vm} />);

act(() => {
liveKitRoom.removeParticipant(aliceParticipant);
remoteRtcMemberships.next([]);
});
expect(audioIsPlaying).toEqual([
// Joining the call
Expand All @@ -185,30 +238,45 @@ test("plays a sound when a user leaves", () => {
]);
});

test("plays no sound when the participant list", () => {
test("plays no sound when the session member count is larger than the max, until decreased", () => {
const audioIsPlaying: string[] = mockMediaPlay();
const members = new Map([alice].map((p) => [p.userId, p]));
const remoteParticipants = new Map<string, RemoteParticipant>([
[aliceParticipant.identity, aliceParticipant],
...Array.from({ length: MAX_PARTICIPANT_COUNT_FOR_SOUND - 1 }).map<
[string, RemoteParticipant]
>((_, index) => {
const p = mockRemoteParticipant({ identity: `user${index}` });
return [p.identity, p];
}),
]);
const matrixRoomMembers = new Map([local, alice].map((p) => [p.userId, p]));
const remoteParticipants = new Map(
[aliceParticipant].map((p) => [p.identity, p]),
);

const mockRtcMemberships: CallMembership[] = [];

for (let i = 0; i < MAX_PARTICIPANT_COUNT_FOR_SOUND; i++) {
mockRtcMemberships.push(
mockRtcMembership(`@user${i}:example.org`, `DEVICE${i}`),
);
}

const remoteRtcMemberships = new BehaviorSubject<CallMembership[]>(
mockRtcMemberships,
);

const liveKitRoom = new EmittableMockLivekitRoom({
localParticipant,
remoteParticipants,
});

const matrixRoom = mockMatrixRoom({
client: {
getUserId: () => localRtcMember.sender,
getDeviceId: () => localRtcMember.deviceId,
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => matrixRoomMembers.get(userId) ?? null,
});

const session = new MockRTCSession(
matrixRoom,
localRtcMember,
).withMemberships(remoteRtcMemberships) as unknown as MatrixRTCSession;

const vm = new CallViewModel(
mockMatrixRoom({
client: {
getUserId: () => "@carol:example.org",
} as Partial<MatrixClient> as MatrixClient,
getMember: (userId) => members.get(userId) ?? null,
}),
session,
liveKitRoom as unknown as Room,
{
kind: E2eeType.PER_PARTICIPANT,
Expand All @@ -217,9 +285,11 @@ test("plays no sound when the participant list", () => {
);
render(<CallEventAudioRenderer vm={vm} />);
expect(audioIsPlaying).toEqual([]);
// When the count drops
// When the count drops to the max we should play the leave sound
act(() => {
liveKitRoom.removeParticipant(aliceParticipant);
remoteRtcMemberships.next(
mockRtcMemberships.slice(0, MAX_PARTICIPANT_COUNT_FOR_SOUND - 1),
);
});
expect(audioIsPlaying).toEqual([leaveSound]);
});
9 changes: 2 additions & 7 deletions src/room/InCallView.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -124,20 +124,15 @@ export const ActiveCall: FC<ActiveCallProps> = (props) => {
useEffect(() => {
if (livekitRoom !== undefined) {
const vm = new CallViewModel(
props.rtcSession.room,
props.rtcSession,
livekitRoom,
props.e2eeSystem,
connStateObservable,
);
setVm(vm);
return (): void => vm.destroy();
}
}, [
props.rtcSession.room,
livekitRoom,
props.e2eeSystem,
connStateObservable,
]);
}, [props.rtcSession, livekitRoom, props.e2eeSystem, connStateObservable]);

if (livekitRoom === undefined || vm === null) return null;

Expand Down
Loading
Loading