2023-04-03 05:19:56 +08:00
|
|
|
/*
|
|
|
|
Copyright 2023 Mikhail Aheichyk
|
|
|
|
Copyright 2023 Nordeck IT + Consulting GmbH.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import React, { ComponentProps } from "react";
|
2024-03-26 02:35:31 +08:00
|
|
|
import { cleanup, queryByRole, render, screen, within } from "@testing-library/react";
|
2023-04-03 05:19:56 +08:00
|
|
|
import userEvent from "@testing-library/user-event";
|
|
|
|
import { mocked } from "jest-mock";
|
2023-08-04 15:36:16 +08:00
|
|
|
import { Room } from "matrix-js-sdk/src/matrix";
|
2024-03-26 02:35:31 +08:00
|
|
|
import { TooltipProvider } from "@vector-im/compound-web";
|
2023-04-03 05:19:56 +08:00
|
|
|
|
|
|
|
import RoomList from "../../../../src/components/views/rooms/RoomList";
|
|
|
|
import ResizeNotifier from "../../../../src/utils/ResizeNotifier";
|
|
|
|
import { MetaSpace } from "../../../../src/stores/spaces";
|
|
|
|
import { shouldShowComponent } from "../../../../src/customisations/helpers/UIComponents";
|
|
|
|
import { UIComponent } from "../../../../src/settings/UIFeature";
|
|
|
|
import dis from "../../../../src/dispatcher/dispatcher";
|
|
|
|
import { Action } from "../../../../src/dispatcher/actions";
|
|
|
|
import * as testUtils from "../../../test-utils";
|
|
|
|
import { mkSpace, stubClient } from "../../../test-utils";
|
|
|
|
import { MatrixClientPeg } from "../../../../src/MatrixClientPeg";
|
|
|
|
import SpaceStore from "../../../../src/stores/spaces/SpaceStore";
|
|
|
|
import DMRoomMap from "../../../../src/utils/DMRoomMap";
|
2024-03-26 02:35:31 +08:00
|
|
|
import RoomListStore from "../../../../src/stores/room-list/RoomListStore";
|
|
|
|
import { ITagMap } from "../../../../src/stores/room-list/algorithms/models";
|
|
|
|
import { DefaultTagID } from "../../../../src/stores/room-list/models";
|
2023-04-03 05:19:56 +08:00
|
|
|
|
|
|
|
jest.mock("../../../../src/customisations/helpers/UIComponents", () => ({
|
|
|
|
shouldShowComponent: jest.fn(),
|
|
|
|
}));
|
|
|
|
|
|
|
|
jest.mock("../../../../src/dispatcher/dispatcher");
|
|
|
|
|
|
|
|
const getUserIdForRoomId = jest.fn();
|
|
|
|
const getDMRoomsForUserId = jest.fn();
|
|
|
|
// @ts-ignore
|
|
|
|
DMRoomMap.sharedInstance = { getUserIdForRoomId, getDMRoomsForUserId };
|
|
|
|
|
|
|
|
describe("RoomList", () => {
|
|
|
|
stubClient();
|
2023-06-06 01:12:23 +08:00
|
|
|
const client = MatrixClientPeg.safeGet();
|
2023-04-03 05:19:56 +08:00
|
|
|
const store = SpaceStore.instance;
|
|
|
|
|
|
|
|
function getComponent(props: Partial<ComponentProps<typeof RoomList>> = {}): JSX.Element {
|
|
|
|
return (
|
2024-03-26 02:35:31 +08:00
|
|
|
<TooltipProvider>
|
|
|
|
<RoomList
|
|
|
|
onKeyDown={jest.fn()}
|
|
|
|
onFocus={jest.fn()}
|
|
|
|
onBlur={jest.fn()}
|
|
|
|
onResize={jest.fn()}
|
|
|
|
resizeNotifier={new ResizeNotifier()}
|
|
|
|
isMinimized={false}
|
|
|
|
activeSpace={MetaSpace.Home}
|
|
|
|
{...props}
|
|
|
|
/>
|
|
|
|
</TooltipProvider>
|
2023-04-03 05:19:56 +08:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
describe("Rooms", () => {
|
|
|
|
describe("when meta space is active", () => {
|
|
|
|
beforeEach(() => {
|
|
|
|
store.setActiveSpace(MetaSpace.Home);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("does not render add room button when UIComponent customisation disables CreateRooms and ExploreRooms", () => {
|
|
|
|
const disabled: UIComponent[] = [UIComponent.CreateRooms, UIComponent.ExploreRooms];
|
|
|
|
mocked(shouldShowComponent).mockImplementation((feature) => !disabled.includes(feature));
|
|
|
|
render(getComponent());
|
|
|
|
|
|
|
|
const roomsList = screen.getByRole("group", { name: "Rooms" });
|
|
|
|
expect(within(roomsList).queryByRole("button", { name: "Add room" })).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("renders add room button with menu when UIComponent customisation allows CreateRooms or ExploreRooms", async () => {
|
|
|
|
let disabled: UIComponent[] = [];
|
|
|
|
mocked(shouldShowComponent).mockImplementation((feature) => !disabled.includes(feature));
|
|
|
|
const { rerender } = render(getComponent());
|
|
|
|
|
|
|
|
const roomsList = screen.getByRole("group", { name: "Rooms" });
|
|
|
|
const addRoomButton = within(roomsList).getByRole("button", { name: "Add room" });
|
|
|
|
expect(screen.queryByRole("menu")).not.toBeInTheDocument();
|
|
|
|
|
|
|
|
await userEvent.click(addRoomButton);
|
|
|
|
|
|
|
|
const menu = screen.getByRole("menu");
|
|
|
|
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "New room" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "Explore public rooms" })).toBeInTheDocument();
|
|
|
|
|
|
|
|
disabled = [UIComponent.CreateRooms];
|
|
|
|
rerender(getComponent());
|
|
|
|
|
|
|
|
expect(addRoomButton).toBeInTheDocument();
|
|
|
|
expect(menu).toBeInTheDocument();
|
|
|
|
expect(within(menu).queryByRole("menuitem", { name: "New room" })).not.toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "Explore public rooms" })).toBeInTheDocument();
|
|
|
|
|
|
|
|
disabled = [UIComponent.ExploreRooms];
|
|
|
|
rerender(getComponent());
|
|
|
|
|
|
|
|
expect(addRoomButton).toBeInTheDocument();
|
|
|
|
expect(menu).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "New room" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).queryByRole("menuitem", { name: "Explore public rooms" })).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("renders add room button and clicks explore public rooms", async () => {
|
|
|
|
mocked(shouldShowComponent).mockReturnValue(true);
|
|
|
|
render(getComponent());
|
|
|
|
|
|
|
|
const roomsList = screen.getByRole("group", { name: "Rooms" });
|
|
|
|
await userEvent.click(within(roomsList).getByRole("button", { name: "Add room" }));
|
|
|
|
|
|
|
|
const menu = screen.getByRole("menu");
|
|
|
|
await userEvent.click(within(menu).getByRole("menuitem", { name: "Explore public rooms" }));
|
|
|
|
|
|
|
|
expect(dis.fire).toHaveBeenCalledWith(Action.ViewRoomDirectory);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
describe("when room space is active", () => {
|
|
|
|
let rooms: Room[];
|
|
|
|
const mkSpaceForRooms = (spaceId: string, children: string[] = []) =>
|
|
|
|
mkSpace(client, spaceId, rooms, children);
|
|
|
|
|
|
|
|
const space1 = "!space1:server";
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
rooms = [];
|
|
|
|
mkSpaceForRooms(space1);
|
|
|
|
mocked(client).getRoom.mockImplementation(
|
|
|
|
(roomId) => rooms.find((room) => room.roomId === roomId) || null,
|
|
|
|
);
|
|
|
|
await testUtils.setupAsyncStoreWithClient(store, client);
|
|
|
|
|
|
|
|
store.setActiveSpace(space1);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("does not render add room button when UIComponent customisation disables CreateRooms and ExploreRooms", () => {
|
|
|
|
const disabled: UIComponent[] = [UIComponent.CreateRooms, UIComponent.ExploreRooms];
|
|
|
|
mocked(shouldShowComponent).mockImplementation((feature) => !disabled.includes(feature));
|
|
|
|
render(getComponent());
|
|
|
|
|
|
|
|
const roomsList = screen.getByRole("group", { name: "Rooms" });
|
|
|
|
expect(within(roomsList).queryByRole("button", { name: "Add room" })).not.toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("renders add room button with menu when UIComponent customisation allows CreateRooms or ExploreRooms", async () => {
|
|
|
|
let disabled: UIComponent[] = [];
|
|
|
|
mocked(shouldShowComponent).mockImplementation((feature) => !disabled.includes(feature));
|
|
|
|
const { rerender } = render(getComponent());
|
|
|
|
|
|
|
|
const roomsList = screen.getByRole("group", { name: "Rooms" });
|
|
|
|
const addRoomButton = within(roomsList).getByRole("button", { name: "Add room" });
|
|
|
|
expect(screen.queryByRole("menu")).not.toBeInTheDocument();
|
|
|
|
|
|
|
|
await userEvent.click(addRoomButton);
|
|
|
|
|
|
|
|
const menu = screen.getByRole("menu");
|
|
|
|
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "Explore rooms" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "New room" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "Add existing room" })).toBeInTheDocument();
|
|
|
|
|
|
|
|
disabled = [UIComponent.CreateRooms];
|
|
|
|
rerender(getComponent());
|
|
|
|
|
|
|
|
expect(addRoomButton).toBeInTheDocument();
|
|
|
|
expect(menu).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "Explore rooms" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).queryByRole("menuitem", { name: "New room" })).not.toBeInTheDocument();
|
|
|
|
expect(within(menu).queryByRole("menuitem", { name: "Add existing room" })).not.toBeInTheDocument();
|
|
|
|
|
|
|
|
disabled = [UIComponent.ExploreRooms];
|
|
|
|
rerender(getComponent());
|
|
|
|
|
|
|
|
expect(addRoomButton).toBeInTheDocument();
|
|
|
|
expect(menu).toBeInTheDocument();
|
|
|
|
expect(within(menu).queryByRole("menuitem", { name: "Explore rooms" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "New room" })).toBeInTheDocument();
|
|
|
|
expect(within(menu).getByRole("menuitem", { name: "Add existing room" })).toBeInTheDocument();
|
|
|
|
});
|
|
|
|
|
|
|
|
it("renders add room button and clicks explore rooms", async () => {
|
|
|
|
mocked(shouldShowComponent).mockReturnValue(true);
|
|
|
|
render(getComponent());
|
|
|
|
|
|
|
|
const roomsList = screen.getByRole("group", { name: "Rooms" });
|
|
|
|
await userEvent.click(within(roomsList).getByRole("button", { name: "Add room" }));
|
|
|
|
|
|
|
|
const menu = screen.getByRole("menu");
|
|
|
|
await userEvent.click(within(menu).getByRole("menuitem", { name: "Explore rooms" }));
|
|
|
|
|
|
|
|
expect(dis.dispatch).toHaveBeenCalledWith({
|
|
|
|
action: Action.ViewRoom,
|
|
|
|
room_id: space1,
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
2024-03-26 02:35:31 +08:00
|
|
|
|
|
|
|
describe("when video meta space is active", () => {
|
|
|
|
const videoRoomPrivate = "!videoRoomPrivate_server";
|
|
|
|
const videoRoomPublic = "!videoRoomPublic_server";
|
|
|
|
const videoRoomKnock = "!videoRoomKnock_server";
|
|
|
|
|
|
|
|
beforeEach(async () => {
|
|
|
|
cleanup();
|
|
|
|
const rooms: Room[] = [];
|
|
|
|
RoomListStore.instance;
|
|
|
|
testUtils.mkRoom(client, videoRoomPrivate, rooms);
|
|
|
|
testUtils.mkRoom(client, videoRoomPublic, rooms);
|
|
|
|
testUtils.mkRoom(client, videoRoomKnock, rooms);
|
|
|
|
|
|
|
|
mocked(client).getRoom.mockImplementation(
|
|
|
|
(roomId) => rooms.find((room) => room.roomId === roomId) || null,
|
|
|
|
);
|
|
|
|
mocked(client).getRooms.mockImplementation(() => rooms);
|
|
|
|
|
|
|
|
const videoRoomKnockRoom = client.getRoom(videoRoomKnock)!;
|
|
|
|
const videoRoomPrivateRoom = client.getRoom(videoRoomPrivate)!;
|
|
|
|
const videoRoomPublicRoom = client.getRoom(videoRoomPublic)!;
|
|
|
|
|
|
|
|
[videoRoomPrivateRoom, videoRoomPublicRoom, videoRoomKnockRoom].forEach((room) => {
|
|
|
|
(room.isCallRoom as jest.Mock).mockReturnValue(true);
|
|
|
|
});
|
|
|
|
|
|
|
|
const roomLists: ITagMap = {};
|
|
|
|
roomLists[DefaultTagID.Conference] = [videoRoomKnockRoom, videoRoomPublicRoom];
|
|
|
|
roomLists[DefaultTagID.Untagged] = [videoRoomPrivateRoom];
|
|
|
|
jest.spyOn(RoomListStore.instance, "orderedLists", "get").mockReturnValue(roomLists);
|
|
|
|
await testUtils.setupAsyncStoreWithClient(store, client);
|
|
|
|
|
|
|
|
store.setActiveSpace(MetaSpace.VideoRooms);
|
|
|
|
});
|
|
|
|
|
|
|
|
it("renders Conferences and Room but no People section", () => {
|
|
|
|
const renderResult = render(getComponent({ activeSpace: MetaSpace.VideoRooms }));
|
|
|
|
const roomsEl = renderResult.getByRole("treeitem", { name: "Rooms" });
|
|
|
|
const conferenceEl = renderResult.getByRole("treeitem", { name: "Conferences" });
|
|
|
|
|
|
|
|
const noInvites = screen.queryByRole("treeitem", { name: "Invites" });
|
|
|
|
const noFavourites = screen.queryByRole("treeitem", { name: "Favourites" });
|
|
|
|
const noPeople = screen.queryByRole("treeitem", { name: "People" });
|
|
|
|
const noLowPriority = screen.queryByRole("treeitem", { name: "Low priority" });
|
|
|
|
const noHistorical = screen.queryByRole("treeitem", { name: "Historical" });
|
|
|
|
|
|
|
|
expect(roomsEl).toBeVisible();
|
|
|
|
expect(conferenceEl).toBeVisible();
|
|
|
|
|
|
|
|
expect(noInvites).toBeFalsy();
|
|
|
|
expect(noFavourites).toBeFalsy();
|
|
|
|
expect(noPeople).toBeFalsy();
|
|
|
|
expect(noLowPriority).toBeFalsy();
|
|
|
|
expect(noHistorical).toBeFalsy();
|
|
|
|
});
|
|
|
|
it("renders Public and Knock rooms in Conferences section", () => {
|
|
|
|
const renderResult = render(getComponent({ activeSpace: MetaSpace.VideoRooms }));
|
|
|
|
const conferenceList = renderResult.getByRole("group", { name: "Conferences" });
|
|
|
|
expect(queryByRole(conferenceList, "treeitem", { name: videoRoomPublic })).toBeVisible();
|
|
|
|
expect(queryByRole(conferenceList, "treeitem", { name: videoRoomKnock })).toBeVisible();
|
|
|
|
expect(queryByRole(conferenceList, "treeitem", { name: videoRoomPrivate })).toBeFalsy();
|
|
|
|
|
|
|
|
const roomsList = renderResult.getByRole("group", { name: "Rooms" });
|
|
|
|
expect(queryByRole(roomsList, "treeitem", { name: videoRoomPrivate })).toBeVisible();
|
|
|
|
expect(queryByRole(roomsList, "treeitem", { name: videoRoomPublic })).toBeFalsy();
|
|
|
|
expect(queryByRole(roomsList, "treeitem", { name: videoRoomKnock })).toBeFalsy();
|
|
|
|
});
|
|
|
|
});
|
2023-04-03 05:19:56 +08:00
|
|
|
});
|
|
|
|
});
|