* Replace MSC3244 support with in-client room version checking * Fix irrelevant ternary * It helps to use Jest correctly
322 lines
14 KiB
TypeScript
322 lines
14 KiB
TypeScript
/*
|
|
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.
|
|
*/
|
|
|
|
import React, { ReactNode } from "react";
|
|
import { IJoinRuleEventContent, JoinRule, RestrictedAllowType } from "matrix-js-sdk/src/@types/partials";
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
|
import { EventType } from "matrix-js-sdk/src/@types/event";
|
|
|
|
import StyledRadioGroup, { IDefinition } from "../elements/StyledRadioGroup";
|
|
import { _t } from "../../../languageHandler";
|
|
import AccessibleButton from "../elements/AccessibleButton";
|
|
import RoomAvatar from "../avatars/RoomAvatar";
|
|
import SpaceStore from "../../../stores/spaces/SpaceStore";
|
|
import { MatrixClientPeg } from "../../../MatrixClientPeg";
|
|
import Modal from "../../../Modal";
|
|
import ManageRestrictedJoinRuleDialog from "../dialogs/ManageRestrictedJoinRuleDialog";
|
|
import RoomUpgradeWarningDialog, { IFinishedOpts } from "../dialogs/RoomUpgradeWarningDialog";
|
|
import { upgradeRoom } from "../../../utils/RoomUpgrade";
|
|
import { arrayHasDiff } from "../../../utils/arrays";
|
|
import { useLocalEcho } from "../../../hooks/useLocalEcho";
|
|
import dis from "../../../dispatcher/dispatcher";
|
|
import { ROOM_SECURITY_TAB } from "../dialogs/RoomSettingsDialog";
|
|
import { Action } from "../../../dispatcher/actions";
|
|
import { ViewRoomPayload } from "../../../dispatcher/payloads/ViewRoomPayload";
|
|
import { doesRoomVersionSupport, PreferredRoomVersions } from "../../../utils/PreferredRoomVersions";
|
|
|
|
interface IProps {
|
|
room: Room;
|
|
promptUpgrade?: boolean;
|
|
closeSettingsFn(): void;
|
|
onError(error: Error): void;
|
|
beforeChange?(joinRule: JoinRule): Promise<boolean>; // if returns false then aborts the change
|
|
aliasWarning?: ReactNode;
|
|
}
|
|
|
|
const JoinRuleSettings = ({ room, promptUpgrade, aliasWarning, onError, beforeChange, closeSettingsFn }: IProps) => {
|
|
const cli = room.client;
|
|
|
|
const roomSupportsRestricted = doesRoomVersionSupport(room.getVersion(), PreferredRoomVersions.RestrictedRooms);
|
|
const preferredRestrictionVersion = !roomSupportsRestricted && promptUpgrade
|
|
? PreferredRoomVersions.RestrictedRooms
|
|
: undefined;
|
|
|
|
const disabled = !room.currentState.mayClientSendStateEvent(EventType.RoomJoinRules, cli);
|
|
|
|
const [content, setContent] = useLocalEcho<IJoinRuleEventContent>(
|
|
() => room.currentState.getStateEvents(EventType.RoomJoinRules, "")?.getContent(),
|
|
content => cli.sendStateEvent(room.roomId, EventType.RoomJoinRules, content, ""),
|
|
onError,
|
|
);
|
|
|
|
const { join_rule: joinRule = JoinRule.Invite } = content || {};
|
|
const restrictedAllowRoomIds = joinRule === JoinRule.Restricted
|
|
? content.allow?.filter(o => o.type === RestrictedAllowType.RoomMembership).map(o => o.room_id)
|
|
: undefined;
|
|
|
|
const editRestrictedRoomIds = async (): Promise<string[] | undefined> => {
|
|
let selected = restrictedAllowRoomIds;
|
|
if (!selected?.length && SpaceStore.instance.activeSpaceRoom) {
|
|
selected = [SpaceStore.instance.activeSpaceRoom.roomId];
|
|
}
|
|
|
|
const matrixClient = MatrixClientPeg.get();
|
|
const { finished } = Modal.createDialog(ManageRestrictedJoinRuleDialog, {
|
|
matrixClient,
|
|
room,
|
|
selected,
|
|
}, "mx_ManageRestrictedJoinRuleDialog_wrapper");
|
|
|
|
const [roomIds] = await finished;
|
|
return roomIds;
|
|
};
|
|
|
|
const definitions: IDefinition<JoinRule>[] = [{
|
|
value: JoinRule.Invite,
|
|
label: _t("Private (invite only)"),
|
|
description: _t("Only invited people can join."),
|
|
checked: joinRule === JoinRule.Invite || (joinRule === JoinRule.Restricted && !restrictedAllowRoomIds?.length),
|
|
}, {
|
|
value: JoinRule.Public,
|
|
label: _t("Public"),
|
|
description: <>
|
|
{ _t("Anyone can find and join.") }
|
|
{ aliasWarning }
|
|
</>,
|
|
}];
|
|
|
|
if (roomSupportsRestricted || preferredRestrictionVersion || joinRule === JoinRule.Restricted) {
|
|
let upgradeRequiredPill;
|
|
if (preferredRestrictionVersion) {
|
|
upgradeRequiredPill = <span className="mx_JoinRuleSettings_upgradeRequired">
|
|
{ _t("Upgrade required") }
|
|
</span>;
|
|
}
|
|
|
|
let description;
|
|
if (joinRule === JoinRule.Restricted && restrictedAllowRoomIds?.length) {
|
|
// only show the first 4 spaces we know about, so that the UI doesn't grow out of proportion there are lots.
|
|
const shownSpaces = restrictedAllowRoomIds
|
|
.map(roomId => cli.getRoom(roomId))
|
|
.filter(room => room?.isSpaceRoom())
|
|
.slice(0, 4);
|
|
|
|
let moreText;
|
|
if (shownSpaces.length < restrictedAllowRoomIds.length) {
|
|
if (shownSpaces.length > 0) {
|
|
moreText = _t("& %(count)s more", {
|
|
count: restrictedAllowRoomIds.length - shownSpaces.length,
|
|
});
|
|
} else {
|
|
moreText = _t("Currently, %(count)s spaces have access", {
|
|
count: restrictedAllowRoomIds.length,
|
|
});
|
|
}
|
|
}
|
|
|
|
const onRestrictedRoomIdsChange = (newAllowRoomIds: string[]) => {
|
|
if (!arrayHasDiff(restrictedAllowRoomIds || [], newAllowRoomIds)) return;
|
|
|
|
if (!newAllowRoomIds.length) {
|
|
setContent({
|
|
join_rule: JoinRule.Invite,
|
|
});
|
|
return;
|
|
}
|
|
|
|
setContent({
|
|
join_rule: JoinRule.Restricted,
|
|
allow: newAllowRoomIds.map(roomId => ({
|
|
"type": RestrictedAllowType.RoomMembership,
|
|
"room_id": roomId,
|
|
})),
|
|
});
|
|
};
|
|
|
|
const onEditRestrictedClick = async () => {
|
|
const restrictedAllowRoomIds = await editRestrictedRoomIds();
|
|
if (!Array.isArray(restrictedAllowRoomIds)) return;
|
|
if (restrictedAllowRoomIds.length > 0) {
|
|
onRestrictedRoomIdsChange(restrictedAllowRoomIds);
|
|
} else {
|
|
onChange(JoinRule.Invite);
|
|
}
|
|
};
|
|
|
|
description = <div>
|
|
<span>
|
|
{ _t("Anyone in a space can find and join. <a>Edit which spaces can access here.</a>", {}, {
|
|
a: sub => <AccessibleButton
|
|
disabled={disabled}
|
|
onClick={onEditRestrictedClick}
|
|
kind="link_inline"
|
|
>
|
|
{ sub }
|
|
</AccessibleButton>,
|
|
}) }
|
|
</span>
|
|
|
|
<div className="mx_JoinRuleSettings_spacesWithAccess">
|
|
<h4>{ _t("Spaces with access") }</h4>
|
|
{ shownSpaces.map(room => {
|
|
return <span key={room.roomId}>
|
|
<RoomAvatar room={room} height={32} width={32} />
|
|
{ room.name }
|
|
</span>;
|
|
}) }
|
|
{ moreText && <span>{ moreText }</span> }
|
|
</div>
|
|
</div>;
|
|
} else if (SpaceStore.instance.activeSpaceRoom) {
|
|
description = _t("Anyone in <spaceName/> can find and join. You can select other spaces too.", {}, {
|
|
spaceName: () => <b>{ SpaceStore.instance.activeSpaceRoom.name }</b>,
|
|
});
|
|
} else {
|
|
description = _t("Anyone in a space can find and join. You can select multiple spaces.");
|
|
}
|
|
|
|
definitions.splice(1, 0, {
|
|
value: JoinRule.Restricted,
|
|
label: <>
|
|
{ _t("Space members") }
|
|
{ upgradeRequiredPill }
|
|
</>,
|
|
description,
|
|
// if there are 0 allowed spaces then render it as invite only instead
|
|
checked: joinRule === JoinRule.Restricted && !!restrictedAllowRoomIds?.length,
|
|
});
|
|
}
|
|
|
|
const onChange = async (joinRule: JoinRule) => {
|
|
const beforeJoinRule = content.join_rule;
|
|
|
|
let restrictedAllowRoomIds: string[];
|
|
if (joinRule === JoinRule.Restricted) {
|
|
if (beforeJoinRule === JoinRule.Restricted || roomSupportsRestricted) {
|
|
// Have the user pick which spaces to allow joins from
|
|
restrictedAllowRoomIds = await editRestrictedRoomIds();
|
|
if (!Array.isArray(restrictedAllowRoomIds)) return;
|
|
} else if (preferredRestrictionVersion) {
|
|
// Block this action on a room upgrade otherwise it'd make their room unjoinable
|
|
const targetVersion = preferredRestrictionVersion;
|
|
|
|
let warning: JSX.Element;
|
|
const userId = cli.getUserId();
|
|
const unableToUpdateSomeParents = Array.from(SpaceStore.instance.getKnownParents(room.roomId))
|
|
.some(roomId => !cli.getRoom(roomId)?.currentState.maySendStateEvent(EventType.SpaceChild, userId));
|
|
if (unableToUpdateSomeParents) {
|
|
warning = <b>
|
|
{ _t("This room is in some spaces you're not an admin of. " +
|
|
"In those spaces, the old room will still be shown, " +
|
|
"but people will be prompted to join the new one.") }
|
|
</b>;
|
|
}
|
|
|
|
Modal.createDialog(RoomUpgradeWarningDialog, {
|
|
roomId: room.roomId,
|
|
targetVersion,
|
|
description: <>
|
|
{ _t("This upgrade will allow members of selected spaces " +
|
|
"access to this room without an invite.") }
|
|
{ warning }
|
|
</>,
|
|
doUpgrade: async (
|
|
opts: IFinishedOpts,
|
|
fn: (progressText: string, progress: number, total: number) => void,
|
|
): Promise<void> => {
|
|
const roomId = await upgradeRoom(
|
|
room,
|
|
targetVersion,
|
|
opts.invite,
|
|
true,
|
|
true,
|
|
true,
|
|
progress => {
|
|
const total = 2 + progress.updateSpacesTotal + progress.inviteUsersTotal;
|
|
if (!progress.roomUpgraded) {
|
|
fn(_t("Upgrading room"), 0, total);
|
|
} else if (!progress.roomSynced) {
|
|
fn(_t("Loading new room"), 1, total);
|
|
} else if (progress.inviteUsersProgress < progress.inviteUsersTotal) {
|
|
fn(_t("Sending invites... (%(progress)s out of %(count)s)", {
|
|
progress: progress.inviteUsersProgress,
|
|
count: progress.inviteUsersTotal,
|
|
}), 2 + progress.inviteUsersProgress, total);
|
|
} else if (progress.updateSpacesProgress < progress.updateSpacesTotal) {
|
|
fn(_t("Updating spaces... (%(progress)s out of %(count)s)", {
|
|
progress: progress.updateSpacesProgress,
|
|
count: progress.updateSpacesTotal,
|
|
}), 2 + progress.inviteUsersProgress + progress.updateSpacesProgress, total);
|
|
}
|
|
},
|
|
);
|
|
closeSettingsFn();
|
|
|
|
// switch to the new room in the background
|
|
dis.dispatch<ViewRoomPayload>({
|
|
action: Action.ViewRoom,
|
|
room_id: roomId,
|
|
metricsTrigger: undefined, // other
|
|
});
|
|
|
|
// open new settings on this tab
|
|
dis.dispatch({
|
|
action: "open_room_settings",
|
|
initial_tab_id: ROOM_SECURITY_TAB,
|
|
});
|
|
},
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
// when setting to 0 allowed rooms/spaces set to invite only instead as per the note
|
|
if (!restrictedAllowRoomIds.length) {
|
|
joinRule = JoinRule.Invite;
|
|
}
|
|
}
|
|
|
|
if (beforeJoinRule === joinRule && !restrictedAllowRoomIds) return;
|
|
if (beforeChange && !(await beforeChange(joinRule))) return;
|
|
|
|
const newContent: IJoinRuleEventContent = {
|
|
join_rule: joinRule,
|
|
};
|
|
|
|
// pre-set the accepted spaces with the currently viewed one as per the microcopy
|
|
if (joinRule === JoinRule.Restricted) {
|
|
newContent.allow = restrictedAllowRoomIds.map(roomId => ({
|
|
"type": RestrictedAllowType.RoomMembership,
|
|
"room_id": roomId,
|
|
}));
|
|
}
|
|
|
|
setContent(newContent);
|
|
};
|
|
|
|
return (
|
|
<StyledRadioGroup
|
|
name="joinRule"
|
|
value={joinRule}
|
|
onChange={onChange}
|
|
definitions={definitions}
|
|
disabled={disabled}
|
|
className="mx_JoinRuleSettings_radioButton"
|
|
/>
|
|
);
|
|
};
|
|
|
|
export default JoinRuleSettings;
|