* chore: update @compound-web * feat(notification decoration): add NotificationDecoration component * feat(room list item): get notification state in view model * feat(room list item): use notification decoration in RoomListItemView * test(notification decoration): add tests * test(room list item view model): add a11yLabel tests * test(room list item): update tests * test(e2e): add decoration tests
97 lines
3.9 KiB
TypeScript
97 lines
3.9 KiB
TypeScript
/*
|
|
* Copyright 2025 New Vector Ltd.
|
|
*
|
|
* 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 { renderHook } from "jest-matrix-react";
|
|
import { type Room } from "matrix-js-sdk/src/matrix";
|
|
import { mocked } from "jest-mock";
|
|
|
|
import dispatcher from "../../../../../src/dispatcher/dispatcher";
|
|
import { Action } from "../../../../../src/dispatcher/actions";
|
|
import { useRoomListItemViewModel } from "../../../../../src/components/viewmodels/roomlist/RoomListItemViewModel";
|
|
import { createTestClient, mkStubRoom } from "../../../../test-utils";
|
|
import { hasAccessToOptionsMenu } from "../../../../../src/components/viewmodels/roomlist/utils";
|
|
import { RoomNotificationState } from "../../../../../src/stores/notifications/RoomNotificationState";
|
|
import { RoomNotificationStateStore } from "../../../../../src/stores/notifications/RoomNotificationStateStore";
|
|
|
|
jest.mock("../../../../../src/components/viewmodels/roomlist/utils", () => ({
|
|
hasAccessToOptionsMenu: jest.fn().mockReturnValue(false),
|
|
}));
|
|
|
|
describe("RoomListItemViewModel", () => {
|
|
let room: Room;
|
|
|
|
beforeEach(() => {
|
|
const matrixClient = createTestClient();
|
|
room = mkStubRoom("roomId", "roomName", matrixClient);
|
|
});
|
|
|
|
it("should dispatch view room action on openRoom", async () => {
|
|
const { result: vm } = renderHook(() => useRoomListItemViewModel(room));
|
|
|
|
const fn = jest.spyOn(dispatcher, "dispatch");
|
|
vm.current.openRoom();
|
|
expect(fn).toHaveBeenCalledWith(
|
|
expect.objectContaining({
|
|
action: Action.ViewRoom,
|
|
room_id: room.roomId,
|
|
metricsTrigger: "RoomList",
|
|
}),
|
|
);
|
|
});
|
|
|
|
it("should show hover menu if user has access to options menu", async () => {
|
|
mocked(hasAccessToOptionsMenu).mockReturnValue(true);
|
|
const { result: vm } = renderHook(() => useRoomListItemViewModel(room));
|
|
expect(vm.current.showHoverMenu).toBe(true);
|
|
});
|
|
|
|
describe("a11yLabel", () => {
|
|
let notificationState: RoomNotificationState;
|
|
beforeEach(() => {
|
|
notificationState = new RoomNotificationState(room, false);
|
|
jest.spyOn(RoomNotificationStateStore.instance, "getRoomState").mockReturnValue(notificationState);
|
|
});
|
|
|
|
it.each([
|
|
{
|
|
label: "unset message",
|
|
mock: () => jest.spyOn(notificationState, "isUnsetMessage", "get").mockReturnValue(true),
|
|
expected: "Open room roomName with an unset message.",
|
|
},
|
|
{
|
|
label: "invitation",
|
|
mock: () => jest.spyOn(notificationState, "invited", "get").mockReturnValue(true),
|
|
expected: "Open room roomName invitation.",
|
|
},
|
|
{
|
|
label: "mention",
|
|
mock: () => {
|
|
jest.spyOn(notificationState, "isMention", "get").mockReturnValue(true);
|
|
jest.spyOn(notificationState, "count", "get").mockReturnValue(3);
|
|
},
|
|
expected: "Open room roomName with 3 unread messages including mentions.",
|
|
},
|
|
{
|
|
label: "unread",
|
|
mock: () => {
|
|
jest.spyOn(notificationState, "hasUnreadCount", "get").mockReturnValue(true);
|
|
jest.spyOn(notificationState, "count", "get").mockReturnValue(3);
|
|
},
|
|
expected: "Open room roomName with 3 unread messages.",
|
|
},
|
|
{
|
|
label: "default",
|
|
expected: "Open room roomName",
|
|
},
|
|
])("should return the $label label", ({ mock, expected }) => {
|
|
mock?.();
|
|
const { result: vm } = renderHook(() => useRoomListItemViewModel(room));
|
|
expect(vm.current.a11yLabel).toBe(expected);
|
|
});
|
|
});
|
|
});
|