2021-03-01 18:10:17 +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-03-26 11:43:01 +00:00
|
|
|
import React from "react";
|
2021-06-29 12:11:58 +00:00
|
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
|
|
|
import { EventType } from "matrix-js-sdk/src/@types/event";
|
2021-11-30 18:08:46 +00:00
|
|
|
import { JoinRule } from "matrix-js-sdk/src/@types/partials";
|
2021-03-01 18:10:17 +00:00
|
|
|
|
2021-07-02 15:07:17 +00:00
|
|
|
import { calculateRoomVia } from "./permalinks/Permalinks";
|
2021-03-02 10:34:28 +00:00
|
|
|
import Modal from "../Modal";
|
|
|
|
import SpaceSettingsDialog from "../components/views/dialogs/SpaceSettingsDialog";
|
2021-03-02 13:32:24 +00:00
|
|
|
import AddExistingToSpaceDialog from "../components/views/dialogs/AddExistingToSpaceDialog";
|
2021-03-02 13:28:05 +00:00
|
|
|
import CreateRoomDialog from "../components/views/dialogs/CreateRoomDialog";
|
2021-06-29 12:11:58 +00:00
|
|
|
import createRoom, { IOpts } from "../createRoom";
|
|
|
|
import { _t } from "../languageHandler";
|
2021-03-26 11:43:01 +00:00
|
|
|
import SpacePublicShare from "../components/views/spaces/SpacePublicShare";
|
|
|
|
import InfoDialog from "../components/views/dialogs/InfoDialog";
|
|
|
|
import { showRoomInviteDialog } from "../RoomInvite";
|
2021-07-23 07:46:20 +00:00
|
|
|
import CreateSubspaceDialog from "../components/views/dialogs/CreateSubspaceDialog";
|
|
|
|
import AddExistingSubspaceDialog from "../components/views/dialogs/AddExistingSubspaceDialog";
|
2021-07-28 18:08:59 +00:00
|
|
|
import defaultDispatcher from "../dispatcher/dispatcher";
|
2022-02-10 10:02:34 +00:00
|
|
|
import dis from "../dispatcher/dispatcher";
|
2021-07-28 18:08:59 +00:00
|
|
|
import RoomViewStore from "../stores/RoomViewStore";
|
|
|
|
import { Action } from "../dispatcher/actions";
|
2021-07-21 08:57:10 +00:00
|
|
|
import { leaveRoomBehaviour } from "./membership";
|
|
|
|
import Spinner from "../components/views/elements/Spinner";
|
|
|
|
import LeaveSpaceDialog from "../components/views/dialogs/LeaveSpaceDialog";
|
2021-12-17 09:26:32 +00:00
|
|
|
import SpacePreferencesDialog, { SpacePreferenceTab } from "../components/views/dialogs/SpacePreferencesDialog";
|
2022-02-10 10:02:34 +00:00
|
|
|
import PosthogTrackers from "../PosthogTrackers";
|
|
|
|
import { ButtonEvent } from "../components/views/elements/AccessibleButton";
|
2022-02-22 10:04:27 +00:00
|
|
|
import { AfterLeaveRoomPayload } from "../dispatcher/payloads/AfterLeaveRoomPayload";
|
2022-03-02 17:37:18 +00:00
|
|
|
import { shouldShowComponent } from "../customisations/helpers/UIComponents";
|
|
|
|
import { UIComponent } from "../settings/UIFeature";
|
2021-03-01 19:05:50 +00:00
|
|
|
|
2021-07-02 15:07:17 +00:00
|
|
|
export const shouldShowSpaceSettings = (space: Room) => {
|
|
|
|
const userId = space.client.getUserId();
|
2021-03-01 18:10:17 +00:00
|
|
|
return space.getMyMembership() === "join"
|
|
|
|
&& (space.currentState.maySendStateEvent(EventType.RoomAvatar, userId)
|
|
|
|
|| space.currentState.maySendStateEvent(EventType.RoomName, userId)
|
|
|
|
|| space.currentState.maySendStateEvent(EventType.RoomTopic, userId)
|
|
|
|
|| space.currentState.maySendStateEvent(EventType.RoomJoinRules, userId));
|
|
|
|
};
|
2021-03-01 19:05:50 +00:00
|
|
|
|
|
|
|
export const makeSpaceParentEvent = (room: Room, canonical = false) => ({
|
|
|
|
type: EventType.SpaceParent,
|
|
|
|
content: {
|
|
|
|
"via": calculateRoomVia(room),
|
|
|
|
"canonical": canonical,
|
|
|
|
},
|
|
|
|
state_key: room.roomId,
|
|
|
|
});
|
2021-03-02 10:34:28 +00:00
|
|
|
|
2021-07-02 15:07:17 +00:00
|
|
|
export const showSpaceSettings = (space: Room) => {
|
2021-03-02 10:34:28 +00:00
|
|
|
Modal.createTrackedDialog("Space Settings", "", SpaceSettingsDialog, {
|
2021-07-02 15:07:17 +00:00
|
|
|
matrixClient: space.client,
|
2021-03-02 10:34:28 +00:00
|
|
|
space,
|
|
|
|
}, /*className=*/null, /*isPriority=*/false, /*isStatic=*/true);
|
|
|
|
};
|
2021-03-02 13:28:05 +00:00
|
|
|
|
2021-07-28 18:08:59 +00:00
|
|
|
export const showAddExistingRooms = (space: Room): void => {
|
|
|
|
Modal.createTrackedDialog(
|
2021-03-02 13:32:24 +00:00
|
|
|
"Space Landing",
|
|
|
|
"Add Existing",
|
|
|
|
AddExistingToSpaceDialog,
|
|
|
|
{
|
2022-02-10 10:02:34 +00:00
|
|
|
onCreateRoomClick: (ev: ButtonEvent) => {
|
|
|
|
showCreateNewRoom(space);
|
|
|
|
PosthogTrackers.trackInteraction("WebAddExistingToSpaceDialogCreateRoomButton", ev);
|
|
|
|
},
|
2021-07-23 07:46:20 +00:00
|
|
|
onAddSubspaceClick: () => showAddExistingSubspace(space),
|
2021-03-02 13:32:24 +00:00
|
|
|
space,
|
2021-07-28 18:08:59 +00:00
|
|
|
onFinished: (added: boolean) => {
|
|
|
|
if (added && RoomViewStore.getRoomId() === space.roomId) {
|
|
|
|
defaultDispatcher.fire(Action.UpdateSpaceHierarchy);
|
|
|
|
}
|
|
|
|
},
|
2021-03-02 13:32:24 +00:00
|
|
|
},
|
|
|
|
"mx_AddExistingToSpaceDialog_wrapper",
|
2021-07-28 18:08:59 +00:00
|
|
|
);
|
2021-03-02 13:32:24 +00:00
|
|
|
};
|
|
|
|
|
2021-07-28 18:08:59 +00:00
|
|
|
export const showCreateNewRoom = async (space: Room): Promise<boolean> => {
|
2021-03-02 13:28:05 +00:00
|
|
|
const modal = Modal.createTrackedDialog<[boolean, IOpts]>(
|
|
|
|
"Space Landing",
|
|
|
|
"Create Room",
|
|
|
|
CreateRoomDialog,
|
|
|
|
{
|
2022-02-10 10:02:34 +00:00
|
|
|
defaultPublic: space.getJoinRule() === JoinRule.Public,
|
2021-03-02 13:28:05 +00:00
|
|
|
parentSpace: space,
|
|
|
|
},
|
|
|
|
);
|
|
|
|
const [shouldCreate, opts] = await modal.finished;
|
|
|
|
if (shouldCreate) {
|
|
|
|
await createRoom(opts);
|
|
|
|
}
|
2021-05-05 16:25:29 +00:00
|
|
|
return shouldCreate;
|
2021-03-02 13:28:05 +00:00
|
|
|
};
|
2021-03-26 11:43:01 +00:00
|
|
|
|
2021-11-30 18:08:46 +00:00
|
|
|
export const shouldShowSpaceInvite = (space: Room) =>
|
2022-03-02 17:37:18 +00:00
|
|
|
(
|
|
|
|
(space?.getMyMembership() === "join" && space.canInvite(space.client.getUserId())) ||
|
|
|
|
space.getJoinRule() === JoinRule.Public
|
|
|
|
) && shouldShowComponent(UIComponent.InviteUsers);
|
2021-11-30 18:08:46 +00:00
|
|
|
|
2021-07-28 18:08:59 +00:00
|
|
|
export const showSpaceInvite = (space: Room, initialText = ""): void => {
|
2021-03-26 11:43:01 +00:00
|
|
|
if (space.getJoinRule() === "public") {
|
|
|
|
const modal = Modal.createTrackedDialog("Space Invite", "User Menu", InfoDialog, {
|
|
|
|
title: _t("Invite to %(spaceName)s", { spaceName: space.name }),
|
|
|
|
description: <React.Fragment>
|
|
|
|
<span>{ _t("Share your public space") }</span>
|
|
|
|
<SpacePublicShare space={space} onFinished={() => modal.close()} />
|
|
|
|
</React.Fragment>,
|
|
|
|
fixedWidth: false,
|
|
|
|
button: false,
|
|
|
|
className: "mx_SpacePanel_sharePublicSpace",
|
|
|
|
hasCloseButton: true,
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
showRoomInviteDialog(space.roomId, initialText);
|
|
|
|
}
|
|
|
|
};
|
2021-07-21 08:57:10 +00:00
|
|
|
|
2021-07-28 18:08:59 +00:00
|
|
|
export const showAddExistingSubspace = (space: Room): void => {
|
|
|
|
Modal.createTrackedDialog(
|
2021-07-23 07:46:20 +00:00
|
|
|
"Space Landing",
|
|
|
|
"Create Subspace",
|
|
|
|
AddExistingSubspaceDialog,
|
|
|
|
{
|
|
|
|
space,
|
|
|
|
onCreateSubspaceClick: () => showCreateNewSubspace(space),
|
2021-07-28 18:08:59 +00:00
|
|
|
onFinished: (added: boolean) => {
|
|
|
|
if (added && RoomViewStore.getRoomId() === space.roomId) {
|
|
|
|
defaultDispatcher.fire(Action.UpdateSpaceHierarchy);
|
|
|
|
}
|
|
|
|
},
|
2021-07-23 07:46:20 +00:00
|
|
|
},
|
|
|
|
"mx_AddExistingToSpaceDialog_wrapper",
|
2021-07-28 18:08:59 +00:00
|
|
|
);
|
2021-07-23 07:46:20 +00:00
|
|
|
};
|
|
|
|
|
2021-07-28 18:08:59 +00:00
|
|
|
export const showCreateNewSubspace = (space: Room): void => {
|
|
|
|
Modal.createTrackedDialog(
|
2021-07-23 07:46:20 +00:00
|
|
|
"Space Landing",
|
|
|
|
"Create Subspace",
|
|
|
|
CreateSubspaceDialog,
|
|
|
|
{
|
|
|
|
space,
|
|
|
|
onAddExistingSpaceClick: () => showAddExistingSubspace(space),
|
2021-07-28 18:08:59 +00:00
|
|
|
onFinished: (added: boolean) => {
|
|
|
|
if (added && RoomViewStore.getRoomId() === space.roomId) {
|
|
|
|
defaultDispatcher.fire(Action.UpdateSpaceHierarchy);
|
|
|
|
}
|
|
|
|
},
|
2021-07-23 07:46:20 +00:00
|
|
|
},
|
|
|
|
"mx_CreateSubspaceDialog_wrapper",
|
2021-07-28 18:08:59 +00:00
|
|
|
);
|
2021-07-23 07:46:20 +00:00
|
|
|
};
|
2021-07-29 14:22:52 +00:00
|
|
|
|
2021-09-17 14:34:49 +00:00
|
|
|
export const bulkSpaceBehaviour = async (
|
|
|
|
space: Room,
|
|
|
|
children: Room[],
|
|
|
|
fn: (room: Room) => Promise<unknown>,
|
|
|
|
): Promise<void> => {
|
|
|
|
const modal = Modal.createDialog(Spinner, null, "mx_Dialog_spinner");
|
|
|
|
try {
|
|
|
|
for (const room of children) {
|
|
|
|
await fn(room);
|
|
|
|
}
|
|
|
|
await fn(space);
|
|
|
|
} finally {
|
|
|
|
modal.close();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-07-21 08:57:10 +00:00
|
|
|
export const leaveSpace = (space: Room) => {
|
|
|
|
Modal.createTrackedDialog("Leave Space", "", LeaveSpaceDialog, {
|
|
|
|
space,
|
|
|
|
onFinished: async (leave: boolean, rooms: Room[]) => {
|
|
|
|
if (!leave) return;
|
2021-09-17 14:34:49 +00:00
|
|
|
await bulkSpaceBehaviour(space, rooms, room => leaveRoomBehaviour(room.roomId));
|
2021-07-21 08:57:10 +00:00
|
|
|
|
2022-02-22 10:04:27 +00:00
|
|
|
dis.dispatch<AfterLeaveRoomPayload>({
|
|
|
|
action: Action.AfterLeaveRoom,
|
2021-07-21 08:57:10 +00:00
|
|
|
room_id: space.roomId,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
}, "mx_LeaveSpaceDialog_wrapper");
|
|
|
|
};
|
2021-08-04 09:37:35 +00:00
|
|
|
|
2021-12-17 09:26:32 +00:00
|
|
|
export const showSpacePreferences = (space: Room, initialTabId?: SpacePreferenceTab): Promise<unknown> => {
|
|
|
|
return Modal.createTrackedDialog("Space preferences", "", SpacePreferencesDialog, {
|
|
|
|
initialTabId,
|
|
|
|
space,
|
|
|
|
}, null, false, true).finished;
|
|
|
|
};
|