/* Copyright 2025 New Vector Ltd. Copyright 2026 Element Creations Ltd. SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial Please see LICENSE in the repository root for full details. */ import { beforeEach, describe, expect, it, type MockedFunction, vi, } from "vitest"; import { render, type RenderResult } from "@testing-library/react"; import { type LocalParticipant } from "livekit-client"; import { BehaviorSubject, of } from "rxjs"; import { BrowserRouter } from "react-router-dom"; import { TooltipProvider } from "@vector-im/compound-web"; import { RoomContext, useLocalParticipant } from "@livekit/components-react"; import userEvent from "@testing-library/user-event"; import { InCallView } from "./InCallView"; import { mockLivekitRoom, mockLocalParticipant, mockMediaDevices, mockMuteStates, mockRemoteParticipant, mockRtcMembership, type MockRTCSession, } from "../utils/test"; import { E2eeType } from "../e2ee/e2eeType"; import { getBasicCallViewModelEnvironment } from "../utils/test-viewmodel"; import { alice, local } from "../utils/test-fixtures"; import { ReactionsSenderProvider } from "../reactions/useReactionsSender"; import { useRoomEncryptionSystem } from "../e2ee/sharedKeyManagement"; import { LivekitRoomAudioRenderer } from "../livekit/MatrixAudioRenderer"; import { MediaDevicesContext } from "../MediaDevicesContext"; import { HeaderStyle } from "../UrlParams"; import { type MediaDevices as ECMediaDevices } from "../state/MediaDevices"; import { initializeWidget } from "../widget"; initializeWidget(); vi.hoisted( () => (global.ImageData = class MockImageData { public data: number[] = []; } as unknown as typeof ImageData), ); vi.mock("../soundUtils"); vi.mock("../useAudioContext"); vi.mock("../tile/GridTile"); vi.mock("../tile/SpotlightTile"); vi.mock("@livekit/components-react"); vi.mock("livekit-client/e2ee-worker?worker"); vi.mock("../e2ee/sharedKeyManagement"); vi.mock("../livekit/MatrixAudioRenderer"); vi.mock("react-use-measure", () => ({ default: (): [() => void, object] => [(): void => {}, {}], })); const localRtcMember = mockRtcMembership("@carol:example.org", "CCCC"); const localParticipant = mockLocalParticipant({ identity: "@local:example.org:AAAAAA", }); const remoteParticipant = mockRemoteParticipant({ identity: "@alice:example.org:AAAAAA", }); let useRoomEncryptionSystemMock: MockedFunction; beforeEach(() => { vi.clearAllMocks(); // MatrixAudioRenderer is tested separately. ( LivekitRoomAudioRenderer as MockedFunction ).mockImplementation((_props) => { return
mocked: MatrixAudioRenderer
; }); ( useLocalParticipant as MockedFunction ).mockImplementation( () => ({ isScreenShareEnabled: false, localParticipant: localRtcMember as unknown as LocalParticipant, }) as unknown as ReturnType, ); useRoomEncryptionSystemMock = useRoomEncryptionSystem as typeof useRoomEncryptionSystemMock; useRoomEncryptionSystemMock.mockReturnValue({ kind: E2eeType.NONE }); }); interface CreateInCallViewArgs { mediaDevices?: ECMediaDevices; } function createInCallView(args: CreateInCallViewArgs = {}): RenderResult & { rtcSession: MockRTCSession; } { const muteState = mockMuteStates(); const livekitRoom = mockLivekitRoom( { localParticipant, }, { remoteParticipants$: of([remoteParticipant]), }, ); const { vm, rtcSession } = getBasicCallViewModelEnvironment( [local, alice], undefined, { mediaDeviceOverride: args.mediaDevices }, ); rtcSession.joined = true; const room = rtcSession.room; const client = room.client; const renderResult = render( , ); return { ...renderResult, rtcSession, }; } describe("InCallView", () => { describe("rendering", () => { it("renders", () => { const { container } = createInCallView(); expect(container).toMatchSnapshot(); }); }); describe("audioOutputSwitcher", () => { it("is visible and can be clicked", async () => { const user = userEvent.setup(); const switchFn = vi.fn(); // Create mediaDevices with a speaker and an earpiece available, // with the speaker currently selected. // This is needed so that the audio switcher button is visible const available$ = new BehaviorSubject( new Map([ ["speaker-id", { type: "speaker" }], ["earpiece-id", { type: "earpiece" }], ]), ); const selected$ = new BehaviorSubject< { id: string; virtualEarpiece: boolean } | undefined >({ id: "speaker-id", virtualEarpiece: false }); const mediaDevices = mockMediaDevices({ audioOutput: { available$, selected$, select: switchFn, }, }); const { getByRole } = createInCallView({ mediaDevices }); // The button should be visible. When current output is "speaker", // the switcher targets "earpiece", so the tooltip label is "Handset". const audioOutputBtn = getByRole("button", { name: "Handset" }); expect(audioOutputBtn).toBeVisible(); await user.click(audioOutputBtn); // Clicking the button should call select -> switchFn with the earpiece device id expect(switchFn).toHaveBeenCalledWith("earpiece-id"); }); }); });