Merge pull request #10010 from matrix-org/johannes/latest-room-in-space
Ensure room is actually in space hierarchy when resolving its latest versionpull/28788/head^2
commit
b45b933a65
|
@ -413,9 +413,18 @@ interface IHierarchyLevelProps {
|
||||||
onToggleClick?(parentId: string, childId: string): void;
|
onToggleClick?(parentId: string, childId: string): void;
|
||||||
}
|
}
|
||||||
|
|
||||||
const toLocalRoom = (cli: MatrixClient, room: IHierarchyRoom): IHierarchyRoom => {
|
export const toLocalRoom = (cli: MatrixClient, room: IHierarchyRoom, hierarchy: RoomHierarchy): IHierarchyRoom => {
|
||||||
const history = cli.getRoomUpgradeHistory(room.room_id, true);
|
const history = cli.getRoomUpgradeHistory(room.room_id, true);
|
||||||
const cliRoom = history[history.length - 1];
|
|
||||||
|
// Pick latest room that is actually part of the hierarchy
|
||||||
|
let cliRoom = null;
|
||||||
|
for (let idx = history.length - 1; idx >= 0; --idx) {
|
||||||
|
if (hierarchy.roomMap.get(history[idx].roomId)) {
|
||||||
|
cliRoom = history[idx];
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (cliRoom) {
|
if (cliRoom) {
|
||||||
return {
|
return {
|
||||||
...room,
|
...room,
|
||||||
|
@ -424,7 +433,7 @@ const toLocalRoom = (cli: MatrixClient, room: IHierarchyRoom): IHierarchyRoom =>
|
||||||
name: cliRoom.name,
|
name: cliRoom.name,
|
||||||
topic: cliRoom.currentState.getStateEvents(EventType.RoomTopic, "")?.getContent().topic,
|
topic: cliRoom.currentState.getStateEvents(EventType.RoomTopic, "")?.getContent().topic,
|
||||||
avatar_url: cliRoom.getMxcAvatarUrl(),
|
avatar_url: cliRoom.getMxcAvatarUrl(),
|
||||||
canonical_alias: cliRoom.getCanonicalAlias(),
|
canonical_alias: cliRoom.getCanonicalAlias() ?? undefined,
|
||||||
aliases: cliRoom.getAltAliases(),
|
aliases: cliRoom.getAltAliases(),
|
||||||
world_readable:
|
world_readable:
|
||||||
cliRoom.currentState.getStateEvents(EventType.RoomHistoryVisibility, "")?.getContent()
|
cliRoom.currentState.getStateEvents(EventType.RoomHistoryVisibility, "")?.getContent()
|
||||||
|
@ -461,7 +470,7 @@ export const HierarchyLevel: React.FC<IHierarchyLevelProps> = ({
|
||||||
(result, ev: IHierarchyRelation) => {
|
(result, ev: IHierarchyRelation) => {
|
||||||
const room = hierarchy.roomMap.get(ev.state_key);
|
const room = hierarchy.roomMap.get(ev.state_key);
|
||||||
if (room && roomSet.has(room)) {
|
if (room && roomSet.has(room)) {
|
||||||
result[room.room_type === RoomType.Space ? 0 : 1].push(toLocalRoom(cli, room));
|
result[room.room_type === RoomType.Space ? 0 : 1].push(toLocalRoom(cli, room, hierarchy));
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
},
|
},
|
||||||
|
|
|
@ -14,15 +14,27 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
import React from "react";
|
||||||
|
import { render } from "@testing-library/react";
|
||||||
import { MatrixClient } from "matrix-js-sdk/src/client";
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
||||||
import { Room } from "matrix-js-sdk/src/models/room";
|
import { Room } from "matrix-js-sdk/src/models/room";
|
||||||
import { RoomHierarchy } from "matrix-js-sdk/src/room-hierarchy";
|
import { RoomHierarchy } from "matrix-js-sdk/src/room-hierarchy";
|
||||||
|
import { IHierarchyRoom } from "matrix-js-sdk/src/@types/spaces";
|
||||||
|
|
||||||
import { MatrixClientPeg } from "../../../src/MatrixClientPeg";
|
import { MatrixClientPeg } from "../../../src/MatrixClientPeg";
|
||||||
import { stubClient } from "../../test-utils";
|
import { mkStubRoom, stubClient } from "../../test-utils";
|
||||||
import dispatcher from "../../../src/dispatcher/dispatcher";
|
import dispatcher from "../../../src/dispatcher/dispatcher";
|
||||||
import { showRoom } from "../../../src/components/structures/SpaceHierarchy";
|
import { HierarchyLevel, showRoom, toLocalRoom } from "../../../src/components/structures/SpaceHierarchy";
|
||||||
import { Action } from "../../../src/dispatcher/actions";
|
import { Action } from "../../../src/dispatcher/actions";
|
||||||
|
import MatrixClientContext from "../../../src/contexts/MatrixClientContext";
|
||||||
|
import DMRoomMap from "../../../src/utils/DMRoomMap";
|
||||||
|
|
||||||
|
// Fake random strings to give a predictable snapshot for checkbox IDs
|
||||||
|
jest.mock("matrix-js-sdk/src/randomstring", () => {
|
||||||
|
return {
|
||||||
|
randomString: () => "abdefghi",
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
describe("SpaceHierarchy", () => {
|
describe("SpaceHierarchy", () => {
|
||||||
describe("showRoom", () => {
|
describe("showRoom", () => {
|
||||||
|
@ -67,4 +79,115 @@ describe("SpaceHierarchy", () => {
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("toLocalRoom", () => {
|
||||||
|
stubClient();
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
const roomV1 = mkStubRoom("room-id-1", "Room V1", client);
|
||||||
|
const roomV2 = mkStubRoom("room-id-2", "Room V2", client);
|
||||||
|
const roomV3 = mkStubRoom("room-id-3", "Room V3", client);
|
||||||
|
jest.spyOn(client, "getRoomUpgradeHistory").mockReturnValue([roomV1, roomV2, roomV3]);
|
||||||
|
|
||||||
|
it("grabs last room that is in hierarchy when latest version is in hierarchy", () => {
|
||||||
|
const hierarchy = {
|
||||||
|
roomMap: new Map([
|
||||||
|
[roomV1.roomId, { room_id: roomV1.roomId } as IHierarchyRoom],
|
||||||
|
[roomV2.roomId, { room_id: roomV2.roomId } as IHierarchyRoom],
|
||||||
|
[roomV3.roomId, { room_id: roomV3.roomId } as IHierarchyRoom],
|
||||||
|
]),
|
||||||
|
} as RoomHierarchy;
|
||||||
|
const localRoomV1 = toLocalRoom(client, { room_id: roomV1.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV1.room_id).toEqual(roomV3.roomId);
|
||||||
|
const localRoomV2 = toLocalRoom(client, { room_id: roomV2.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV2.room_id).toEqual(roomV3.roomId);
|
||||||
|
const localRoomV3 = toLocalRoom(client, { room_id: roomV3.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV3.room_id).toEqual(roomV3.roomId);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("grabs last room that is in hierarchy when latest version is *not* in hierarchy", () => {
|
||||||
|
const hierarchy = {
|
||||||
|
roomMap: new Map([
|
||||||
|
[roomV1.roomId, { room_id: roomV1.roomId } as IHierarchyRoom],
|
||||||
|
[roomV2.roomId, { room_id: roomV2.roomId } as IHierarchyRoom],
|
||||||
|
]),
|
||||||
|
} as RoomHierarchy;
|
||||||
|
const localRoomV1 = toLocalRoom(client, { room_id: roomV1.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV1.room_id).toEqual(roomV2.roomId);
|
||||||
|
const localRoomV2 = toLocalRoom(client, { room_id: roomV2.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV2.room_id).toEqual(roomV2.roomId);
|
||||||
|
const localRoomV3 = toLocalRoom(client, { room_id: roomV3.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV3.room_id).toEqual(roomV2.roomId);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("returns specified room when none of the versions is in hierarchy", () => {
|
||||||
|
const hierarchy = { roomMap: new Map([]) } as RoomHierarchy;
|
||||||
|
const localRoomV1 = toLocalRoom(client, { room_id: roomV1.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV1.room_id).toEqual(roomV1.roomId);
|
||||||
|
const localRoomV2 = toLocalRoom(client, { room_id: roomV2.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV2.room_id).toEqual(roomV2.roomId);
|
||||||
|
const localRoomV3 = toLocalRoom(client, { room_id: roomV3.roomId } as IHierarchyRoom, hierarchy);
|
||||||
|
expect(localRoomV3.room_id).toEqual(roomV3.roomId);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("<HierarchyLevel />", () => {
|
||||||
|
stubClient();
|
||||||
|
const client = MatrixClientPeg.get();
|
||||||
|
|
||||||
|
const dmRoomMap = {
|
||||||
|
getUserIdForRoomId: jest.fn(),
|
||||||
|
} as unknown as DMRoomMap;
|
||||||
|
jest.spyOn(DMRoomMap, "shared").mockReturnValue(dmRoomMap);
|
||||||
|
|
||||||
|
const root = mkStubRoom("room-id-1", "Room 1", client);
|
||||||
|
const room1 = mkStubRoom("room-id-2", "Room 2", client);
|
||||||
|
const room2 = mkStubRoom("room-id-3", "Room 3", client);
|
||||||
|
|
||||||
|
const hierarchyRoot = {
|
||||||
|
room_id: root.roomId,
|
||||||
|
num_joined_members: 1,
|
||||||
|
children_state: [
|
||||||
|
{
|
||||||
|
state_key: room1.roomId,
|
||||||
|
content: { order: "1" },
|
||||||
|
},
|
||||||
|
{
|
||||||
|
state_key: room2.roomId,
|
||||||
|
content: { order: "2" },
|
||||||
|
},
|
||||||
|
],
|
||||||
|
} as IHierarchyRoom;
|
||||||
|
const hierarchyRoom1 = { room_id: room1.roomId, num_joined_members: 2 } as IHierarchyRoom;
|
||||||
|
const hierarchyRoom2 = { room_id: root.roomId, num_joined_members: 3 } as IHierarchyRoom;
|
||||||
|
|
||||||
|
const roomHierarchy = {
|
||||||
|
roomMap: new Map([
|
||||||
|
[root.roomId, hierarchyRoot],
|
||||||
|
[room1.roomId, hierarchyRoom1],
|
||||||
|
[room2.roomId, hierarchyRoom2],
|
||||||
|
]),
|
||||||
|
isSuggested: jest.fn(),
|
||||||
|
} as unknown as RoomHierarchy;
|
||||||
|
|
||||||
|
it("renders", () => {
|
||||||
|
const defaultProps = {
|
||||||
|
root: hierarchyRoot,
|
||||||
|
roomSet: new Set([hierarchyRoom1, hierarchyRoom2]),
|
||||||
|
hierarchy: roomHierarchy,
|
||||||
|
parents: new Set<string>(),
|
||||||
|
selectedMap: new Map<string, Set<string>>(),
|
||||||
|
onViewRoomClick: jest.fn(),
|
||||||
|
onJoinRoomClick: jest.fn(),
|
||||||
|
onToggleClick: jest.fn(),
|
||||||
|
};
|
||||||
|
const getComponent = (props = {}): React.ReactElement => (
|
||||||
|
<MatrixClientContext.Provider value={client}>
|
||||||
|
<HierarchyLevel {...defaultProps} {...props} />;
|
||||||
|
</MatrixClientContext.Provider>
|
||||||
|
);
|
||||||
|
|
||||||
|
const { container } = render(getComponent());
|
||||||
|
expect(container).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -0,0 +1,159 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`SpaceHierarchy <HierarchyLevel /> renders 1`] = `
|
||||||
|
<div>
|
||||||
|
<li
|
||||||
|
class="mx_SpaceHierarchy_roomTileWrapper"
|
||||||
|
role="treeitem"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_AccessibleButton mx_SpaceHierarchy_roomTile"
|
||||||
|
role="button"
|
||||||
|
tabindex="-1"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_item"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_avatar"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
class="mx_BaseAvatar mx_BaseAvatar_image"
|
||||||
|
data-testid="avatar-img"
|
||||||
|
src="http://this.is.a.url/avatar.url/room.png"
|
||||||
|
style="width: 20px; height: 20px;"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_name"
|
||||||
|
>
|
||||||
|
Unnamed Room
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_joined"
|
||||||
|
>
|
||||||
|
Joined
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_info"
|
||||||
|
>
|
||||||
|
2 members
|
||||||
|
·
|
||||||
|
<span
|
||||||
|
dir="auto"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_actions"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_AccessibleButton mx_AccessibleButton_hasKind mx_AccessibleButton_kind_primary_outline"
|
||||||
|
role="button"
|
||||||
|
tabindex="-1"
|
||||||
|
>
|
||||||
|
View
|
||||||
|
</div>
|
||||||
|
<span
|
||||||
|
class="mx_Checkbox mx_Checkbox_hasKind mx_Checkbox_kind_solid"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
id="checkbox_abdefghi"
|
||||||
|
tabindex="-1"
|
||||||
|
type="checkbox"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
for="checkbox_abdefghi"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_Checkbox_background"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_Checkbox_checkmark"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
<li
|
||||||
|
class="mx_SpaceHierarchy_roomTileWrapper"
|
||||||
|
role="treeitem"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_AccessibleButton mx_SpaceHierarchy_roomTile"
|
||||||
|
role="button"
|
||||||
|
tabindex="-1"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_item"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_avatar"
|
||||||
|
>
|
||||||
|
<img
|
||||||
|
alt=""
|
||||||
|
class="mx_BaseAvatar mx_BaseAvatar_image"
|
||||||
|
data-testid="avatar-img"
|
||||||
|
src="http://this.is.a.url/avatar.url/room.png"
|
||||||
|
style="width: 20px; height: 20px;"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_name"
|
||||||
|
>
|
||||||
|
Unnamed Room
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_joined"
|
||||||
|
>
|
||||||
|
Joined
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_roomTile_info"
|
||||||
|
>
|
||||||
|
3 members
|
||||||
|
·
|
||||||
|
<span
|
||||||
|
dir="auto"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="mx_SpaceHierarchy_actions"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_AccessibleButton mx_AccessibleButton_hasKind mx_AccessibleButton_kind_primary_outline"
|
||||||
|
role="button"
|
||||||
|
tabindex="-1"
|
||||||
|
>
|
||||||
|
View
|
||||||
|
</div>
|
||||||
|
<span
|
||||||
|
class="mx_Checkbox mx_Checkbox_hasKind mx_Checkbox_kind_solid"
|
||||||
|
>
|
||||||
|
<input
|
||||||
|
id="checkbox_abdefghi"
|
||||||
|
tabindex="-1"
|
||||||
|
type="checkbox"
|
||||||
|
/>
|
||||||
|
<label
|
||||||
|
for="checkbox_abdefghi"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_Checkbox_background"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="mx_Checkbox_checkmark"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</li>
|
||||||
|
;
|
||||||
|
</div>
|
||||||
|
`;
|
Loading…
Reference in New Issue