2021-02-25 15:18:54 +00:00
|
|
|
/*
|
|
|
|
Copyright 2021 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2021-05-04 11:27:27 +00:00
|
|
|
import {ListIteratee, Many, sortBy, throttle} from "lodash";
|
2021-03-08 15:52:21 +00:00
|
|
|
import {EventType, RoomType} from "matrix-js-sdk/src/@types/event";
|
2021-02-25 15:18:54 +00:00
|
|
|
import {Room} from "matrix-js-sdk/src/models/room";
|
|
|
|
import {MatrixEvent} from "matrix-js-sdk/src/models/event";
|
|
|
|
|
|
|
|
import {AsyncStoreWithClient} from "./AsyncStoreWithClient";
|
|
|
|
import defaultDispatcher from "../dispatcher/dispatcher";
|
|
|
|
import {ActionPayload} from "../dispatcher/payloads";
|
|
|
|
import RoomListStore from "./room-list/RoomListStore";
|
|
|
|
import SettingsStore from "../settings/SettingsStore";
|
|
|
|
import DMRoomMap from "../utils/DMRoomMap";
|
|
|
|
import {FetchRoomFn} from "./notifications/ListNotificationState";
|
|
|
|
import {SpaceNotificationState} from "./notifications/SpaceNotificationState";
|
|
|
|
import {RoomNotificationStateStore} from "./notifications/RoomNotificationStateStore";
|
|
|
|
import {DefaultTagID} from "./room-list/models";
|
|
|
|
import {EnhancedMap, mapDiff} from "../utils/maps";
|
|
|
|
import {setHasDiff} from "../utils/sets";
|
2021-03-08 15:52:21 +00:00
|
|
|
import {ISpaceSummaryEvent, ISpaceSummaryRoom} from "../components/structures/SpaceRoomDirectory";
|
2021-03-24 19:58:35 +00:00
|
|
|
import RoomViewStore from "./RoomViewStore";
|
2021-02-25 15:18:54 +00:00
|
|
|
|
|
|
|
interface IState {}
|
|
|
|
|
|
|
|
const ACTIVE_SPACE_LS_KEY = "mx_active_space";
|
2021-04-20 10:39:11 +00:00
|
|
|
|
2021-03-08 15:52:21 +00:00
|
|
|
export const SUGGESTED_ROOMS = Symbol("suggested-rooms");
|
2021-02-25 15:18:54 +00:00
|
|
|
|
|
|
|
export const UPDATE_TOP_LEVEL_SPACES = Symbol("top-level-spaces");
|
2021-04-22 10:01:49 +00:00
|
|
|
export const UPDATE_INVITED_SPACES = Symbol("invited-spaces");
|
2021-02-25 15:18:54 +00:00
|
|
|
export const UPDATE_SELECTED_SPACE = Symbol("selected-space");
|
2021-05-04 17:06:43 +00:00
|
|
|
// Space Room ID will be emitted when a Space's children change
|
2021-02-25 15:18:54 +00:00
|
|
|
|
2021-03-08 15:52:21 +00:00
|
|
|
const MAX_SUGGESTED_ROOMS = 20;
|
|
|
|
|
2021-05-04 17:06:43 +00:00
|
|
|
const getSpaceContextKey = (space?: Room) => `mx_space_context_${space?.roomId || "ALL_ROOMS"}`;
|
2021-04-20 10:39:11 +00:00
|
|
|
|
2021-02-25 15:18:54 +00:00
|
|
|
const partitionSpacesAndRooms = (arr: Room[]): [Room[], Room[]] => { // [spaces, rooms]
|
|
|
|
return arr.reduce((result, room: Room) => {
|
|
|
|
result[room.isSpaceRoom() ? 0 : 1].push(room);
|
|
|
|
return result;
|
|
|
|
}, [[], []]);
|
|
|
|
};
|
|
|
|
|
2021-05-04 11:27:27 +00:00
|
|
|
// For sorting space children using a validated `order`, `m.room.create`'s `origin_server_ts`, `room_id`
|
|
|
|
export const getOrder = (order: string, creationTs: number, roomId: string): Array<Many<ListIteratee<any>>> => {
|
|
|
|
let validatedOrder: string = null;
|
|
|
|
|
|
|
|
if (typeof order === "string" && Array.from(order).every((c: string) => {
|
2021-02-25 15:18:54 +00:00
|
|
|
const charCode = c.charCodeAt(0);
|
2021-05-14 21:59:03 +00:00
|
|
|
return charCode >= 0x20 && charCode <= 0x7E;
|
2021-02-25 15:18:54 +00:00
|
|
|
})) {
|
2021-05-04 11:27:27 +00:00
|
|
|
validatedOrder = order;
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
2021-05-04 11:27:27 +00:00
|
|
|
|
|
|
|
return [validatedOrder, creationTs, roomId];
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const getRoomFn: FetchRoomFn = (room: Room) => {
|
|
|
|
return RoomNotificationStateStore.instance.getRoomState(room);
|
|
|
|
};
|
|
|
|
|
|
|
|
export class SpaceStoreClass extends AsyncStoreWithClient<IState> {
|
|
|
|
constructor() {
|
|
|
|
super(defaultDispatcher, {});
|
|
|
|
}
|
|
|
|
|
|
|
|
// The spaces representing the roots of the various tree-like hierarchies
|
|
|
|
private rootSpaces: Room[] = [];
|
|
|
|
// Map from room ID to set of spaces which list it as a child
|
|
|
|
private parentMap = new EnhancedMap<string, Set<string>>();
|
2021-05-04 17:06:43 +00:00
|
|
|
// Map from spaceId to SpaceNotificationState instance representing that space
|
|
|
|
private notificationStateMap = new Map<string, SpaceNotificationState>();
|
2021-02-25 15:18:54 +00:00
|
|
|
// Map from space key to Set of room IDs that should be shown as part of that space's filter
|
2021-05-04 17:06:43 +00:00
|
|
|
private spaceFilteredRooms = new Map<string, Set<string>>();
|
|
|
|
// The space currently selected in the Space Panel - if null then All Rooms is selected
|
2021-02-25 15:18:54 +00:00
|
|
|
private _activeSpace?: Room = null;
|
2021-03-08 15:52:21 +00:00
|
|
|
private _suggestedRooms: ISpaceSummaryRoom[] = [];
|
2021-04-22 10:01:49 +00:00
|
|
|
private _invitedSpaces = new Set<Room>();
|
|
|
|
|
|
|
|
public get invitedSpaces(): Room[] {
|
|
|
|
return Array.from(this._invitedSpaces);
|
|
|
|
}
|
2021-02-25 15:18:54 +00:00
|
|
|
|
|
|
|
public get spacePanelSpaces(): Room[] {
|
|
|
|
return this.rootSpaces;
|
|
|
|
}
|
|
|
|
|
|
|
|
public get activeSpace(): Room | null {
|
|
|
|
return this._activeSpace || null;
|
|
|
|
}
|
|
|
|
|
2021-03-08 15:52:21 +00:00
|
|
|
public get suggestedRooms(): ISpaceSummaryRoom[] {
|
|
|
|
return this._suggestedRooms;
|
|
|
|
}
|
|
|
|
|
2021-05-04 20:17:15 +00:00
|
|
|
/**
|
|
|
|
* Sets the active space, updates room list filters,
|
|
|
|
* optionally switches the user's room back to where they were when they last viewed that space.
|
|
|
|
* @param space which space to switch to.
|
|
|
|
* @param contextSwitch whether to switch the user's context,
|
|
|
|
* should not be done when the space switch is done implicitly due to another event like switching room.
|
|
|
|
*/
|
2021-04-22 07:22:11 +00:00
|
|
|
public async setActiveSpace(space: Room | null, contextSwitch = true) {
|
2021-04-24 10:32:55 +00:00
|
|
|
if (space === this.activeSpace || (space && !space?.isSpaceRoom())) return;
|
2021-02-25 15:18:54 +00:00
|
|
|
|
|
|
|
this._activeSpace = space;
|
|
|
|
this.emit(UPDATE_SELECTED_SPACE, this.activeSpace);
|
2021-03-08 15:52:21 +00:00
|
|
|
this.emit(SUGGESTED_ROOMS, this._suggestedRooms = []);
|
2021-02-25 15:18:54 +00:00
|
|
|
|
2021-04-22 07:22:11 +00:00
|
|
|
if (contextSwitch) {
|
|
|
|
// view last selected room from space
|
2021-04-22 08:05:02 +00:00
|
|
|
const roomId = window.localStorage.getItem(getSpaceContextKey(this.activeSpace));
|
2021-03-25 15:15:41 +00:00
|
|
|
|
2021-04-22 11:07:16 +00:00
|
|
|
// if the space being selected is an invite then always view that invite
|
|
|
|
// else if the last viewed room in this space is joined then view that
|
|
|
|
// else view space home or home depending on what is being clicked on
|
|
|
|
if (space?.getMyMembership !== "invite" &&
|
|
|
|
this.matrixClient?.getRoom(roomId)?.getMyMembership() === "join"
|
|
|
|
) {
|
2021-04-22 07:22:11 +00:00
|
|
|
defaultDispatcher.dispatch({
|
|
|
|
action: "view_room",
|
|
|
|
room_id: roomId,
|
2021-04-22 07:24:19 +00:00
|
|
|
context_switch: true,
|
2021-04-22 07:22:11 +00:00
|
|
|
});
|
|
|
|
} else if (space) {
|
|
|
|
defaultDispatcher.dispatch({
|
|
|
|
action: "view_room",
|
|
|
|
room_id: space.roomId,
|
2021-04-22 07:24:19 +00:00
|
|
|
context_switch: true,
|
2021-04-22 07:22:11 +00:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
defaultDispatcher.dispatch({
|
|
|
|
action: "view_home_page",
|
|
|
|
});
|
|
|
|
}
|
2021-03-25 19:51:21 +00:00
|
|
|
}
|
2021-03-25 15:15:41 +00:00
|
|
|
|
2021-02-25 15:18:54 +00:00
|
|
|
// persist space selected
|
|
|
|
if (space) {
|
|
|
|
window.localStorage.setItem(ACTIVE_SPACE_LS_KEY, space.roomId);
|
|
|
|
} else {
|
|
|
|
window.localStorage.removeItem(ACTIVE_SPACE_LS_KEY);
|
|
|
|
}
|
2021-03-08 15:52:21 +00:00
|
|
|
|
|
|
|
if (space) {
|
2021-03-16 14:18:45 +00:00
|
|
|
const data = await this.fetchSuggestedRooms(space);
|
|
|
|
if (this._activeSpace === space) {
|
|
|
|
this._suggestedRooms = data.rooms.filter(roomInfo => {
|
2021-03-26 11:48:31 +00:00
|
|
|
return roomInfo.room_type !== RoomType.Space
|
|
|
|
&& this.matrixClient.getRoom(roomInfo.room_id)?.getMyMembership() !== "join";
|
2021-03-16 14:18:45 +00:00
|
|
|
});
|
|
|
|
this.emit(SUGGESTED_ROOMS, this._suggestedRooms);
|
2021-03-08 15:52:21 +00:00
|
|
|
}
|
|
|
|
}
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
2021-03-16 14:18:45 +00:00
|
|
|
public fetchSuggestedRooms = async (space: Room, limit = MAX_SUGGESTED_ROOMS) => {
|
|
|
|
try {
|
|
|
|
const data: {
|
|
|
|
rooms: ISpaceSummaryRoom[];
|
|
|
|
events: ISpaceSummaryEvent[];
|
|
|
|
} = await this.matrixClient.getSpaceSummary(space.roomId, 0, true, false, limit);
|
|
|
|
return data;
|
|
|
|
} catch (e) {
|
|
|
|
console.error(e);
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
rooms: [],
|
|
|
|
events: [],
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
2021-03-04 13:04:58 +00:00
|
|
|
public addRoomToSpace(space: Room, roomId: string, via: string[], suggested = false, autoJoin = false) {
|
2021-02-25 15:18:54 +00:00
|
|
|
return this.matrixClient.sendStateEvent(space.roomId, EventType.SpaceChild, {
|
|
|
|
via,
|
2021-03-04 13:04:58 +00:00
|
|
|
suggested,
|
2021-02-25 15:18:54 +00:00
|
|
|
auto_join: autoJoin,
|
|
|
|
}, roomId);
|
|
|
|
}
|
|
|
|
|
|
|
|
private getChildren(spaceId: string): Room[] {
|
|
|
|
const room = this.matrixClient?.getRoom(spaceId);
|
|
|
|
const childEvents = room?.currentState.getStateEvents(EventType.SpaceChild).filter(ev => ev.getContent()?.via);
|
2021-05-04 11:27:27 +00:00
|
|
|
return sortBy(childEvents, ev => {
|
|
|
|
const roomId = ev.getStateKey();
|
|
|
|
const childRoom = this.matrixClient?.getRoom(roomId);
|
|
|
|
const createTs = childRoom?.currentState.getStateEvents(EventType.RoomCreate, "")?.getTs();
|
|
|
|
return getOrder(ev.getContent().order, createTs, roomId);
|
2021-05-05 16:30:14 +00:00
|
|
|
}).map(ev => {
|
|
|
|
return this.matrixClient.getRoom(ev.getStateKey());
|
|
|
|
}).filter(room => {
|
|
|
|
return room?.getMyMembership() === "join" || room?.getMyMembership() === "invite";
|
|
|
|
}) || [];
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public getChildRooms(spaceId: string): Room[] {
|
|
|
|
return this.getChildren(spaceId).filter(r => !r.isSpaceRoom());
|
|
|
|
}
|
|
|
|
|
|
|
|
public getChildSpaces(spaceId: string): Room[] {
|
2021-04-24 10:31:52 +00:00
|
|
|
// don't show invited subspaces as they surface at the top level for better visibility
|
|
|
|
return this.getChildren(spaceId).filter(r => r.isSpaceRoom() && r.getMyMembership() === "join");
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public getParents(roomId: string, canonicalOnly = false): Room[] {
|
|
|
|
const room = this.matrixClient?.getRoom(roomId);
|
|
|
|
return room?.currentState.getStateEvents(EventType.SpaceParent)
|
|
|
|
.filter(ev => {
|
|
|
|
const content = ev.getContent();
|
|
|
|
if (!content?.via) return false;
|
|
|
|
// TODO apply permissions check to verify that the parent mapping is valid
|
|
|
|
if (canonicalOnly && !content?.canonical) return false;
|
|
|
|
return true;
|
|
|
|
})
|
|
|
|
.map(ev => this.matrixClient.getRoom(ev.getStateKey()))
|
|
|
|
.filter(Boolean) || [];
|
|
|
|
}
|
|
|
|
|
|
|
|
public getCanonicalParent(roomId: string): Room | null {
|
|
|
|
const parents = this.getParents(roomId, true);
|
|
|
|
return sortBy(parents, r => r.roomId)?.[0] || null;
|
|
|
|
}
|
|
|
|
|
|
|
|
public getSpaceFilteredRoomIds = (space: Room | null): Set<string> => {
|
2021-05-04 17:06:43 +00:00
|
|
|
if (!space) {
|
|
|
|
return new Set(this.matrixClient.getVisibleRooms().map(r => r.roomId));
|
|
|
|
}
|
|
|
|
return this.spaceFilteredRooms.get(space.roomId) || new Set();
|
2021-02-25 15:18:54 +00:00
|
|
|
};
|
|
|
|
|
2021-03-24 14:01:24 +00:00
|
|
|
private rebuild = throttle(() => {
|
2021-04-22 10:01:49 +00:00
|
|
|
const [visibleSpaces, visibleRooms] = partitionSpacesAndRooms(this.matrixClient.getVisibleRooms());
|
|
|
|
const [joinedSpaces, invitedSpaces] = visibleSpaces.reduce((arr, s) => {
|
|
|
|
if (s.getMyMembership() === "join") {
|
|
|
|
arr[0].push(s);
|
|
|
|
} else if (s.getMyMembership() === "invite") {
|
|
|
|
arr[1].push(s);
|
|
|
|
}
|
|
|
|
return arr;
|
|
|
|
}, [[], []]);
|
2021-02-25 15:18:54 +00:00
|
|
|
|
2021-04-22 10:01:49 +00:00
|
|
|
// exclude invited spaces from unseenChildren as they will be forcibly shown at the top level of the treeview
|
|
|
|
const unseenChildren = new Set<Room>([...visibleRooms, ...joinedSpaces]);
|
2021-02-25 15:18:54 +00:00
|
|
|
const backrefs = new EnhancedMap<string, Set<string>>();
|
|
|
|
|
2021-03-24 14:01:24 +00:00
|
|
|
// Sort spaces by room ID to force the cycle breaking to be deterministic
|
2021-04-22 10:01:49 +00:00
|
|
|
const spaces = sortBy(joinedSpaces, space => space.roomId);
|
2021-03-22 17:47:48 +00:00
|
|
|
|
2021-02-25 15:18:54 +00:00
|
|
|
// TODO handle cleaning up links when a Space is removed
|
|
|
|
spaces.forEach(space => {
|
|
|
|
const children = this.getChildren(space.roomId);
|
|
|
|
children.forEach(child => {
|
|
|
|
unseenChildren.delete(child);
|
|
|
|
|
|
|
|
backrefs.getOrCreate(child.roomId, new Set()).add(space.roomId);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2021-05-04 17:06:43 +00:00
|
|
|
const [rootSpaces] = partitionSpacesAndRooms(Array.from(unseenChildren));
|
2021-02-25 15:18:54 +00:00
|
|
|
|
2021-03-24 14:01:24 +00:00
|
|
|
// somewhat algorithm to handle full-cycles
|
2021-02-25 15:18:54 +00:00
|
|
|
const detachedNodes = new Set<Room>(spaces);
|
|
|
|
|
|
|
|
const markTreeChildren = (rootSpace: Room, unseen: Set<Room>) => {
|
|
|
|
const stack = [rootSpace];
|
|
|
|
while (stack.length) {
|
|
|
|
const op = stack.pop();
|
|
|
|
unseen.delete(op);
|
|
|
|
this.getChildSpaces(op.roomId).forEach(space => {
|
|
|
|
if (unseen.has(space)) {
|
|
|
|
stack.push(space);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
rootSpaces.forEach(rootSpace => {
|
|
|
|
markTreeChildren(rootSpace, detachedNodes);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Handle spaces forming fully cyclical relationships.
|
|
|
|
// In order, assume each detachedNode is a root unless it has already
|
|
|
|
// been claimed as the child of prior detached node.
|
|
|
|
// Work from a copy of the detachedNodes set as it will be mutated as part of this operation.
|
|
|
|
Array.from(detachedNodes).forEach(detachedNode => {
|
|
|
|
if (!detachedNodes.has(detachedNode)) return;
|
|
|
|
// declare this detached node a new root, find its children, without ever looping back to it
|
|
|
|
detachedNodes.delete(detachedNode);
|
|
|
|
rootSpaces.push(detachedNode);
|
|
|
|
markTreeChildren(detachedNode, detachedNodes);
|
|
|
|
|
|
|
|
// TODO only consider a detached node a root space if it has no *parents other than the ones forming cycles
|
|
|
|
});
|
|
|
|
|
|
|
|
// TODO neither of these handle an A->B->C->A with an additional C->D
|
|
|
|
// detachedNodes.forEach(space => {
|
|
|
|
// rootSpaces.push(space);
|
|
|
|
// });
|
|
|
|
|
|
|
|
this.rootSpaces = rootSpaces;
|
|
|
|
this.parentMap = backrefs;
|
|
|
|
|
|
|
|
// if the currently selected space no longer exists, remove its selection
|
|
|
|
if (this._activeSpace && detachedNodes.has(this._activeSpace)) {
|
2021-05-04 20:17:15 +00:00
|
|
|
this.setActiveSpace(null, false);
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
this.onRoomsUpdate(); // TODO only do this if a change has happened
|
|
|
|
this.emit(UPDATE_TOP_LEVEL_SPACES, this.spacePanelSpaces);
|
2021-04-22 10:01:49 +00:00
|
|
|
|
|
|
|
// build initial state of invited spaces as we would have missed the emitted events about the room at launch
|
|
|
|
this._invitedSpaces = new Set(invitedSpaces);
|
|
|
|
this.emit(UPDATE_INVITED_SPACES, this.invitedSpaces);
|
2021-02-25 15:18:54 +00:00
|
|
|
}, 100, {trailing: true, leading: true});
|
|
|
|
|
|
|
|
onSpaceUpdate = () => {
|
|
|
|
this.rebuild();
|
|
|
|
}
|
|
|
|
|
2021-03-25 17:41:26 +00:00
|
|
|
private onSpaceMembersChange = (ev: MatrixEvent) => {
|
2021-03-25 17:37:39 +00:00
|
|
|
// skip this update if we do not have a DM with this user
|
|
|
|
if (DMRoomMap.shared().getDMRoomsForUserId(ev.getStateKey()).length < 1) return;
|
|
|
|
this.onRoomsUpdate();
|
|
|
|
};
|
|
|
|
|
2021-02-25 15:18:54 +00:00
|
|
|
private onRoomsUpdate = throttle(() => {
|
|
|
|
// TODO resolve some updates as deltas
|
|
|
|
const visibleRooms = this.matrixClient.getVisibleRooms();
|
|
|
|
|
|
|
|
const oldFilteredRooms = this.spaceFilteredRooms;
|
|
|
|
this.spaceFilteredRooms = new Map();
|
|
|
|
|
|
|
|
this.rootSpaces.forEach(s => {
|
|
|
|
// traverse each space tree in DFS to build up the supersets as you go up,
|
|
|
|
// reusing results from like subtrees.
|
|
|
|
const fn = (spaceId: string, parentPath: Set<string>): Set<string> => {
|
|
|
|
if (parentPath.has(spaceId)) return; // prevent cycles
|
|
|
|
|
|
|
|
// reuse existing results if multiple similar branches exist
|
|
|
|
if (this.spaceFilteredRooms.has(spaceId)) {
|
|
|
|
return this.spaceFilteredRooms.get(spaceId);
|
|
|
|
}
|
|
|
|
|
|
|
|
const [childSpaces, childRooms] = partitionSpacesAndRooms(this.getChildren(spaceId));
|
|
|
|
const roomIds = new Set(childRooms.map(r => r.roomId));
|
|
|
|
const space = this.matrixClient?.getRoom(spaceId);
|
|
|
|
|
|
|
|
// Add relevant DMs
|
|
|
|
space?.getJoinedMembers().forEach(member => {
|
|
|
|
DMRoomMap.shared().getDMRoomsForUserId(member.userId).forEach(roomId => {
|
|
|
|
roomIds.add(roomId);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
const newPath = new Set(parentPath).add(spaceId);
|
|
|
|
childSpaces.forEach(childSpace => {
|
|
|
|
fn(childSpace.roomId, newPath)?.forEach(roomId => {
|
|
|
|
roomIds.add(roomId);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
this.spaceFilteredRooms.set(spaceId, roomIds);
|
|
|
|
return roomIds;
|
|
|
|
};
|
|
|
|
|
|
|
|
fn(s.roomId, new Set());
|
|
|
|
});
|
|
|
|
|
|
|
|
const diff = mapDiff(oldFilteredRooms, this.spaceFilteredRooms);
|
|
|
|
// filter out keys which changed by reference only by checking whether the sets differ
|
|
|
|
const changed = diff.changed.filter(k => setHasDiff(oldFilteredRooms.get(k), this.spaceFilteredRooms.get(k)));
|
|
|
|
[...diff.added, ...diff.removed, ...changed].forEach(k => {
|
|
|
|
this.emit(k);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.spaceFilteredRooms.forEach((roomIds, s) => {
|
|
|
|
// Update NotificationStates
|
2021-05-04 15:44:29 +00:00
|
|
|
this.getNotificationState(s)?.setRooms(visibleRooms.filter(room => {
|
|
|
|
if (roomIds.has(room.roomId)) {
|
2021-05-04 17:06:43 +00:00
|
|
|
return !DMRoomMap.shared().getUserIdForRoomId(room.roomId)
|
|
|
|
|| RoomListStore.instance.getTagsForRoom(room).includes(DefaultTagID.Favourite);
|
2021-05-04 15:44:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}));
|
2021-02-25 15:18:54 +00:00
|
|
|
});
|
|
|
|
}, 100, {trailing: true, leading: true});
|
|
|
|
|
2021-04-29 12:52:48 +00:00
|
|
|
private switchToRelatedSpace = (roomId: string) => {
|
|
|
|
if (this.suggestedRooms.find(r => r.room_id === roomId)) return;
|
|
|
|
|
|
|
|
let parent = this.getCanonicalParent(roomId);
|
|
|
|
if (!parent) {
|
|
|
|
parent = this.rootSpaces.find(s => this.spaceFilteredRooms.get(s.roomId)?.has(roomId));
|
|
|
|
}
|
|
|
|
if (!parent) {
|
|
|
|
const parents = Array.from(this.parentMap.get(roomId) || []);
|
|
|
|
parent = parents.find(p => this.matrixClient.getRoom(p));
|
|
|
|
}
|
|
|
|
|
|
|
|
// don't trigger a context switch when we are switching a space to match the chosen room
|
|
|
|
this.setActiveSpace(parent || null, false);
|
|
|
|
};
|
|
|
|
|
2021-04-26 07:37:45 +00:00
|
|
|
private onRoom = (room: Room, newMembership?: string, oldMembership?: string) => {
|
|
|
|
const membership = newMembership || room.getMyMembership();
|
|
|
|
|
|
|
|
if (!room.isSpaceRoom()) {
|
2021-02-25 15:18:54 +00:00
|
|
|
// this.onRoomUpdate(room);
|
|
|
|
this.onRoomsUpdate();
|
2021-03-09 13:41:37 +00:00
|
|
|
|
2021-04-26 07:37:45 +00:00
|
|
|
if (membership === "join") {
|
|
|
|
// the user just joined a room, remove it from the suggested list if it was there
|
2021-04-12 10:45:22 +00:00
|
|
|
const numSuggestedRooms = this._suggestedRooms.length;
|
|
|
|
this._suggestedRooms = this._suggestedRooms.filter(r => r.room_id !== room.roomId);
|
|
|
|
if (numSuggestedRooms !== this._suggestedRooms.length) {
|
|
|
|
this.emit(SUGGESTED_ROOMS, this._suggestedRooms);
|
|
|
|
}
|
2021-04-29 12:52:48 +00:00
|
|
|
|
|
|
|
// if the room currently being viewed was just joined then switch to its related space
|
|
|
|
if (newMembership === "join" && room.roomId === RoomViewStore.getRoomId()) {
|
|
|
|
this.switchToRelatedSpace(room.roomId);
|
|
|
|
}
|
2021-03-26 11:48:31 +00:00
|
|
|
}
|
2021-04-26 07:37:45 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Space
|
|
|
|
if (membership === "invite") {
|
|
|
|
this._invitedSpaces.add(room);
|
|
|
|
this.emit(UPDATE_INVITED_SPACES, this.invitedSpaces);
|
|
|
|
} else if (oldMembership === "invite" && membership !== "join") {
|
|
|
|
this._invitedSpaces.delete(room);
|
|
|
|
this.emit(UPDATE_INVITED_SPACES, this.invitedSpaces);
|
|
|
|
} else {
|
|
|
|
this.onSpaceUpdate();
|
|
|
|
this.emit(room.roomId);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (membership === "join" && room.roomId === RoomViewStore.getRoomId()) {
|
|
|
|
// if the user was looking at the space and then joined: select that space
|
2021-04-29 12:52:48 +00:00
|
|
|
this.setActiveSpace(room, false);
|
2021-03-09 13:41:37 +00:00
|
|
|
}
|
2021-02-25 15:18:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
private onRoomState = (ev: MatrixEvent) => {
|
|
|
|
const room = this.matrixClient.getRoom(ev.getRoomId());
|
|
|
|
if (!room) return;
|
|
|
|
|
2021-03-25 17:37:39 +00:00
|
|
|
switch (ev.getType()) {
|
|
|
|
case EventType.SpaceChild:
|
|
|
|
if (room.isSpaceRoom()) {
|
|
|
|
this.onSpaceUpdate();
|
|
|
|
this.emit(room.roomId);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EventType.SpaceParent:
|
|
|
|
// TODO rebuild the space parent and not the room - check permissions?
|
|
|
|
// TODO confirm this after implementing parenting behaviour
|
|
|
|
if (room.isSpaceRoom()) {
|
|
|
|
this.onSpaceUpdate();
|
|
|
|
}
|
|
|
|
this.emit(room.roomId);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case EventType.RoomMember:
|
|
|
|
if (room.isSpaceRoom()) {
|
2021-03-25 17:41:26 +00:00
|
|
|
this.onSpaceMembersChange(ev);
|
2021-03-25 17:37:39 +00:00
|
|
|
}
|
|
|
|
break;
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-04-24 10:32:12 +00:00
|
|
|
protected async reset() {
|
|
|
|
this.rootSpaces = [];
|
|
|
|
this.parentMap = new EnhancedMap();
|
|
|
|
this.notificationStateMap = new Map();
|
|
|
|
this.spaceFilteredRooms = new Map();
|
|
|
|
this._activeSpace = null;
|
|
|
|
this._suggestedRooms = [];
|
|
|
|
this._invitedSpaces = new Set();
|
|
|
|
}
|
|
|
|
|
2021-02-25 15:18:54 +00:00
|
|
|
protected async onNotReady() {
|
|
|
|
if (!SettingsStore.getValue("feature_spaces")) return;
|
|
|
|
if (this.matrixClient) {
|
|
|
|
this.matrixClient.removeListener("Room", this.onRoom);
|
|
|
|
this.matrixClient.removeListener("Room.myMembership", this.onRoom);
|
|
|
|
this.matrixClient.removeListener("RoomState.events", this.onRoomState);
|
|
|
|
}
|
2021-04-24 10:32:12 +00:00
|
|
|
await this.reset();
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
protected async onReady() {
|
|
|
|
if (!SettingsStore.getValue("feature_spaces")) return;
|
|
|
|
this.matrixClient.on("Room", this.onRoom);
|
|
|
|
this.matrixClient.on("Room.myMembership", this.onRoom);
|
|
|
|
this.matrixClient.on("RoomState.events", this.onRoomState);
|
|
|
|
|
|
|
|
await this.onSpaceUpdate(); // trigger an initial update
|
|
|
|
|
|
|
|
// restore selected state from last session if any and still valid
|
|
|
|
const lastSpaceId = window.localStorage.getItem(ACTIVE_SPACE_LS_KEY);
|
|
|
|
if (lastSpaceId) {
|
2021-04-29 12:52:48 +00:00
|
|
|
this.setActiveSpace(this.matrixClient.getRoom(lastSpaceId));
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected async onAction(payload: ActionPayload) {
|
2021-03-02 15:35:02 +00:00
|
|
|
if (!SettingsStore.getValue("feature_spaces")) return;
|
2021-02-25 15:18:54 +00:00
|
|
|
switch (payload.action) {
|
|
|
|
case "view_room": {
|
2021-04-21 08:01:22 +00:00
|
|
|
// Don't auto-switch rooms when reacting to a context-switch
|
|
|
|
// as this is not helpful and can create loops of rooms/space switching
|
2021-04-29 12:52:48 +00:00
|
|
|
if (payload.context_switch) break;
|
2021-04-21 08:01:22 +00:00
|
|
|
|
2021-04-29 12:52:48 +00:00
|
|
|
const roomId = payload.room_id;
|
|
|
|
const room = this.matrixClient?.getRoom(roomId);
|
|
|
|
if (room?.isSpaceRoom()) {
|
2021-04-26 13:55:11 +00:00
|
|
|
// Don't context switch when navigating to the space room
|
|
|
|
// as it will cause you to end up in the wrong room
|
|
|
|
this.setActiveSpace(room, false);
|
2021-05-07 10:08:18 +00:00
|
|
|
} else if (this.activeSpace && !this.getSpaceFilteredRoomIds(this.activeSpace).has(roomId)) {
|
2021-04-29 12:52:48 +00:00
|
|
|
this.switchToRelatedSpace(roomId);
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
2021-04-22 07:22:11 +00:00
|
|
|
|
|
|
|
// Persist last viewed room from a space
|
|
|
|
// we don't await setActiveSpace above as we only care about this.activeSpace being up to date
|
|
|
|
// synchronously for the below code - everything else can and should be async.
|
2021-04-22 08:05:02 +00:00
|
|
|
window.localStorage.setItem(getSpaceContextKey(this.activeSpace), payload.room_id);
|
2021-02-25 15:18:54 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case "after_leave_room":
|
|
|
|
if (this._activeSpace && payload.room_id === this._activeSpace.roomId) {
|
2021-05-04 20:17:15 +00:00
|
|
|
this.setActiveSpace(null, false);
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-04 17:06:43 +00:00
|
|
|
public getNotificationState(key: string): SpaceNotificationState {
|
2021-02-25 15:18:54 +00:00
|
|
|
if (this.notificationStateMap.has(key)) {
|
|
|
|
return this.notificationStateMap.get(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
const state = new SpaceNotificationState(key, getRoomFn);
|
|
|
|
this.notificationStateMap.set(key, state);
|
|
|
|
return state;
|
|
|
|
}
|
2021-04-19 15:15:24 +00:00
|
|
|
|
2021-04-23 08:55:30 +00:00
|
|
|
// traverse space tree with DFS calling fn on each space including the given root one,
|
|
|
|
// if includeRooms is true then fn will be called on each leaf room, if it is present in multiple sub-spaces
|
|
|
|
// then fn will be called with it multiple times.
|
2021-04-19 15:15:24 +00:00
|
|
|
public traverseSpace(
|
|
|
|
spaceId: string,
|
|
|
|
fn: (roomId: string) => void,
|
|
|
|
includeRooms = false,
|
|
|
|
parentPath?: Set<string>,
|
|
|
|
) {
|
|
|
|
if (parentPath && parentPath.has(spaceId)) return; // prevent cycles
|
|
|
|
|
|
|
|
fn(spaceId);
|
|
|
|
|
|
|
|
const newPath = new Set(parentPath).add(spaceId);
|
|
|
|
const [childSpaces, childRooms] = partitionSpacesAndRooms(this.getChildren(spaceId));
|
|
|
|
|
|
|
|
if (includeRooms) {
|
|
|
|
childRooms.forEach(r => fn(r.roomId));
|
|
|
|
}
|
|
|
|
childSpaces.forEach(s => this.traverseSpace(s.roomId, fn, includeRooms, newPath));
|
|
|
|
}
|
2021-02-25 15:18:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export default class SpaceStore {
|
|
|
|
private static internalInstance = new SpaceStoreClass();
|
|
|
|
|
|
|
|
public static get instance(): SpaceStoreClass {
|
|
|
|
return SpaceStore.internalInstance;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
window.mxSpaceStore = SpaceStore.instance;
|