mirror of
https://github.com/vector-im/element-web.git
synced 2024-11-18 14:44:58 +08:00
e946674df3
* Add Stores and StoresContext and use it in MatrixChat and RoomView Added a new kind of class: - Add God object `Stores` which will hold refs to all known stores and the `MatrixClient`. This object is NOT a singleton. - Add `StoresContext` to hold onto a ref of `Stores` for use inside components. `StoresContext` is created via: - Create `Stores` in `MatrixChat`, assigning the `MatrixClient` when we have one set. Currently sets the RVS to `RoomViewStore.instance`. - Wrap `MatrixChat`s `render()` function in a `StoresContext.Provider` so it can be used anywhere. `StoresContext` is currently only used in `RoomView` via the following changes: - Remove the HOC, which redundantly set `mxClient` as a prop. We don't need this as `RoomView` was using the client from `this.context`. - Change the type of context accepted from `MatrixClientContext` to `StoresContext`. - Modify alllll the places where `this.context` is used to interact with the client and suffix `.client`. - Modify places where we use `RoomViewStore.instance` and replace them with `this.context.roomViewStore`. This makes `RoomView` use a non-global instance of RVS. * Linting * SDKContext and make client an optional constructor arg * Move SDKContext to /src/contexts * Inject all RVS deps * Linting * Remove reset calls; deep copy the INITIAL_STATE to avoid test pollution * DI singletons used in RoomView; DI them in RoomView-test too * Initial RoomViewStore.instance after all files are imported to avoid cyclical deps * Lazily init stores to allow for circular dependencies Rather than stores accepting a list of other stores in their constructors, which doesn't work when A needs B and B needs A, make new-style stores simply accept Stores. When a store needs another store, they access it via `Stores` which then lazily constructs that store if it needs it. This breaks the circular dependency at constructor time, without needing to introduce wiring diagrams or any complex DI framework. * Delete RoomViewStore.instance Replaced with Stores.instance.roomViewStore * Linting * Move OverridableStores to test/TestStores * Rejig how eager stores get made; don't automatically do it else tests break * Linting * Linting and review comments * Fix new code to use Stores.instance * s/Stores/SdkContextClass/g * Update docs * Remove unused imports * Update src/stores/RoomViewStore.tsx Co-authored-by: Michael Telatynski <7t3chguy@gmail.com> * Remove empty c'tor to make sonar happy Co-authored-by: Michael Telatynski <7t3chguy@gmail.com>
123 lines
5.3 KiB
TypeScript
123 lines
5.3 KiB
TypeScript
/*
|
|
Copyright 2022 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.
|
|
*/
|
|
|
|
import classnames from "classnames";
|
|
|
|
import defaultDispatcher from "../dispatcher/dispatcher";
|
|
import { ActionPayload } from "../dispatcher/payloads";
|
|
import Modal from "../Modal";
|
|
import RoomSettingsDialog from "../components/views/dialogs/RoomSettingsDialog";
|
|
import ForwardDialog from "../components/views/dialogs/ForwardDialog";
|
|
import { MatrixClientPeg } from "../MatrixClientPeg";
|
|
import { Action } from "../dispatcher/actions";
|
|
import ReportEventDialog from "../components/views/dialogs/ReportEventDialog";
|
|
import SpacePreferencesDialog from "../components/views/dialogs/SpacePreferencesDialog";
|
|
import SpaceSettingsDialog from "../components/views/dialogs/SpaceSettingsDialog";
|
|
import InviteDialog from "../components/views/dialogs/InviteDialog";
|
|
import AddExistingToSpaceDialog from "../components/views/dialogs/AddExistingToSpaceDialog";
|
|
import { ButtonEvent } from "../components/views/elements/AccessibleButton";
|
|
import PosthogTrackers from "../PosthogTrackers";
|
|
import { showAddExistingSubspace, showCreateNewRoom } from "./space";
|
|
import { SdkContextClass } from "../contexts/SDKContext";
|
|
|
|
/**
|
|
* Auxiliary class to listen for dialog opening over the dispatcher and
|
|
* open the required dialogs. Not all dialogs run through here, but the
|
|
* ones which cause import cycles are good candidates.
|
|
*/
|
|
export class DialogOpener {
|
|
public static readonly instance = new DialogOpener();
|
|
|
|
private isRegistered = false;
|
|
|
|
private constructor() {
|
|
}
|
|
|
|
// We could do this in the constructor, but then we wouldn't have
|
|
// a function to call from Lifecycle to capture the class.
|
|
public prepare() {
|
|
if (this.isRegistered) return;
|
|
defaultDispatcher.register(this.onDispatch);
|
|
this.isRegistered = true;
|
|
}
|
|
|
|
private onDispatch = (payload: ActionPayload) => {
|
|
switch (payload.action) {
|
|
case 'open_room_settings':
|
|
Modal.createDialog(RoomSettingsDialog, {
|
|
roomId: payload.room_id || SdkContextClass.instance.roomViewStore.getRoomId(),
|
|
initialTabId: payload.initial_tab_id,
|
|
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
|
|
break;
|
|
case Action.OpenForwardDialog:
|
|
Modal.createDialog(ForwardDialog, {
|
|
matrixClient: MatrixClientPeg.get(),
|
|
event: payload.event,
|
|
permalinkCreator: payload.permalinkCreator,
|
|
});
|
|
break;
|
|
case Action.OpenReportEventDialog:
|
|
Modal.createDialog(ReportEventDialog, {
|
|
mxEvent: payload.event,
|
|
}, 'mx_Dialog_reportEvent');
|
|
break;
|
|
case Action.OpenSpacePreferences:
|
|
Modal.createDialog(SpacePreferencesDialog, {
|
|
initialTabId: payload.initalTabId,
|
|
space: payload.space,
|
|
}, null, false, true);
|
|
break;
|
|
case Action.OpenSpaceSettings:
|
|
Modal.createDialog(SpaceSettingsDialog, {
|
|
matrixClient: payload.space.client,
|
|
space: payload.space,
|
|
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
|
|
break;
|
|
case Action.OpenInviteDialog:
|
|
Modal.createDialog(InviteDialog, {
|
|
kind: payload.kind,
|
|
call: payload.call,
|
|
roomId: payload.roomId,
|
|
}, classnames("mx_InviteDialog_flexWrapper", payload.className), false, true).finished
|
|
.then((results) => {
|
|
payload.onFinishedCallback?.(results);
|
|
});
|
|
break;
|
|
case Action.OpenAddToExistingSpaceDialog: {
|
|
const space = payload.space;
|
|
Modal.createDialog(
|
|
AddExistingToSpaceDialog,
|
|
{
|
|
onCreateRoomClick: (ev: ButtonEvent) => {
|
|
showCreateNewRoom(space);
|
|
PosthogTrackers.trackInteraction("WebAddExistingToSpaceDialogCreateRoomButton", ev);
|
|
},
|
|
onAddSubspaceClick: () => showAddExistingSubspace(space),
|
|
space,
|
|
onFinished: (added: boolean) => {
|
|
if (added && SdkContextClass.instance.roomViewStore.getRoomId() === space.roomId) {
|
|
defaultDispatcher.fire(Action.UpdateSpaceHierarchy);
|
|
}
|
|
},
|
|
},
|
|
"mx_AddExistingToSpaceDialog_wrapper",
|
|
);
|
|
break;
|
|
}
|
|
}
|
|
};
|
|
}
|