* make tiles based on rtc member * display missing lk participant + fix tile multiplier * add show_non_member_participants config option * per member tiles * merge fixes * linter * linter and tests * tests * adapt tests (wip) * Remove unused keys * Fix optionality of nonMemberItemCount * video is optional * Mock RTC members * Lint * Merge fixes * Fix user id * Add explicit types for public fields * isRTCParticipantAvailable => isLiveKitParticipantAvailable * isLiveKitParticipantAvailable * Readonly * More keys removal * Make local field based on view model class not observable * Wording * Fix RTC members in tes * Tests again * Lint * Disable showing non-member tiles by default * Duplicate screen sharing tiles like we used to * Lint * Revert function reordering * Remove throttleTime from bad merge * Cleanup * Tidy config of show non-member settings * tidy up handling of local rtc member in tests * tidy up test init * Fix mocks * Cleanup * Apply local override where participant not yet known * Handle no visible media id * Assertions for one-on-one view * Remove isLiveKitParticipantAvailable and show via encryption status * Handle no local media (yet) * Remove unused effect for setting * Tidy settings * Avoid case of one-to-one layout with missing local or remote * Iterate * Remove option to show non-member tiles to simplify code review * Remove unused code * Remove more remnants of show-non-member-tiles * iterate * back * Fix unit test * Refactor * Expose TestScheduler as global * Fix incorrect type assertion * Simplify speaking observer * Fix * Whitespace * Make it clear that we are mocking MatrixRTC memberships * Test case for only showing tiles for MatrixRTC session members * Simplify diff * Simplify diff These changes are in https://github.com/element-hq/element-call/pull/2809 * . * Whitespaces * Use asObservable when exposing subject * Show "waiting for media..." when no participant * Additional test case * Don't show "waiting for media..." in case of local participant * Make the loading state more subtle - instead of a label we show a animated gradient * Use correct key for matrix rtc foci in code comment. (#2838) * Update src/tile/SpotlightTile.tsx Co-authored-by: Timo <16718859+toger5@users.noreply.github.com> * Update src/state/CallViewModel.ts Co-authored-by: Timo <16718859+toger5@users.noreply.github.com> * Make the purpose of BaseMediaViewModel.local explicit * Use named object instead of unnamed array for spotlightAndPip * Refactor spotlightAndPip into spotlight and pip * Use if statement instead of ternary for readability in spotlight and pip logic * Review feedback * Fix tests for CallEventAudioRenderer * Lint * Revert "Make the loading state more subtle" This reverts commit 765f7b4f319b86839fcb4fde28d1e0604e542577. * Update src/state/CallViewModel.ts Co-authored-by: Timo <16718859+toger5@users.noreply.github.com> * Fix spelling * Remove a non-null assertion that failed at runtime --------- Co-authored-by: Hugh Nimmo-Smith <hughns@element.io> Co-authored-by: Hugh Nimmo-Smith <hughns@users.noreply.github.com>
138 lines
4.2 KiB
TypeScript
138 lines
4.2 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only
|
|
Please see LICENSE in the repository root for full details.
|
|
*/
|
|
|
|
import { describe, expect, it, test } from "vitest";
|
|
import { render, screen } from "@testing-library/react";
|
|
import { axe } from "vitest-axe";
|
|
import { TooltipProvider } from "@vector-im/compound-web";
|
|
import {
|
|
TrackReference,
|
|
TrackReferencePlaceholder,
|
|
} from "@livekit/components-core";
|
|
import { Track, TrackPublication } from "livekit-client";
|
|
import { type ComponentProps } from "react";
|
|
|
|
import { MediaView } from "./MediaView";
|
|
import { EncryptionStatus } from "../state/MediaViewModel";
|
|
import { mockLocalParticipant } from "../utils/test";
|
|
|
|
describe("MediaView", () => {
|
|
const participant = mockLocalParticipant({});
|
|
const trackReferencePlaceholder: TrackReferencePlaceholder = {
|
|
participant,
|
|
source: Track.Source.Camera,
|
|
};
|
|
const trackReference: TrackReference = {
|
|
...trackReferencePlaceholder,
|
|
publication: new TrackPublication(Track.Kind.Video, "id", "name"),
|
|
};
|
|
|
|
const baseProps: ComponentProps<typeof MediaView> = {
|
|
displayName: "some name",
|
|
videoEnabled: true,
|
|
videoFit: "contain",
|
|
targetWidth: 300,
|
|
targetHeight: 200,
|
|
encryptionStatus: EncryptionStatus.Connecting,
|
|
mirror: false,
|
|
unencryptedWarning: false,
|
|
video: trackReference,
|
|
member: undefined,
|
|
localParticipant: false,
|
|
};
|
|
|
|
test("is accessible", async () => {
|
|
const { container } = render(<MediaView {...baseProps} />);
|
|
expect(await axe(container)).toHaveNoViolations();
|
|
});
|
|
|
|
describe("placeholder track", () => {
|
|
test("neither video nor avatar are shown", () => {
|
|
render(<MediaView {...baseProps} video={trackReferencePlaceholder} />);
|
|
expect(screen.queryByTestId("video")).toBeNull();
|
|
expect(screen.queryAllByRole("img", { name: "some name" }).length).toBe(
|
|
0,
|
|
);
|
|
});
|
|
});
|
|
|
|
describe("with no participant", () => {
|
|
it("shows avatar for local user", () => {
|
|
render(
|
|
<MediaView {...baseProps} video={undefined} localParticipant={true} />,
|
|
);
|
|
expect(screen.getByRole("img", { name: "some name" })).toBeVisible();
|
|
expect(screen.queryAllByText("video_tile.waiting_for_media").length).toBe(
|
|
0,
|
|
);
|
|
});
|
|
it("shows avatar and label for remote user", () => {
|
|
render(
|
|
<MediaView {...baseProps} video={undefined} localParticipant={false} />,
|
|
);
|
|
expect(screen.getByRole("img", { name: "some name" })).toBeVisible();
|
|
expect(screen.getByText("video_tile.waiting_for_media")).toBeVisible();
|
|
});
|
|
});
|
|
|
|
describe("name tag", () => {
|
|
test("is shown with name", () => {
|
|
render(<MediaView {...baseProps} displayName="Bob" />);
|
|
expect(screen.getByTestId("name_tag")).toHaveTextContent("Bob");
|
|
});
|
|
});
|
|
|
|
describe("unencryptedWarning", () => {
|
|
test("is shown and accessible", async () => {
|
|
const { container } = render(
|
|
<TooltipProvider>
|
|
<MediaView {...baseProps} unencryptedWarning={true} />
|
|
</TooltipProvider>,
|
|
);
|
|
expect(await axe(container)).toHaveNoViolations();
|
|
expect(
|
|
screen.getByRole("img", { name: "common.unencrypted" }),
|
|
).toBeTruthy();
|
|
});
|
|
|
|
test("is not shown", () => {
|
|
render(
|
|
<TooltipProvider>
|
|
<MediaView {...baseProps} unencryptedWarning={false} />
|
|
</TooltipProvider>,
|
|
);
|
|
expect(
|
|
screen.queryAllByRole("img", { name: "common.unencrypted" }).length,
|
|
).toBe(0);
|
|
});
|
|
});
|
|
|
|
describe("videoEnabled", () => {
|
|
test("just video is visible", () => {
|
|
render(
|
|
<TooltipProvider>
|
|
<MediaView {...baseProps} videoEnabled={true} />
|
|
</TooltipProvider>,
|
|
);
|
|
expect(screen.getByTestId("video")).toBeVisible();
|
|
expect(screen.queryAllByRole("img", { name: "some name" }).length).toBe(
|
|
0,
|
|
);
|
|
});
|
|
|
|
test("just avatar is visible", () => {
|
|
render(
|
|
<TooltipProvider>
|
|
<MediaView {...baseProps} videoEnabled={false} />
|
|
</TooltipProvider>,
|
|
);
|
|
expect(screen.getByRole("img", { name: "some name" })).toBeVisible();
|
|
expect(screen.getByTestId("video")).not.toBeVisible();
|
|
});
|
|
});
|
|
});
|