/* 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 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 { RoomSettingsTab } from "../dialogs/RoomSettingsDialog"; import { Action } from "../../../dispatcher/actions"; import { ViewRoomPayload } from "../../../dispatcher/payloads/ViewRoomPayload"; import { doesRoomVersionSupport, PreferredRoomVersions } from "../../../utils/PreferredRoomVersions"; export interface JoinRuleSettingsProps { room: Room; promptUpgrade?: boolean; closeSettingsFn(): void; onError(error: unknown): void; beforeChange?(joinRule: JoinRule): Promise; // if returns false then aborts the change aliasWarning?: ReactNode; } const JoinRuleSettings: React.FC = ({ room, promptUpgrade, aliasWarning, onError, beforeChange, closeSettingsFn, }) => { 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( () => 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 => { let selected = restrictedAllowRoomIds; if (!selected?.length && SpaceStore.instance.activeSpaceRoom) { selected = [SpaceStore.instance.activeSpaceRoom.roomId]; } const { finished } = Modal.createDialog( ManageRestrictedJoinRuleDialog, { room, selected, }, "mx_ManageRestrictedJoinRuleDialog_wrapper", ); const [roomIds] = await finished; return roomIds; }; const definitions: IDefinition[] = [ { 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 = {_t("Upgrade required")}; } 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) as Room[]; 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[]): void => { 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 (): Promise => { const restrictedAllowRoomIds = await editRestrictedRoomIds(); if (!Array.isArray(restrictedAllowRoomIds)) return; if (restrictedAllowRoomIds.length > 0) { onRestrictedRoomIdsChange(restrictedAllowRoomIds); } else { onChange(JoinRule.Invite); } }; description = (
{_t( "Anyone in a space can find and join. Edit which spaces can access here.", {}, { a: (sub) => ( {sub} ), }, )}

{_t("Spaces with access")}

{shownSpaces.map((room) => { return ( {room.name} ); })} {moreText && {moreText}}
); } else if (SpaceStore.instance.activeSpaceRoom) { description = _t( "Anyone in can find and join. You can select other spaces too.", {}, { spaceName: () => {SpaceStore.instance.activeSpaceRoom!.name}, }, ); } 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): Promise => { const beforeJoinRule = content?.join_rule; let restrictedAllowRoomIds: string[] | undefined; 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 | undefined; 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 = ( {_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.", )} ); } 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 => { 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 !== undefined && 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 !== undefined && progress.updateSpacesProgress < progress.updateSpacesTotal ) { fn( _t("Updating spaces... (%(progress)s out of %(count)s)", { progress: progress.updateSpacesProgress, count: progress.updateSpacesTotal, }), 2 + (progress.inviteUsersProgress ?? 0) + progress.updateSpacesProgress, total, ); } }, ); closeSettingsFn(); // switch to the new room in the background dis.dispatch({ action: Action.ViewRoom, room_id: roomId, metricsTrigger: undefined, // other }); // open new settings on this tab dis.dispatch({ action: "open_room_settings", initial_tab_id: RoomSettingsTab.Security, }); }, }); 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 ( ); }; export default JoinRuleSettings;