2017-05-24 15:56:13 +00:00
|
|
|
/*
|
|
|
|
Copyright 2017 Vector Creations Ltd
|
2018-07-03 10:16:44 +00:00
|
|
|
Copyright 2017, 2018 New Vector Ltd
|
2022-03-23 05:26:30 +00:00
|
|
|
Copyright 2019 - 2022 The Matrix.org Foundation C.I.C.
|
2017-05-24 15:56:13 +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.
|
|
|
|
*/
|
2020-07-21 09:14:12 +00:00
|
|
|
|
2021-12-14 15:34:54 +00:00
|
|
|
import React, { ReactNode } from "react";
|
2021-05-24 13:34:06 +00:00
|
|
|
import { Store } from 'flux/utils';
|
|
|
|
import { MatrixError } from "matrix-js-sdk/src/http-api";
|
2021-10-22 22:23:32 +00:00
|
|
|
import { logger } from "matrix-js-sdk/src/logger";
|
2022-02-10 14:29:55 +00:00
|
|
|
import { ViewRoom as ViewRoomEvent } from "matrix-analytics-events/types/typescript/ViewRoom";
|
2022-02-17 18:03:27 +00:00
|
|
|
import { JoinedRoom as JoinedRoomEvent } from "matrix-analytics-events/types/typescript/JoinedRoom";
|
2022-02-14 19:31:13 +00:00
|
|
|
import { JoinRule } from "matrix-js-sdk/src/@types/partials";
|
2022-02-17 18:03:27 +00:00
|
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
2022-03-24 21:50:04 +00:00
|
|
|
import { ClientEvent, MatrixClient } from "matrix-js-sdk/src/client";
|
2020-07-21 09:14:12 +00:00
|
|
|
|
|
|
|
import dis from '../dispatcher/dispatcher';
|
2021-05-24 13:34:06 +00:00
|
|
|
import { MatrixClientPeg } from '../MatrixClientPeg';
|
2017-06-02 10:53:10 +00:00
|
|
|
import Modal from '../Modal';
|
|
|
|
import { _t } from '../languageHandler';
|
2019-11-12 11:43:18 +00:00
|
|
|
import { getCachedRoomIDForAlias, storeRoomAliasInCache } from '../RoomAliasCache';
|
2021-05-24 13:34:06 +00:00
|
|
|
import { ActionPayload } from "../dispatcher/payloads";
|
|
|
|
import { Action } from "../dispatcher/actions";
|
|
|
|
import { retry } from "../utils/promise";
|
2021-10-19 15:05:34 +00:00
|
|
|
import { TimelineRenderingType } from "../contexts/RoomContext";
|
2022-02-10 14:29:55 +00:00
|
|
|
import { PosthogAnalytics } from "../PosthogAnalytics";
|
|
|
|
import { ViewRoomPayload } from "../dispatcher/payloads/ViewRoomPayload";
|
2022-02-14 19:31:13 +00:00
|
|
|
import DMRoomMap from "../utils/DMRoomMap";
|
|
|
|
import SpaceStore from "./spaces/SpaceStore";
|
|
|
|
import { isMetaSpace, MetaSpace } from "./spaces";
|
2022-02-17 18:03:27 +00:00
|
|
|
import { JoinRoomPayload } from "../dispatcher/payloads/JoinRoomPayload";
|
|
|
|
import { JoinRoomReadyPayload } from "../dispatcher/payloads/JoinRoomReadyPayload";
|
2022-02-22 10:04:27 +00:00
|
|
|
import { JoinRoomErrorPayload } from "../dispatcher/payloads/JoinRoomErrorPayload";
|
|
|
|
import { ViewRoomErrorPayload } from "../dispatcher/payloads/ViewRoomErrorPayload";
|
2022-03-02 23:33:40 +00:00
|
|
|
import ErrorDialog from "../components/views/dialogs/ErrorDialog";
|
2022-03-24 21:50:04 +00:00
|
|
|
import { ActiveRoomChangedPayload } from "../dispatcher/payloads/ActiveRoomChangedPayload";
|
2021-09-21 15:48:09 +00:00
|
|
|
|
2020-09-14 14:16:29 +00:00
|
|
|
const NUM_JOIN_RETRY = 5;
|
2017-05-24 15:56:13 +00:00
|
|
|
|
|
|
|
const INITIAL_STATE = {
|
2017-09-15 14:07:09 +00:00
|
|
|
// Whether we're joining the currently viewed room (see isJoining())
|
2017-05-24 15:56:13 +00:00
|
|
|
joining: false,
|
2017-06-08 15:00:12 +00:00
|
|
|
// Any error that has occurred during joining
|
2017-05-24 15:56:13 +00:00
|
|
|
joinError: null,
|
2017-06-08 15:00:12 +00:00
|
|
|
// The room ID of the room currently being viewed
|
2017-05-24 15:56:13 +00:00
|
|
|
roomId: null,
|
2017-06-08 14:47:41 +00:00
|
|
|
|
2017-06-08 16:27:04 +00:00
|
|
|
// The event to scroll to when the room is first viewed
|
2017-06-08 14:47:41 +00:00
|
|
|
initialEventId: null,
|
2020-07-21 09:14:12 +00:00
|
|
|
initialEventPixelOffset: null,
|
2017-06-08 14:47:41 +00:00
|
|
|
// Whether to highlight the initial event
|
|
|
|
isInitialEventHighlighted: false,
|
|
|
|
|
2017-05-24 15:56:13 +00:00
|
|
|
// The room alias of the room (or null if not originally specified in view_room)
|
|
|
|
roomAlias: null,
|
|
|
|
// Whether the current room is loading
|
|
|
|
roomLoading: false,
|
|
|
|
// Any error that has occurred during loading
|
|
|
|
roomLoadError: null,
|
2017-06-16 15:12:52 +00:00
|
|
|
|
2020-07-21 09:14:12 +00:00
|
|
|
replyingToEvent: null,
|
|
|
|
|
|
|
|
shouldPeek: false,
|
2021-04-28 08:04:02 +00:00
|
|
|
|
|
|
|
viaServers: [],
|
2021-04-29 08:37:21 +00:00
|
|
|
|
|
|
|
wasContextSwitch: false,
|
2017-05-24 15:56:13 +00:00
|
|
|
};
|
|
|
|
|
2022-03-23 05:29:02 +00:00
|
|
|
type Listener = (isActive: boolean) => void;
|
|
|
|
|
2017-05-24 15:56:13 +00:00
|
|
|
/**
|
|
|
|
* A class for storing application state for RoomView. This is the RoomView's interface
|
|
|
|
* with a subset of the js-sdk.
|
|
|
|
* ```
|
|
|
|
*/
|
2022-03-23 05:26:30 +00:00
|
|
|
export class RoomViewStore extends Store<ActionPayload> {
|
2022-03-23 05:29:02 +00:00
|
|
|
// Important: This cannot be a dynamic getter (lazily-constructed instance) because
|
|
|
|
// otherwise we'll miss view_room dispatches during startup, breaking relaunches of
|
|
|
|
// the app. We need to eagerly create the instance.
|
2022-03-23 05:26:30 +00:00
|
|
|
public static readonly instance = new RoomViewStore();
|
|
|
|
|
2020-07-21 09:14:12 +00:00
|
|
|
private state = INITIAL_STATE; // initialize state
|
|
|
|
|
2022-03-23 05:29:02 +00:00
|
|
|
// Keep these out of state to avoid causing excessive/recursive updates
|
|
|
|
private roomIdActivityListeners: Record<string, Listener[]> = {};
|
|
|
|
|
2022-03-23 05:26:30 +00:00
|
|
|
public constructor() {
|
2019-01-17 09:29:37 +00:00
|
|
|
super(dis);
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2022-03-23 05:29:02 +00:00
|
|
|
public addRoomListener(roomId: string, fn: Listener) {
|
|
|
|
if (!this.roomIdActivityListeners[roomId]) this.roomIdActivityListeners[roomId] = [];
|
|
|
|
this.roomIdActivityListeners[roomId].push(fn);
|
|
|
|
}
|
|
|
|
|
|
|
|
public removeRoomListener(roomId: string, fn: Listener) {
|
|
|
|
if (this.roomIdActivityListeners[roomId]) {
|
|
|
|
const i = this.roomIdActivityListeners[roomId].indexOf(fn);
|
|
|
|
if (i > -1) {
|
|
|
|
this.roomIdActivityListeners[roomId].splice(i, 1);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
logger.warn("Unregistering unrecognised listener (roomId=" + roomId + ")");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private emitForRoom(roomId: string, isActive: boolean) {
|
|
|
|
if (!this.roomIdActivityListeners[roomId]) return;
|
|
|
|
|
|
|
|
for (const fn of this.roomIdActivityListeners[roomId]) {
|
|
|
|
fn.call(null, isActive);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-23 05:26:30 +00:00
|
|
|
private setState(newState: Partial<typeof INITIAL_STATE>) {
|
2020-02-06 17:57:17 +00:00
|
|
|
// If values haven't changed, there's nothing to do.
|
|
|
|
// This only tries a shallow comparison, so unchanged objects will slip
|
|
|
|
// through, but that's probably okay for now.
|
|
|
|
let stateChanged = false;
|
|
|
|
for (const key of Object.keys(newState)) {
|
2020-07-21 09:14:12 +00:00
|
|
|
if (this.state[key] !== newState[key]) {
|
2020-02-06 17:57:17 +00:00
|
|
|
stateChanged = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!stateChanged) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2022-03-23 05:29:02 +00:00
|
|
|
const lastRoomId = this.state.roomId;
|
2020-07-21 09:14:12 +00:00
|
|
|
this.state = Object.assign(this.state, newState);
|
2022-03-23 05:29:02 +00:00
|
|
|
if (lastRoomId !== this.state.roomId) {
|
|
|
|
if (lastRoomId) this.emitForRoom(lastRoomId, false);
|
|
|
|
if (this.state.roomId) this.emitForRoom(this.state.roomId, true);
|
2022-03-24 21:50:04 +00:00
|
|
|
|
|
|
|
// Fired so we can reduce dependency on event emitters to this store, which is relatively
|
|
|
|
// central to the application and can easily cause import cycles.
|
2022-04-01 01:38:00 +00:00
|
|
|
dis.dispatch<ActiveRoomChangedPayload>({
|
2022-03-24 21:50:04 +00:00
|
|
|
action: Action.ActiveRoomChanged,
|
|
|
|
oldRoomId: lastRoomId,
|
|
|
|
newRoomId: this.state.roomId,
|
2022-04-01 01:38:00 +00:00
|
|
|
});
|
2022-03-23 05:29:02 +00:00
|
|
|
}
|
|
|
|
|
2017-05-24 15:56:13 +00:00
|
|
|
this.__emitChange();
|
|
|
|
}
|
|
|
|
|
2022-03-23 05:26:30 +00:00
|
|
|
protected __onDispatch(payload) { // eslint-disable-line @typescript-eslint/naming-convention
|
2017-05-24 15:56:13 +00:00
|
|
|
switch (payload.action) {
|
|
|
|
// view_room:
|
2017-06-08 13:17:49 +00:00
|
|
|
// - room_alias: '#somealias:matrix.org'
|
|
|
|
// - room_id: '!roomid123:matrix.org'
|
|
|
|
// - event_id: '$213456782:matrix.org'
|
|
|
|
// - event_offset: 100
|
|
|
|
// - highlighted: true
|
2021-11-25 20:49:43 +00:00
|
|
|
case Action.ViewRoom:
|
2020-07-21 09:14:12 +00:00
|
|
|
this.viewRoom(payload);
|
2017-05-24 15:56:13 +00:00
|
|
|
break;
|
2020-07-21 09:22:03 +00:00
|
|
|
// for these events blank out the roomId as we are no longer in the RoomView
|
2020-07-21 09:20:30 +00:00
|
|
|
case 'view_welcome_page':
|
2022-02-22 10:04:27 +00:00
|
|
|
case Action.ViewHomePage:
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState({
|
2017-10-24 15:32:52 +00:00
|
|
|
roomId: null,
|
|
|
|
roomAlias: null,
|
2021-04-28 08:04:02 +00:00
|
|
|
viaServers: [],
|
2021-04-29 08:37:21 +00:00
|
|
|
wasContextSwitch: false,
|
2017-10-24 15:32:52 +00:00
|
|
|
});
|
|
|
|
break;
|
2022-02-22 10:04:27 +00:00
|
|
|
case Action.ViewRoomError:
|
2020-07-21 09:14:12 +00:00
|
|
|
this.viewRoomError(payload);
|
2017-06-01 17:01:30 +00:00
|
|
|
break;
|
2017-05-25 16:04:42 +00:00
|
|
|
case 'will_join':
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState({
|
2017-05-25 16:04:42 +00:00
|
|
|
joining: true,
|
|
|
|
});
|
|
|
|
break;
|
|
|
|
case 'cancel_join':
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState({
|
2017-05-25 16:04:42 +00:00
|
|
|
joining: false,
|
|
|
|
});
|
|
|
|
break;
|
2017-05-24 15:56:13 +00:00
|
|
|
// join_room:
|
|
|
|
// - opts: options for joinRoom
|
2021-05-24 13:34:06 +00:00
|
|
|
case Action.JoinRoom:
|
2020-07-21 09:14:12 +00:00
|
|
|
this.joinRoom(payload);
|
2017-05-24 15:56:13 +00:00
|
|
|
break;
|
2021-05-24 13:34:06 +00:00
|
|
|
case Action.JoinRoomError:
|
2020-07-21 09:14:12 +00:00
|
|
|
this.joinRoomError(payload);
|
2017-06-02 10:53:10 +00:00
|
|
|
break;
|
2022-02-17 18:03:27 +00:00
|
|
|
case Action.JoinRoomReady: {
|
2022-01-07 09:23:54 +00:00
|
|
|
if (this.state.roomId === payload.roomId) {
|
|
|
|
this.setState({ shouldPeek: false });
|
|
|
|
}
|
2022-02-17 18:03:27 +00:00
|
|
|
|
2022-04-01 01:43:17 +00:00
|
|
|
const cli = MatrixClientPeg.get();
|
2022-02-17 18:03:27 +00:00
|
|
|
|
|
|
|
const updateMetrics = () => {
|
|
|
|
const room = cli.getRoom(payload.roomId);
|
|
|
|
const numMembers = room.getJoinedMemberCount();
|
|
|
|
const roomSize = numMembers > 1000 ? "MoreThanAThousand"
|
|
|
|
: numMembers > 100 ? "OneHundredAndOneToAThousand"
|
|
|
|
: numMembers > 10 ? "ElevenToOneHundred"
|
|
|
|
: numMembers > 2 ? "ThreeToTen"
|
|
|
|
: numMembers > 1 ? "Two"
|
|
|
|
: "One";
|
|
|
|
|
|
|
|
PosthogAnalytics.instance.trackEvent<JoinedRoomEvent>({
|
|
|
|
eventName: "JoinedRoom",
|
|
|
|
trigger: payload.metricsTrigger,
|
|
|
|
roomSize,
|
|
|
|
isDM: !!DMRoomMap.shared().getUserIdForRoomId(room.roomId),
|
|
|
|
isSpace: room.isSpaceRoom(),
|
|
|
|
});
|
|
|
|
|
2022-02-22 12:18:08 +00:00
|
|
|
cli.off(ClientEvent.Room, updateMetrics);
|
2022-02-17 18:03:27 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if (cli.getRoom(payload.roomId)) {
|
|
|
|
updateMetrics();
|
|
|
|
} else {
|
2022-02-22 12:18:08 +00:00
|
|
|
cli.on(ClientEvent.Room, updateMetrics);
|
2022-02-17 18:03:27 +00:00
|
|
|
}
|
|
|
|
|
2020-03-31 09:37:56 +00:00
|
|
|
break;
|
2022-02-17 18:03:27 +00:00
|
|
|
}
|
2019-07-03 22:46:37 +00:00
|
|
|
case 'on_client_not_viable':
|
2017-05-25 16:16:16 +00:00
|
|
|
case 'on_logged_out':
|
|
|
|
this.reset();
|
|
|
|
break;
|
2018-02-19 23:41:07 +00:00
|
|
|
case 'reply_to_event':
|
2019-09-09 08:34:08 +00:00
|
|
|
// If currently viewed room does not match the room in which we wish to reply then change rooms
|
2022-03-11 15:40:49 +00:00
|
|
|
// this can happen when performing a search across all rooms. Persist the data from this event for
|
|
|
|
// both room and search timeline rendering types, search will get auto-closed by RoomView at this time.
|
|
|
|
if ([TimelineRenderingType.Room, TimelineRenderingType.Search].includes(payload.context)) {
|
2022-03-11 17:21:28 +00:00
|
|
|
if (payload.event && payload.event.getRoomId() !== this.state.roomId) {
|
2022-02-10 14:29:55 +00:00
|
|
|
dis.dispatch<ViewRoomPayload>({
|
2021-11-25 20:49:43 +00:00
|
|
|
action: Action.ViewRoom,
|
2021-10-19 15:05:34 +00:00
|
|
|
room_id: payload.event.getRoomId(),
|
|
|
|
replyingToEvent: payload.event,
|
2022-02-17 18:03:27 +00:00
|
|
|
metricsTrigger: undefined, // room doesn't change
|
2021-10-19 15:05:34 +00:00
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.setState({
|
|
|
|
replyingToEvent: payload.event,
|
|
|
|
});
|
|
|
|
}
|
2019-09-09 08:34:08 +00:00
|
|
|
}
|
2018-02-19 23:41:07 +00:00
|
|
|
break;
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-10 14:29:55 +00:00
|
|
|
private async viewRoom(payload: ViewRoomPayload): Promise<void> {
|
2017-06-01 17:01:30 +00:00
|
|
|
if (payload.room_id) {
|
2022-02-17 18:03:27 +00:00
|
|
|
if (payload.metricsTrigger !== null && payload.room_id !== this.state.roomId) {
|
2022-02-14 19:31:13 +00:00
|
|
|
let activeSpace: ViewRoomEvent["activeSpace"];
|
|
|
|
if (SpaceStore.instance.activeSpace === MetaSpace.Home) {
|
|
|
|
activeSpace = "Home";
|
|
|
|
} else if (isMetaSpace(SpaceStore.instance.activeSpace)) {
|
|
|
|
activeSpace = "Meta";
|
|
|
|
} else {
|
|
|
|
activeSpace = SpaceStore.instance.activeSpaceRoom.getJoinRule() === JoinRule.Public
|
|
|
|
? "Public"
|
|
|
|
: "Private";
|
|
|
|
}
|
|
|
|
|
2022-02-10 14:29:55 +00:00
|
|
|
PosthogAnalytics.instance.trackEvent<ViewRoomEvent>({
|
|
|
|
eventName: "ViewRoom",
|
2022-02-17 18:03:27 +00:00
|
|
|
trigger: payload.metricsTrigger,
|
|
|
|
viaKeyboard: payload.metricsViaKeyboard,
|
2022-02-14 19:31:13 +00:00
|
|
|
isDM: !!DMRoomMap.shared().getUserIdForRoomId(payload.room_id),
|
2022-04-01 01:43:17 +00:00
|
|
|
isSpace: MatrixClientPeg.get().getRoom(payload.room_id)?.isSpaceRoom(),
|
2022-02-14 19:31:13 +00:00
|
|
|
activeSpace,
|
2022-02-10 14:29:55 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-06-08 14:54:23 +00:00
|
|
|
const newState = {
|
2017-06-01 17:01:30 +00:00
|
|
|
roomId: payload.room_id,
|
2017-06-14 11:05:25 +00:00
|
|
|
roomAlias: payload.room_alias,
|
2017-06-08 14:47:41 +00:00
|
|
|
initialEventId: payload.event_id,
|
|
|
|
isInitialEventHighlighted: payload.highlighted,
|
2017-06-02 08:22:48 +00:00
|
|
|
roomLoading: false,
|
|
|
|
roomLoadError: null,
|
2017-06-16 17:24:07 +00:00
|
|
|
// should peek by default
|
|
|
|
shouldPeek: payload.should_peek === undefined ? true : payload.should_peek,
|
2017-09-14 21:22:21 +00:00
|
|
|
// have we sent a join request for this room and are waiting for a response?
|
|
|
|
joining: payload.joining || false,
|
2018-02-19 23:41:07 +00:00
|
|
|
// Reset replyingToEvent because we don't want cross-room because bad UX
|
|
|
|
replyingToEvent: null,
|
2021-04-28 08:04:02 +00:00
|
|
|
viaServers: payload.via_servers,
|
2021-04-29 08:37:21 +00:00
|
|
|
wasContextSwitch: payload.context_switch,
|
2017-06-08 14:54:23 +00:00
|
|
|
};
|
2019-01-17 09:29:37 +00:00
|
|
|
|
2019-09-09 08:34:08 +00:00
|
|
|
// Allow being given an event to be replied to when switching rooms but sanity check its for this room
|
2022-02-10 14:29:55 +00:00
|
|
|
if (payload.replyingToEvent?.getRoomId() === payload.room_id) {
|
2019-09-09 08:34:08 +00:00
|
|
|
newState.replyingToEvent = payload.replyingToEvent;
|
2022-03-23 13:38:21 +00:00
|
|
|
} else if (this.state.roomId === payload.room_id) {
|
|
|
|
// if the room isn't being changed, e.g visiting a permalink then maintain replyingToEvent
|
|
|
|
newState.replyingToEvent = this.state.replyingToEvent;
|
2019-09-09 08:34:08 +00:00
|
|
|
}
|
|
|
|
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState(newState);
|
2019-01-17 09:29:37 +00:00
|
|
|
|
2017-09-14 22:06:00 +00:00
|
|
|
if (payload.auto_join) {
|
2022-02-17 18:03:27 +00:00
|
|
|
dis.dispatch<JoinRoomPayload>({
|
2021-05-24 13:34:06 +00:00
|
|
|
...payload,
|
|
|
|
action: Action.JoinRoom,
|
|
|
|
roomId: payload.room_id,
|
2022-02-17 18:03:27 +00:00
|
|
|
metricsTrigger: payload.metricsTrigger as JoinRoomPayload["metricsTrigger"],
|
2021-05-24 13:34:06 +00:00
|
|
|
});
|
2017-09-14 22:06:00 +00:00
|
|
|
}
|
2017-06-02 08:22:48 +00:00
|
|
|
} else if (payload.room_alias) {
|
2019-11-12 11:43:18 +00:00
|
|
|
// Try the room alias to room ID navigation cache first to avoid
|
|
|
|
// blocking room navigation on the homeserver.
|
2019-11-12 13:29:01 +00:00
|
|
|
let roomId = getCachedRoomIDForAlias(payload.room_alias);
|
|
|
|
if (!roomId) {
|
|
|
|
// Room alias cache miss, so let's ask the homeserver. Resolve the alias
|
|
|
|
// and then do a second dispatch with the room ID acquired.
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState({
|
2019-11-12 13:29:01 +00:00
|
|
|
roomId: null,
|
|
|
|
initialEventId: null,
|
|
|
|
initialEventPixelOffset: null,
|
|
|
|
isInitialEventHighlighted: null,
|
|
|
|
roomAlias: payload.room_alias,
|
|
|
|
roomLoading: true,
|
|
|
|
roomLoadError: null,
|
2021-04-28 08:04:02 +00:00
|
|
|
viaServers: payload.via_servers,
|
2021-04-29 08:37:21 +00:00
|
|
|
wasContextSwitch: payload.context_switch,
|
2019-11-12 11:43:18 +00:00
|
|
|
});
|
2019-11-12 13:29:01 +00:00
|
|
|
try {
|
2022-04-01 01:43:17 +00:00
|
|
|
const result = await MatrixClientPeg.get().getRoomIdForAlias(payload.room_alias);
|
2019-11-12 13:29:01 +00:00
|
|
|
storeRoomAliasInCache(payload.room_alias, result.room_id);
|
|
|
|
roomId = result.room_id;
|
|
|
|
} catch (err) {
|
2021-10-15 14:30:53 +00:00
|
|
|
logger.error("RVS failed to get room id for alias: ", err);
|
2022-02-22 10:04:27 +00:00
|
|
|
dis.dispatch<ViewRoomErrorPayload>({
|
|
|
|
action: Action.ViewRoomError,
|
2019-11-12 13:29:01 +00:00
|
|
|
room_id: null,
|
|
|
|
room_alias: payload.room_alias,
|
|
|
|
err,
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
2019-11-12 11:43:18 +00:00
|
|
|
}
|
2019-11-12 13:29:01 +00:00
|
|
|
|
2022-01-12 20:12:28 +00:00
|
|
|
// Re-fire the payload with the newly found room_id
|
2019-11-12 13:29:01 +00:00
|
|
|
dis.dispatch({
|
2022-01-12 20:12:28 +00:00
|
|
|
...payload,
|
2019-11-12 13:29:01 +00:00
|
|
|
room_id: roomId,
|
2017-05-24 15:56:13 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-22 10:04:27 +00:00
|
|
|
private viewRoomError(payload: ViewRoomErrorPayload) {
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState({
|
2017-06-01 17:01:30 +00:00
|
|
|
roomId: payload.room_id,
|
|
|
|
roomAlias: payload.room_alias,
|
|
|
|
roomLoading: false,
|
|
|
|
roomLoadError: payload.err,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2022-02-17 18:03:27 +00:00
|
|
|
private async joinRoom(payload: JoinRoomPayload) {
|
2020-07-21 09:14:12 +00:00
|
|
|
this.setState({
|
2017-05-24 15:56:13 +00:00
|
|
|
joining: true,
|
|
|
|
});
|
2020-09-14 14:16:29 +00:00
|
|
|
|
2022-04-01 01:43:17 +00:00
|
|
|
const cli = MatrixClientPeg.get();
|
2022-01-07 09:23:54 +00:00
|
|
|
// take a copy of roomAlias & roomId as they may change by the time the join is complete
|
|
|
|
const { roomAlias, roomId } = this.state;
|
|
|
|
const address = roomAlias || roomId;
|
2021-04-28 08:07:02 +00:00
|
|
|
const viaServers = this.state.viaServers || [];
|
2020-09-14 14:16:29 +00:00
|
|
|
try {
|
2022-02-17 18:03:27 +00:00
|
|
|
await retry<Room, MatrixError>(() => cli.joinRoom(address, {
|
2021-04-28 08:04:02 +00:00
|
|
|
viaServers,
|
2022-02-17 18:03:27 +00:00
|
|
|
...(payload.opts || {}),
|
2021-03-24 16:45:53 +00:00
|
|
|
}), NUM_JOIN_RETRY, (err) => {
|
2020-09-14 14:16:29 +00:00
|
|
|
// if we received a Gateway timeout then retry
|
|
|
|
return err.httpStatus === 504;
|
|
|
|
});
|
2022-02-10 14:29:55 +00:00
|
|
|
|
2020-03-31 09:37:56 +00:00
|
|
|
// We do *not* clear the 'joining' flag because the Room object and/or our 'joined' member event may not
|
|
|
|
// have come down the sync stream yet, and that's the point at which we'd consider the user joined to the
|
|
|
|
// room.
|
2022-02-17 18:03:27 +00:00
|
|
|
dis.dispatch<JoinRoomReadyPayload>({
|
2021-05-24 13:34:06 +00:00
|
|
|
action: Action.JoinRoomReady,
|
2022-01-07 09:23:54 +00:00
|
|
|
roomId,
|
2022-02-17 18:03:27 +00:00
|
|
|
metricsTrigger: payload.metricsTrigger,
|
2021-05-24 13:34:06 +00:00
|
|
|
});
|
2020-09-14 14:16:29 +00:00
|
|
|
} catch (err) {
|
2019-01-17 09:29:37 +00:00
|
|
|
dis.dispatch({
|
2021-05-24 13:34:06 +00:00
|
|
|
action: Action.JoinRoomError,
|
2022-01-07 09:23:54 +00:00
|
|
|
roomId,
|
2017-06-02 10:53:10 +00:00
|
|
|
err: err,
|
|
|
|
});
|
2020-09-14 14:16:29 +00:00
|
|
|
}
|
2017-06-02 10:53:10 +00:00
|
|
|
}
|
|
|
|
|
2022-03-23 05:26:30 +00:00
|
|
|
private getInvitingUserId(roomId: string): string {
|
2022-04-01 01:43:17 +00:00
|
|
|
const cli = MatrixClientPeg.get();
|
2020-07-29 14:51:37 +00:00
|
|
|
const room = cli.getRoom(roomId);
|
|
|
|
if (room && room.getMyMembership() === "invite") {
|
|
|
|
const myMember = room.getMember(cli.getUserId());
|
|
|
|
const inviteEvent = myMember ? myMember.events.member : null;
|
|
|
|
return inviteEvent && inviteEvent.getSender();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-14 15:34:54 +00:00
|
|
|
public showJoinRoomError(err: MatrixError, roomId: string) {
|
2022-03-29 14:02:12 +00:00
|
|
|
let description: ReactNode = err.message ? err.message : JSON.stringify(err);
|
|
|
|
logger.log("Failed to join room:", description);
|
2021-05-24 13:34:06 +00:00
|
|
|
|
|
|
|
if (err.name === "ConnectionError") {
|
2022-03-29 14:02:12 +00:00
|
|
|
description = _t("There was an error joining.");
|
2021-05-24 13:34:06 +00:00
|
|
|
} else if (err.errcode === 'M_INCOMPATIBLE_ROOM_VERSION') {
|
2022-03-29 14:02:12 +00:00
|
|
|
description = <div>
|
|
|
|
{ _t("Sorry, your homeserver is too old to participate here.") }<br />
|
2021-07-19 21:43:11 +00:00
|
|
|
{ _t("Please contact your homeserver administrator.") }
|
2021-05-24 13:34:06 +00:00
|
|
|
</div>;
|
|
|
|
} else if (err.httpStatus === 404) {
|
2022-03-23 05:26:30 +00:00
|
|
|
const invitingUserId = this.getInvitingUserId(roomId);
|
2021-05-24 13:34:06 +00:00
|
|
|
// only provide a better error message for invites
|
|
|
|
if (invitingUserId) {
|
|
|
|
// if the inviting user is on the same HS, there can only be one cause: they left.
|
2022-04-01 01:43:17 +00:00
|
|
|
if (invitingUserId.endsWith(`:${MatrixClientPeg.get().getDomain()}`)) {
|
2022-03-29 14:02:12 +00:00
|
|
|
description = _t("The person who invited you has already left.");
|
2021-05-24 13:34:06 +00:00
|
|
|
} else {
|
2022-03-29 14:02:12 +00:00
|
|
|
description = _t("The person who invited you has already left, or their server is offline.");
|
2021-05-24 13:34:06 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Modal.createTrackedDialog('Failed to join room', '', ErrorDialog, {
|
2022-03-29 14:02:12 +00:00
|
|
|
title: _t("Failed to join"),
|
|
|
|
description,
|
2021-05-24 13:34:06 +00:00
|
|
|
});
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2022-02-22 10:04:27 +00:00
|
|
|
private joinRoomError(payload: JoinRoomErrorPayload) {
|
2021-10-27 14:24:31 +00:00
|
|
|
this.setState({
|
|
|
|
joining: false,
|
|
|
|
joinError: payload.err,
|
|
|
|
});
|
2022-01-07 09:23:54 +00:00
|
|
|
this.showJoinRoomError(payload.err, payload.roomId);
|
2021-10-27 14:24:31 +00:00
|
|
|
}
|
|
|
|
|
2020-07-21 09:14:12 +00:00
|
|
|
public reset() {
|
|
|
|
this.state = Object.assign({}, INITIAL_STATE);
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 15:00:12 +00:00
|
|
|
// The room ID of the room currently being viewed
|
2020-07-21 09:14:12 +00:00
|
|
|
public getRoomId() {
|
|
|
|
return this.state.roomId;
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 16:28:56 +00:00
|
|
|
// The event to scroll to when the room is first viewed
|
2020-07-21 09:14:12 +00:00
|
|
|
public getInitialEventId() {
|
|
|
|
return this.state.initialEventId;
|
2017-06-08 13:17:49 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 15:00:12 +00:00
|
|
|
// Whether to highlight the initial event
|
2020-07-21 09:14:12 +00:00
|
|
|
public isInitialEventHighlighted() {
|
|
|
|
return this.state.isInitialEventHighlighted;
|
2017-06-08 13:17:49 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 15:00:12 +00:00
|
|
|
// The room alias of the room (or null if not originally specified in view_room)
|
2020-07-21 09:14:12 +00:00
|
|
|
public getRoomAlias() {
|
|
|
|
return this.state.roomAlias;
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 15:00:12 +00:00
|
|
|
// Whether the current room is loading (true whilst resolving an alias)
|
2020-07-21 09:14:12 +00:00
|
|
|
public isRoomLoading() {
|
|
|
|
return this.state.roomLoading;
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 15:00:12 +00:00
|
|
|
// Any error that has occurred during loading
|
2020-07-21 09:14:12 +00:00
|
|
|
public getRoomLoadError() {
|
|
|
|
return this.state.roomLoadError;
|
2017-06-01 17:01:30 +00:00
|
|
|
}
|
|
|
|
|
2017-09-15 14:07:09 +00:00
|
|
|
// True if we're expecting the user to be joined to the room currently being
|
|
|
|
// viewed. Note that this is left true after the join request has finished,
|
|
|
|
// since we should still consider a join to be in progress until the room
|
|
|
|
// & member events come down the sync.
|
|
|
|
//
|
|
|
|
// This flag remains true after the room has been sucessfully joined,
|
|
|
|
// (this store doesn't listen for the appropriate member events)
|
2017-09-19 09:21:20 +00:00
|
|
|
// so you should always observe the joined state from the member event
|
|
|
|
// if a room object is present.
|
2017-09-15 14:07:09 +00:00
|
|
|
// ie. The correct logic is:
|
2017-09-19 09:21:20 +00:00
|
|
|
// if (room) {
|
|
|
|
// if (myMember.membership == 'joined') {
|
|
|
|
// // user is joined to the room
|
|
|
|
// } else {
|
|
|
|
// // Not joined
|
|
|
|
// }
|
2017-09-15 14:07:09 +00:00
|
|
|
// } else {
|
2022-03-23 05:26:30 +00:00
|
|
|
// if (RoomViewStore.instance.isJoining()) {
|
2017-09-15 14:07:09 +00:00
|
|
|
// // show spinner
|
|
|
|
// } else {
|
|
|
|
// // show join prompt
|
|
|
|
// }
|
|
|
|
// }
|
2020-07-21 09:14:12 +00:00
|
|
|
public isJoining() {
|
|
|
|
return this.state.joining;
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
|
|
|
|
2017-06-08 15:00:12 +00:00
|
|
|
// Any error that has occurred during joining
|
2020-07-21 09:14:12 +00:00
|
|
|
public getJoinError() {
|
|
|
|
return this.state.joinError;
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|
2017-06-16 15:12:52 +00:00
|
|
|
|
2017-12-10 12:50:41 +00:00
|
|
|
// The mxEvent if one is currently being replied to/quoted
|
2020-07-21 09:14:12 +00:00
|
|
|
public getQuotingEvent() {
|
|
|
|
return this.state.replyingToEvent;
|
2017-12-10 12:50:41 +00:00
|
|
|
}
|
|
|
|
|
2020-07-21 09:14:12 +00:00
|
|
|
public shouldPeek() {
|
|
|
|
return this.state.shouldPeek;
|
2017-06-16 17:24:07 +00:00
|
|
|
}
|
2021-04-29 08:37:21 +00:00
|
|
|
|
|
|
|
public getWasContextSwitch() {
|
|
|
|
return this.state.wasContextSwitch;
|
|
|
|
}
|
2017-05-24 15:56:13 +00:00
|
|
|
}
|