mirror of
https://github.com/vector-im/element-web.git
synced 2025-11-08 20:21:42 +01:00
* Remember whether sidebar is shown for calls when switching rooms Stores the sidebar state per-room in LegacyCallHandler, along with other details about calls. * Hide the Show/Hide Sidebar from the Picture-in-Picture preview The toggle sidebar button currently does nothing in PIP mode, since PIP mode never shows a sidebar (even when the call is made fullscreen from the PIP preview) * Add test for Show/Hide Sidebar feature * Add more tests for LegacyCallView and LegacyCallViewForRoom Also, fix issue where LegacyCallViewForRoom used roomId and not callId for checking for sidebar state
627 lines
22 KiB
TypeScript
627 lines
22 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
Copyright 2021 The Matrix.org Foundation C.I.C.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only OR LicenseRef-Element-Commercial
|
|
Please see LICENSE files in the repository root for full details.
|
|
*/
|
|
|
|
import {
|
|
type IProtocol,
|
|
LOCAL_NOTIFICATION_SETTINGS_PREFIX,
|
|
MatrixEvent,
|
|
PushRuleKind,
|
|
type Room,
|
|
RuleId,
|
|
TweakName,
|
|
} from "matrix-js-sdk/src/matrix";
|
|
import { KnownMembership } from "matrix-js-sdk/src/types";
|
|
import { CallEvent, CallState, CallType, MatrixCall } from "matrix-js-sdk/src/webrtc/call";
|
|
import EventEmitter from "events";
|
|
import { mocked } from "jest-mock";
|
|
import { CallEventHandlerEvent } from "matrix-js-sdk/src/webrtc/callEventHandler";
|
|
import fetchMock from "fetch-mock-jest";
|
|
import { waitFor } from "jest-matrix-react";
|
|
import { PushProcessor } from "matrix-js-sdk/src/pushprocessor";
|
|
|
|
import LegacyCallHandler, {
|
|
AudioID,
|
|
LegacyCallHandlerEvent,
|
|
PROTOCOL_PSTN,
|
|
PROTOCOL_PSTN_PREFIXED,
|
|
} from "../../src/LegacyCallHandler";
|
|
import { mkStubRoom, stubClient, untilDispatch } from "../test-utils";
|
|
import { MatrixClientPeg } from "../../src/MatrixClientPeg";
|
|
import DMRoomMap from "../../src/utils/DMRoomMap";
|
|
import SdkConfig from "../../src/SdkConfig";
|
|
import { Action } from "../../src/dispatcher/actions";
|
|
import { getFunctionalMembers } from "../../src/utils/room/getFunctionalMembers";
|
|
import SettingsStore from "../../src/settings/SettingsStore";
|
|
import { UIFeature } from "../../src/settings/UIFeature";
|
|
import { createAudioContext } from "../../src/audio/compat";
|
|
import * as ManagedHybrid from "../../src/widgets/ManagedHybrid";
|
|
|
|
jest.mock("../../src/Modal");
|
|
|
|
// mock VoiceRecording because it contains all the audio APIs
|
|
jest.mock("../../src/audio/VoiceRecording", () => ({
|
|
VoiceRecording: jest.fn().mockReturnValue({
|
|
disableMaxLength: jest.fn(),
|
|
liveData: {
|
|
onUpdate: jest.fn(),
|
|
},
|
|
off: jest.fn(),
|
|
on: jest.fn(),
|
|
start: jest.fn(),
|
|
stop: jest.fn(),
|
|
destroy: jest.fn(),
|
|
contentType: "audio/ogg",
|
|
}),
|
|
}));
|
|
|
|
jest.mock("../../src/utils/room/getFunctionalMembers", () => ({
|
|
getFunctionalMembers: jest.fn(),
|
|
}));
|
|
|
|
jest.mock("../../src/audio/compat", () => ({
|
|
...jest.requireActual("../../src/audio/compat"),
|
|
createAudioContext: jest.fn(),
|
|
}));
|
|
|
|
// The Matrix IDs that the user sees when talking to Alice & Bob
|
|
const NATIVE_ALICE = "@alice:example.org";
|
|
const NATIVE_BOB = "@bob:example.org";
|
|
const NATIVE_CHARLIE = "@charlie:example.org";
|
|
|
|
//const REAL_ROOM_ID = "$room1:example.org";
|
|
// The rooms the user sees when they're communicating with these users
|
|
const NATIVE_ROOM_ALICE = "$alice_room:example.org";
|
|
const NATIVE_ROOM_BOB = "$bob_room:example.org";
|
|
const NATIVE_ROOM_CHARLIE = "$charlie_room:example.org";
|
|
|
|
const FUNCTIONAL_USER = "@bot:example.com";
|
|
|
|
// Bob's phone number
|
|
const BOB_PHONE_NUMBER = "01818118181";
|
|
|
|
function mkStubDM(roomId: string, userId: string) {
|
|
const room = mkStubRoom(roomId, "room", MatrixClientPeg.safeGet());
|
|
room.getJoinedMembers = jest.fn().mockReturnValue([
|
|
{
|
|
userId: "@me:example.org",
|
|
name: "Member",
|
|
rawDisplayName: "Member",
|
|
roomId: roomId,
|
|
membership: KnownMembership.Join,
|
|
getAvatarUrl: () => "mxc://avatar.url/image.png",
|
|
getMxcAvatarUrl: () => "mxc://avatar.url/image.png",
|
|
},
|
|
{
|
|
userId: userId,
|
|
name: "Member",
|
|
rawDisplayName: "Member",
|
|
roomId: roomId,
|
|
membership: KnownMembership.Join,
|
|
getAvatarUrl: () => "mxc://avatar.url/image.png",
|
|
getMxcAvatarUrl: () => "mxc://avatar.url/image.png",
|
|
},
|
|
{
|
|
userId: FUNCTIONAL_USER,
|
|
name: "Bot user",
|
|
rawDisplayName: "Bot user",
|
|
roomId: roomId,
|
|
membership: KnownMembership.Join,
|
|
getAvatarUrl: () => "mxc://avatar.url/image.png",
|
|
getMxcAvatarUrl: () => "mxc://avatar.url/image.png",
|
|
},
|
|
]);
|
|
room.currentState.getMembers = room.getJoinedMembers;
|
|
return room;
|
|
}
|
|
|
|
class FakeCall extends EventEmitter {
|
|
roomId: string;
|
|
callId = "fake call id";
|
|
|
|
constructor(roomId: string) {
|
|
super();
|
|
|
|
this.roomId = roomId;
|
|
}
|
|
|
|
setRemoteOnHold() {}
|
|
setRemoteAudioElement() {}
|
|
|
|
placeVoiceCall() {
|
|
this.emit(CallEvent.State, CallState.Connected, null);
|
|
}
|
|
}
|
|
|
|
describe("LegacyCallHandler", () => {
|
|
let dmRoomMap;
|
|
let callHandler: LegacyCallHandler;
|
|
let audioElement: HTMLAudioElement;
|
|
let fakeCall: MatrixCall | null;
|
|
|
|
// what addresses the app has looked up via pstn and native lookup
|
|
let pstnLookup: string | null;
|
|
const deviceId = "my-device";
|
|
|
|
beforeEach(async () => {
|
|
stubClient();
|
|
fakeCall = null;
|
|
MatrixClientPeg.safeGet().createCall = (roomId: string): MatrixCall | null => {
|
|
if (fakeCall && fakeCall.roomId !== roomId) {
|
|
throw new Error("Only one call is supported!");
|
|
}
|
|
fakeCall = new FakeCall(roomId) as unknown as MatrixCall;
|
|
return fakeCall as unknown as MatrixCall;
|
|
};
|
|
MatrixClientPeg.safeGet().deviceId = deviceId;
|
|
|
|
MatrixClientPeg.safeGet().getThirdpartyProtocols = () => {
|
|
return Promise.resolve({
|
|
"m.id.phone": {} as IProtocol,
|
|
});
|
|
};
|
|
|
|
callHandler = new LegacyCallHandler();
|
|
callHandler.start();
|
|
|
|
mocked(getFunctionalMembers).mockReturnValue([FUNCTIONAL_USER]);
|
|
|
|
const nativeRoomAlice = mkStubDM(NATIVE_ROOM_ALICE, NATIVE_ALICE);
|
|
const nativeRoomBob = mkStubDM(NATIVE_ROOM_BOB, NATIVE_BOB);
|
|
const nativeRoomCharie = mkStubDM(NATIVE_ROOM_CHARLIE, NATIVE_CHARLIE);
|
|
|
|
MatrixClientPeg.safeGet().getRoom = (roomId: string): Room | null => {
|
|
switch (roomId) {
|
|
case NATIVE_ROOM_ALICE:
|
|
return nativeRoomAlice;
|
|
case NATIVE_ROOM_BOB:
|
|
return nativeRoomBob;
|
|
case NATIVE_ROOM_CHARLIE:
|
|
return nativeRoomCharie;
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
dmRoomMap = {
|
|
getUserIdForRoomId: (roomId: string) => {
|
|
if (roomId === NATIVE_ROOM_ALICE) {
|
|
return NATIVE_ALICE;
|
|
} else if (roomId === NATIVE_ROOM_BOB) {
|
|
return NATIVE_BOB;
|
|
} else if (roomId === NATIVE_ROOM_CHARLIE) {
|
|
return NATIVE_CHARLIE;
|
|
} else {
|
|
return null;
|
|
}
|
|
},
|
|
getDMRoomsForUserId: (userId: string) => {
|
|
if (userId === NATIVE_ALICE) {
|
|
return [NATIVE_ROOM_ALICE];
|
|
} else if (userId === NATIVE_BOB) {
|
|
return [NATIVE_ROOM_BOB];
|
|
} else if (userId === NATIVE_CHARLIE) {
|
|
return [NATIVE_ROOM_CHARLIE];
|
|
} else {
|
|
return [];
|
|
}
|
|
},
|
|
} as unknown as DMRoomMap;
|
|
DMRoomMap.setShared(dmRoomMap);
|
|
|
|
pstnLookup = null;
|
|
|
|
MatrixClientPeg.safeGet().getThirdpartyUser = (proto: string, params: any) => {
|
|
if ([PROTOCOL_PSTN, PROTOCOL_PSTN_PREFIXED].includes(proto)) {
|
|
pstnLookup = params["m.id.phone"];
|
|
return Promise.resolve([
|
|
{
|
|
userid: NATIVE_BOB,
|
|
protocol: "m.id.phone",
|
|
fields: {},
|
|
},
|
|
]);
|
|
}
|
|
return Promise.resolve([]);
|
|
};
|
|
|
|
audioElement = document.createElement("audio");
|
|
audioElement.id = "remoteAudio";
|
|
document.body.appendChild(audioElement);
|
|
});
|
|
|
|
afterEach(() => {
|
|
callHandler.stop();
|
|
// @ts-ignore
|
|
DMRoomMap.setShared(null);
|
|
// @ts-ignore
|
|
window.mxLegacyCallHandler = null;
|
|
MatrixClientPeg.unset();
|
|
|
|
document.body.removeChild(audioElement);
|
|
SdkConfig.reset();
|
|
jest.restoreAllMocks();
|
|
});
|
|
|
|
it("should look up the correct user and start a call in the room when a phone number is dialled", async () => {
|
|
await callHandler.dialNumber(BOB_PHONE_NUMBER);
|
|
|
|
expect(pstnLookup).toEqual(BOB_PHONE_NUMBER);
|
|
|
|
// we should have switched to the native room for Bob
|
|
const viewRoomPayload = await untilDispatch(Action.ViewRoom);
|
|
expect(viewRoomPayload.room_id).toEqual(NATIVE_ROOM_BOB);
|
|
|
|
// Check that a call was started: its room on the protocol level
|
|
expect(fakeCall).not.toBeNull();
|
|
expect(fakeCall?.roomId).toEqual(NATIVE_ROOM_BOB);
|
|
|
|
// but it should appear to the user to be in thw native room for Bob
|
|
expect(callHandler.roomIdForCall(fakeCall!)).toEqual(NATIVE_ROOM_BOB);
|
|
});
|
|
|
|
it("should look up the correct user and start a call in the room when a call is transferred", async () => {
|
|
// we can pass a very minimal object as as the call since we pass consultFirst=true:
|
|
// we don't need to actually do any transferring
|
|
const mockTransferreeCall = { type: CallType.Voice } as unknown as MatrixCall;
|
|
await callHandler.startTransferToPhoneNumber(mockTransferreeCall, BOB_PHONE_NUMBER, true);
|
|
|
|
// same checks as above
|
|
const viewRoomPayload = await untilDispatch(Action.ViewRoom);
|
|
expect(viewRoomPayload.room_id).toEqual(NATIVE_ROOM_BOB);
|
|
|
|
expect(fakeCall).not.toBeNull();
|
|
expect(fakeCall!.roomId).toEqual(NATIVE_ROOM_BOB);
|
|
|
|
expect(callHandler.roomIdForCall(fakeCall!)).toEqual(NATIVE_ROOM_BOB);
|
|
});
|
|
|
|
it("should move calls between rooms when remote asserted identity changes", async () => {
|
|
callHandler.placeCall(NATIVE_ROOM_ALICE, CallType.Voice);
|
|
|
|
// We placed the call in Alice's room so it should start off there
|
|
expect(callHandler.getCallForRoom(NATIVE_ROOM_ALICE)).toBe(fakeCall);
|
|
|
|
let callRoomChangeEventCount = 0;
|
|
const roomChangePromise = new Promise<void>((resolve) => {
|
|
callHandler.addListener(LegacyCallHandlerEvent.CallChangeRoom, () => {
|
|
++callRoomChangeEventCount;
|
|
resolve();
|
|
});
|
|
});
|
|
|
|
// Now emit an asserted identity for Bob: this should be ignored
|
|
// because we haven't set the config option to obey asserted identity
|
|
expect(fakeCall).not.toBeNull();
|
|
fakeCall!.getRemoteAssertedIdentity = jest.fn().mockReturnValue({
|
|
id: NATIVE_BOB,
|
|
});
|
|
fakeCall!.emit(CallEvent.AssertedIdentityChanged, fakeCall!);
|
|
|
|
// Now set the config option
|
|
SdkConfig.add({
|
|
voip: {
|
|
obey_asserted_identity: true,
|
|
},
|
|
});
|
|
|
|
// ...and send another asserted identity event for a different user
|
|
fakeCall!.getRemoteAssertedIdentity = jest.fn().mockReturnValue({
|
|
id: NATIVE_CHARLIE,
|
|
});
|
|
fakeCall!.emit(CallEvent.AssertedIdentityChanged, fakeCall!);
|
|
|
|
await roomChangePromise;
|
|
callHandler.removeAllListeners();
|
|
|
|
// If everything's gone well, we should have seen only one room change
|
|
// event and the call should now be in Charlie's room.
|
|
// If it's not obeying any, the call will still be in NATIVE_ROOM_ALICE.
|
|
// If it incorrectly obeyed both asserted identity changes, either it will
|
|
// have just processed one and the call will be in the wrong room, or we'll
|
|
// have seen two room change dispatches.
|
|
expect(callRoomChangeEventCount).toEqual(1);
|
|
expect(callHandler.getCallForRoom(NATIVE_ROOM_BOB)).toBeNull();
|
|
expect(callHandler.getCallForRoom(NATIVE_ROOM_CHARLIE)).toBe(fakeCall);
|
|
});
|
|
|
|
it("should place calls using managed hybrid widget if enabled", async () => {
|
|
const spy = jest.spyOn(ManagedHybrid, "addManagedHybridWidget");
|
|
jest.spyOn(ManagedHybrid, "isManagedHybridWidgetEnabled").mockReturnValue(true);
|
|
await callHandler.placeCall(NATIVE_ROOM_ALICE, CallType.Voice);
|
|
expect(spy).toHaveBeenCalledWith(MatrixClientPeg.safeGet().getRoom(NATIVE_ROOM_ALICE));
|
|
});
|
|
});
|
|
|
|
describe("LegacyCallHandler without third party protocols", () => {
|
|
let dmRoomMap;
|
|
let callHandler: LegacyCallHandler;
|
|
let audioElement: HTMLAudioElement;
|
|
let fakeCall: MatrixCall | null;
|
|
|
|
const mockAudioBufferSourceNode = {
|
|
addEventListener: jest.fn(),
|
|
connect: jest.fn(),
|
|
start: jest.fn(),
|
|
stop: jest.fn(),
|
|
};
|
|
const mockAudioContext = {
|
|
decodeAudioData: jest.fn().mockResolvedValue({}),
|
|
suspend: jest.fn(),
|
|
resume: jest.fn(),
|
|
createBufferSource: jest.fn().mockReturnValue(mockAudioBufferSourceNode),
|
|
currentTime: 1337,
|
|
};
|
|
|
|
beforeEach(() => {
|
|
stubClient();
|
|
fakeCall = null;
|
|
MatrixClientPeg.safeGet().createCall = (roomId) => {
|
|
if (fakeCall && fakeCall.roomId !== roomId) {
|
|
throw new Error("Only one call is supported!");
|
|
}
|
|
fakeCall = new FakeCall(roomId) as unknown as MatrixCall;
|
|
return fakeCall;
|
|
};
|
|
|
|
MatrixClientPeg.safeGet().getThirdpartyProtocols = () => {
|
|
throw new Error("Endpoint unsupported.");
|
|
};
|
|
|
|
mocked(createAudioContext).mockReturnValue(mockAudioContext as unknown as AudioContext);
|
|
callHandler = new LegacyCallHandler();
|
|
callHandler.start();
|
|
|
|
const nativeRoomAlice = mkStubDM(NATIVE_ROOM_ALICE, NATIVE_ALICE);
|
|
|
|
MatrixClientPeg.safeGet().getRoom = (roomId: string): Room | null => {
|
|
switch (roomId) {
|
|
case NATIVE_ROOM_ALICE:
|
|
return nativeRoomAlice;
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
dmRoomMap = {
|
|
getUserIdForRoomId: (roomId: string) => {
|
|
if (roomId === NATIVE_ROOM_ALICE) {
|
|
return NATIVE_ALICE;
|
|
} else {
|
|
return null;
|
|
}
|
|
},
|
|
getDMRoomsForUserId: (userId: string) => {
|
|
if (userId === NATIVE_ALICE) {
|
|
return [NATIVE_ROOM_ALICE];
|
|
} else {
|
|
return [];
|
|
}
|
|
},
|
|
} as DMRoomMap;
|
|
DMRoomMap.setShared(dmRoomMap);
|
|
|
|
MatrixClientPeg.safeGet().getThirdpartyUser = (_proto, _params) => {
|
|
throw new Error("Endpoint unsupported.");
|
|
};
|
|
|
|
// @ts-expect-error
|
|
MatrixClientPeg.safeGet().pushProcessor = new PushProcessor(MatrixClientPeg.safeGet());
|
|
|
|
audioElement = document.createElement("audio");
|
|
audioElement.id = "remoteAudio";
|
|
document.body.appendChild(audioElement);
|
|
|
|
fetchMock.get(
|
|
"/media/ring.mp3",
|
|
{ body: new Blob(["1", "2", "3", "4"], { type: "audio/mpeg" }) },
|
|
{ sendAsJson: false },
|
|
);
|
|
});
|
|
|
|
afterEach(() => {
|
|
callHandler.stop();
|
|
// @ts-ignore
|
|
DMRoomMap.setShared(null);
|
|
// @ts-ignore
|
|
window.mxLegacyCallHandler = null;
|
|
MatrixClientPeg.unset();
|
|
|
|
document.body.removeChild(audioElement);
|
|
SdkConfig.reset();
|
|
});
|
|
|
|
it("should cache sounds between playbacks", async () => {
|
|
await callHandler.play(AudioID.Ring);
|
|
expect(mockAudioBufferSourceNode.start).toHaveBeenCalled();
|
|
expect(fetchMock.calls("/media/ring.mp3")).toHaveLength(1);
|
|
await callHandler.play(AudioID.Ring);
|
|
expect(fetchMock.calls("/media/ring.mp3")).toHaveLength(1);
|
|
});
|
|
|
|
it("should allow silencing an incoming call ring", async () => {
|
|
await callHandler.play(AudioID.Ring);
|
|
await callHandler.silenceCall("call123");
|
|
expect(mockAudioBufferSourceNode.stop).toHaveBeenCalled();
|
|
});
|
|
|
|
it("should still start a native call", async () => {
|
|
callHandler.placeCall(NATIVE_ROOM_ALICE, CallType.Voice);
|
|
|
|
// Check that a call was started: its room on the protocol level
|
|
expect(fakeCall).not.toBeNull();
|
|
expect(fakeCall!.roomId).toEqual(NATIVE_ROOM_ALICE);
|
|
|
|
// but it should appear to the user to be in thw native room for Bob
|
|
expect(callHandler.roomIdForCall(fakeCall!)).toEqual(NATIVE_ROOM_ALICE);
|
|
});
|
|
|
|
describe("incoming calls", () => {
|
|
const roomId = "test-room-id";
|
|
|
|
beforeEach(() => {
|
|
jest.clearAllMocks();
|
|
jest.spyOn(SettingsStore, "getValue").mockImplementation((setting) => setting === UIFeature.Voip);
|
|
|
|
jest.spyOn(MatrixClientPeg.safeGet(), "supportsVoip").mockReturnValue(true);
|
|
|
|
MatrixClientPeg.safeGet().isFallbackICEServerAllowed = jest.fn();
|
|
|
|
MatrixClientPeg.safeGet().pushRules = {
|
|
global: {
|
|
[PushRuleKind.Override]: [
|
|
{
|
|
rule_id: RuleId.IncomingCall,
|
|
default: false,
|
|
enabled: true,
|
|
actions: [
|
|
{
|
|
set_tweak: TweakName.Sound,
|
|
value: "ring",
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
};
|
|
|
|
// silence local notifications by default
|
|
jest.spyOn(MatrixClientPeg.safeGet(), "getAccountData").mockImplementation((eventType) => {
|
|
if (eventType.includes(LOCAL_NOTIFICATION_SETTINGS_PREFIX.name)) {
|
|
return new MatrixEvent({
|
|
type: eventType,
|
|
content: {
|
|
is_silenced: true,
|
|
},
|
|
});
|
|
}
|
|
});
|
|
});
|
|
|
|
it("listens for incoming call events when voip is enabled", () => {
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
|
|
// call added to call map
|
|
expect(callHandler.getCallForRoom(roomId)).toEqual(call);
|
|
});
|
|
|
|
it("rings when incoming call state is ringing and notifications set to ring", async () => {
|
|
// remove local notification silencing mock for this test
|
|
jest.spyOn(MatrixClientPeg.safeGet(), "getAccountData").mockReturnValue(undefined);
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
|
|
// call added to call map
|
|
expect(callHandler.getCallForRoom(roomId)).toEqual(call);
|
|
call.emit(CallEvent.State, CallState.Ringing, CallState.Connected, fakeCall!);
|
|
|
|
// ringer audio started
|
|
await waitFor(() => expect(mockAudioBufferSourceNode.start).toHaveBeenCalled());
|
|
});
|
|
|
|
it("does not ring when incoming call state is ringing but local notifications are silenced", () => {
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
|
|
// call added to call map
|
|
expect(callHandler.getCallForRoom(roomId)).toEqual(call);
|
|
call.emit(CallEvent.State, CallState.Ringing, CallState.Connected, fakeCall!);
|
|
|
|
// ringer audio element started
|
|
expect(mockAudioBufferSourceNode.start).not.toHaveBeenCalled();
|
|
expect(callHandler.isCallSilenced(call.callId)).toEqual(true);
|
|
});
|
|
|
|
it("should force calls to silent when local notifications are silenced", async () => {
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
|
|
expect(callHandler.isForcedSilent()).toEqual(true);
|
|
expect(callHandler.isCallSilenced(call.callId)).toEqual(true);
|
|
});
|
|
|
|
it("does not unsilence calls when local notifications are silenced", async () => {
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
const callHandlerEmitSpy = jest.spyOn(callHandler, "emit");
|
|
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
// reset emit call count
|
|
callHandlerEmitSpy.mockClear();
|
|
|
|
callHandler.unSilenceCall(call.callId);
|
|
expect(callHandlerEmitSpy).not.toHaveBeenCalled();
|
|
// call still silenced
|
|
expect(callHandler.isCallSilenced(call.callId)).toEqual(true);
|
|
// ringer not played
|
|
expect(mockAudioBufferSourceNode.start).not.toHaveBeenCalled();
|
|
});
|
|
});
|
|
|
|
describe("sidebar state", () => {
|
|
const roomId = "test-room-id";
|
|
|
|
it("should default to showing sidebar", () => {
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
|
|
expect(callHandler.isCallSidebarShown(call.callId)).toEqual(true);
|
|
});
|
|
|
|
it("should remember sidebar state per call", () => {
|
|
const call = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
const cli = MatrixClientPeg.safeGet();
|
|
cli.emit(CallEventHandlerEvent.Incoming, call);
|
|
|
|
expect(callHandler.isCallSidebarShown(call.callId)).toEqual(true);
|
|
callHandler.setCallSidebarShown(call.callId, false);
|
|
expect(callHandler.isCallSidebarShown(call.callId)).toEqual(false);
|
|
|
|
call.emit(CallEvent.Hangup, call);
|
|
|
|
const call2 = new MatrixCall({
|
|
client: MatrixClientPeg.safeGet(),
|
|
roomId,
|
|
});
|
|
cli.emit(CallEventHandlerEvent.Incoming, call2);
|
|
|
|
expect(callHandler.isCallSidebarShown(call2.callId)).toEqual(true);
|
|
});
|
|
});
|
|
});
|