* Convert tabbedview to functional component The 'Tab' is still a class, so now it's a functional component that has a supporting class, which is maybe a bit... jarring, but I think is actually perfectly logical. * put comment back * Fix bad tab ID behaviour * Make TabbedView a controlled component This does mean the logic of keeping what tab is active is now in each container component, but for a functional component, this is a single line. It makes TabbedView simpler and the container components always know exactly what tab is being displayed rather than having to effectively keep the state separately themselves if they wanted it. Based on https://github.com/matrix-org/matrix-react-sdk/pull/12478 * Fix some types & unused prop * Remove weird behaviour of using first tab is active isn't valid * Don't pass initialTabID here now it no longer has the prop * Fix test * bleh... id, not icon * Change to sub-components and use contitional call syntax * Comments * Fix element IDs * Fix merge * Test DesktopCapturerSourcePicker to make sonarcloud the right colour * Use custom hook for the fllback tab behaviour
151 lines
6 KiB
TypeScript
151 lines
6 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 { ComponentProps } from "react";
|
|
import { MatrixClient } from "matrix-js-sdk/src/matrix";
|
|
|
|
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 { 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 matrixClient?: MatrixClient;
|
|
|
|
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(matrixClient: MatrixClient): void {
|
|
this.matrixClient = matrixClient;
|
|
if (this.isRegistered) return;
|
|
defaultDispatcher.register(this.onDispatch);
|
|
this.isRegistered = true;
|
|
}
|
|
|
|
private onDispatch = (payload: ActionPayload): void => {
|
|
if (!this.matrixClient) return;
|
|
switch (payload.action) {
|
|
case "open_room_settings":
|
|
Modal.createDialog(
|
|
RoomSettingsDialog,
|
|
{
|
|
roomId: payload.room_id || SdkContextClass.instance.roomViewStore.getRoomId(),
|
|
initialTabId: payload.initial_tab_id,
|
|
},
|
|
/*className=*/ undefined,
|
|
/*isPriority=*/ false,
|
|
/*isStatic=*/ true,
|
|
);
|
|
break;
|
|
case Action.OpenForwardDialog:
|
|
Modal.createDialog(ForwardDialog, {
|
|
matrixClient: this.matrixClient,
|
|
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,
|
|
{
|
|
space: payload.space,
|
|
},
|
|
undefined,
|
|
false,
|
|
true,
|
|
);
|
|
break;
|
|
case Action.OpenSpaceSettings:
|
|
Modal.createDialog(
|
|
SpaceSettingsDialog,
|
|
{
|
|
matrixClient: payload.space.client,
|
|
space: payload.space,
|
|
},
|
|
/*className=*/ undefined,
|
|
/*isPriority=*/ false,
|
|
/*isStatic=*/ true,
|
|
);
|
|
break;
|
|
case Action.OpenInviteDialog:
|
|
Modal.createDialog(
|
|
InviteDialog,
|
|
{
|
|
kind: payload.kind,
|
|
call: payload.call,
|
|
roomId: payload.roomId,
|
|
} as Omit<ComponentProps<typeof InviteDialog>, "onFinished">,
|
|
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;
|
|
}
|
|
}
|
|
};
|
|
}
|