2020-06-08 23:11:58 +00:00
|
|
|
/*
|
2021-12-01 10:50:06 +00:00
|
|
|
Copyright 2020 - 2021 The Matrix.org Foundation C.I.C.
|
2020-06-08 23:11:58 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2022-02-22 12:18:08 +00:00
|
|
|
import { Room, RoomEvent } from "matrix-js-sdk/src/models/room";
|
2021-12-01 10:50:06 +00:00
|
|
|
import { isNullOrUndefined } from "matrix-js-sdk/src/utils";
|
2022-02-22 12:18:08 +00:00
|
|
|
import { ClientEvent } from "matrix-js-sdk/src/client";
|
2021-12-01 10:50:06 +00:00
|
|
|
|
|
|
|
import SettingsStore from "../settings/SettingsStore";
|
2020-06-08 23:11:58 +00:00
|
|
|
import { AsyncStoreWithClient } from "./AsyncStoreWithClient";
|
|
|
|
import defaultDispatcher from "../dispatcher/dispatcher";
|
|
|
|
import { arrayHasDiff } from "../utils/arrays";
|
2020-07-28 17:53:43 +00:00
|
|
|
import { SettingLevel } from "../settings/SettingLevel";
|
2021-07-27 16:19:45 +00:00
|
|
|
import { Action } from "../dispatcher/actions";
|
|
|
|
import { SettingUpdatedPayload } from "../dispatcher/payloads/SettingUpdatedPayload";
|
2022-02-22 10:04:27 +00:00
|
|
|
import { ViewRoomPayload } from "../dispatcher/payloads/ViewRoomPayload";
|
2022-05-17 14:08:36 +00:00
|
|
|
import { JoinRoomPayload } from "../dispatcher/payloads/JoinRoomPayload";
|
2020-06-08 23:11:58 +00:00
|
|
|
|
|
|
|
const MAX_ROOMS = 20; // arbitrary
|
|
|
|
const AUTOJOIN_WAIT_THRESHOLD_MS = 90000; // 90s, the time we wait for an autojoined room to show up
|
|
|
|
|
|
|
|
interface IState {
|
|
|
|
enabled?: boolean;
|
|
|
|
rooms?: Room[];
|
|
|
|
}
|
|
|
|
|
|
|
|
export class BreadcrumbsStore extends AsyncStoreWithClient<IState> {
|
2022-08-30 19:13:39 +00:00
|
|
|
private static readonly internalInstance = (() => {
|
|
|
|
const instance = new BreadcrumbsStore();
|
|
|
|
instance.start();
|
|
|
|
return instance;
|
|
|
|
})();
|
2020-06-08 23:11:58 +00:00
|
|
|
|
|
|
|
private waitingRooms: { roomId: string, addedTs: number }[] = [];
|
|
|
|
|
|
|
|
private constructor() {
|
|
|
|
super(defaultDispatcher);
|
|
|
|
|
|
|
|
SettingsStore.monitorSetting("breadcrumb_rooms", null);
|
|
|
|
SettingsStore.monitorSetting("breadcrumbs", null);
|
2021-12-01 10:50:06 +00:00
|
|
|
SettingsStore.monitorSetting("feature_breadcrumbs_v2", null);
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public static get instance(): BreadcrumbsStore {
|
|
|
|
return BreadcrumbsStore.internalInstance;
|
|
|
|
}
|
|
|
|
|
|
|
|
public get rooms(): Room[] {
|
|
|
|
return this.state.rooms || [];
|
|
|
|
}
|
|
|
|
|
|
|
|
public get visible(): boolean {
|
2020-07-14 00:46:17 +00:00
|
|
|
return this.state.enabled && this.meetsRoomRequirement;
|
|
|
|
}
|
|
|
|
|
2021-12-01 10:50:06 +00:00
|
|
|
public get meetsRoomRequirement(): boolean {
|
|
|
|
if (SettingsStore.getValue("feature_breadcrumbs_v2")) return true;
|
|
|
|
return this.matrixClient?.getVisibleRooms().length >= 20;
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
2022-05-17 14:08:36 +00:00
|
|
|
protected async onAction(payload: SettingUpdatedPayload | ViewRoomPayload | JoinRoomPayload) {
|
2020-06-08 23:11:58 +00:00
|
|
|
if (!this.matrixClient) return;
|
2021-07-27 16:19:45 +00:00
|
|
|
if (payload.action === Action.SettingUpdated) {
|
2022-02-22 10:04:27 +00:00
|
|
|
if (payload.settingName === 'breadcrumb_rooms') {
|
2020-06-08 23:11:58 +00:00
|
|
|
await this.updateRooms();
|
2022-02-22 10:04:27 +00:00
|
|
|
} else if (payload.settingName === 'breadcrumbs' ||
|
|
|
|
payload.settingName === 'feature_breadcrumbs_v2'
|
2021-12-01 10:50:06 +00:00
|
|
|
) {
|
2021-06-29 12:11:58 +00:00
|
|
|
await this.updateState({ enabled: SettingsStore.getValue("breadcrumbs", null) });
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
2021-11-25 20:49:43 +00:00
|
|
|
} else if (payload.action === Action.ViewRoom) {
|
2020-06-08 23:11:58 +00:00
|
|
|
if (payload.auto_join && !this.matrixClient.getRoom(payload.room_id)) {
|
|
|
|
// Queue the room instead of pushing it immediately. We're probably just
|
|
|
|
// waiting for a room join to complete.
|
2021-06-29 12:11:58 +00:00
|
|
|
this.waitingRooms.push({ roomId: payload.room_id, addedTs: Date.now() });
|
2020-06-08 23:11:58 +00:00
|
|
|
} else {
|
2020-06-09 00:26:43 +00:00
|
|
|
// The tests might not result in a valid room object.
|
|
|
|
const room = this.matrixClient.getRoom(payload.room_id);
|
2022-05-17 14:08:36 +00:00
|
|
|
const membership = room?.getMyMembership();
|
|
|
|
if (room && membership==="join") await this.appendRoom(room);
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
2022-05-17 14:08:36 +00:00
|
|
|
} else if (payload.action === Action.JoinRoom) {
|
|
|
|
const room = this.matrixClient.getRoom(payload.roomId);
|
|
|
|
if (room) await this.appendRoom(room);
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected async onReady() {
|
|
|
|
await this.updateRooms();
|
2021-06-29 12:11:58 +00:00
|
|
|
await this.updateState({ enabled: SettingsStore.getValue("breadcrumbs", null) });
|
2020-06-08 23:11:58 +00:00
|
|
|
|
2022-02-22 12:18:08 +00:00
|
|
|
this.matrixClient.on(RoomEvent.MyMembership, this.onMyMembership);
|
|
|
|
this.matrixClient.on(ClientEvent.Room, this.onRoom);
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
protected async onNotReady() {
|
2022-02-22 12:18:08 +00:00
|
|
|
this.matrixClient.removeListener(RoomEvent.MyMembership, this.onMyMembership);
|
|
|
|
this.matrixClient.removeListener(ClientEvent.Room, this.onRoom);
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private onMyMembership = async (room: Room) => {
|
2020-07-14 00:46:17 +00:00
|
|
|
// Only turn on breadcrumbs is the user hasn't explicitly turned it off again.
|
|
|
|
const settingValueRaw = SettingsStore.getValue("breadcrumbs", null, /*excludeDefault=*/true);
|
|
|
|
if (this.meetsRoomRequirement && isNullOrUndefined(settingValueRaw)) {
|
2020-06-08 23:11:58 +00:00
|
|
|
await SettingsStore.setValue("breadcrumbs", null, SettingLevel.ACCOUNT, true);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
private onRoom = async (room: Room) => {
|
|
|
|
const waitingRoom = this.waitingRooms.find(r => r.roomId === room.roomId);
|
|
|
|
if (!waitingRoom) return;
|
|
|
|
this.waitingRooms.splice(this.waitingRooms.indexOf(waitingRoom), 1);
|
|
|
|
|
|
|
|
if ((Date.now() - waitingRoom.addedTs) > AUTOJOIN_WAIT_THRESHOLD_MS) return; // Too long ago.
|
|
|
|
await this.appendRoom(room);
|
|
|
|
};
|
|
|
|
|
|
|
|
private async updateRooms() {
|
|
|
|
let roomIds = SettingsStore.getValue("breadcrumb_rooms");
|
|
|
|
if (!roomIds || roomIds.length === 0) roomIds = [];
|
|
|
|
|
|
|
|
const rooms = roomIds.map(r => this.matrixClient.getRoom(r)).filter(r => !!r);
|
|
|
|
const currentRooms = this.state.rooms || [];
|
|
|
|
if (!arrayHasDiff(rooms, currentRooms)) return; // no change (probably echo)
|
2021-06-29 12:11:58 +00:00
|
|
|
await this.updateState({ rooms });
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
private async appendRoom(room: Room) {
|
2020-07-09 00:31:44 +00:00
|
|
|
let updated = false;
|
2020-06-09 00:18:34 +00:00
|
|
|
const rooms = (this.state.rooms || []).slice(); // cheap clone
|
2020-06-08 23:11:58 +00:00
|
|
|
|
|
|
|
// If the room is upgraded, use that room instead. We'll also splice out
|
|
|
|
// any children of the room.
|
|
|
|
const history = this.matrixClient.getRoomUpgradeHistory(room.roomId);
|
|
|
|
if (history.length > 1) {
|
|
|
|
room = history[history.length - 1]; // Last room is most recent in history
|
|
|
|
|
|
|
|
// Take out any room that isn't the most recent room
|
|
|
|
for (let i = 0; i < history.length - 1; i++) {
|
|
|
|
const idx = rooms.findIndex(r => r.roomId === history[i].roomId);
|
2020-07-09 00:31:44 +00:00
|
|
|
if (idx !== -1) {
|
|
|
|
rooms.splice(idx, 1);
|
|
|
|
updated = true;
|
|
|
|
}
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove the existing room, if it is present
|
|
|
|
const existingIdx = rooms.findIndex(r => r.roomId === room.roomId);
|
|
|
|
|
2020-07-09 00:31:44 +00:00
|
|
|
// If we're focusing on the first room no-op
|
|
|
|
if (existingIdx !== 0) {
|
|
|
|
if (existingIdx !== -1) {
|
|
|
|
rooms.splice(existingIdx, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Splice the room to the start of the list
|
|
|
|
rooms.splice(0, 0, room);
|
|
|
|
updated = true;
|
|
|
|
}
|
2020-06-08 23:11:58 +00:00
|
|
|
|
|
|
|
if (rooms.length > MAX_ROOMS) {
|
|
|
|
// This looks weird, but it's saying to start at the MAX_ROOMS point in the
|
|
|
|
// list and delete everything after it.
|
|
|
|
rooms.splice(MAX_ROOMS, rooms.length - MAX_ROOMS);
|
2020-07-09 00:31:44 +00:00
|
|
|
updated = true;
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
|
2020-07-09 00:31:44 +00:00
|
|
|
if (updated) {
|
|
|
|
// Update the breadcrumbs
|
2021-06-29 12:11:58 +00:00
|
|
|
await this.updateState({ rooms });
|
2020-07-09 00:31:44 +00:00
|
|
|
const roomIds = rooms.map(r => r.roomId);
|
|
|
|
if (roomIds.length > 0) {
|
|
|
|
await SettingsStore.setValue("breadcrumb_rooms", null, SettingLevel.ACCOUNT, roomIds);
|
|
|
|
}
|
2020-06-08 23:11:58 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|