diff --git a/res/css/_components.scss b/res/css/_components.scss
index 49cfd8fe22..69edd301d0 100644
--- a/res/css/_components.scss
+++ b/res/css/_components.scss
@@ -143,6 +143,7 @@
@import "./views/settings/tabs/_GeneralUserSettingsTab.scss";
@import "./views/settings/tabs/_HelpSettingsTab.scss";
@import "./views/settings/tabs/_PreferencesSettingsTab.scss";
+@import "./views/settings/tabs/_RolesRoomSettingsTab.scss";
@import "./views/settings/tabs/_SecurityRoomSettingsTab.scss";
@import "./views/settings/tabs/_SecuritySettingsTab.scss";
@import "./views/settings/tabs/_SettingsTab.scss";
diff --git a/res/css/views/settings/tabs/_RolesRoomSettingsTab.scss b/res/css/views/settings/tabs/_RolesRoomSettingsTab.scss
new file mode 100644
index 0000000000..657d23af26
--- /dev/null
+++ b/res/css/views/settings/tabs/_RolesRoomSettingsTab.scss
@@ -0,0 +1,24 @@
+/*
+Copyright 2019 New Vector Ltd
+
+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.
+*/
+
+.mx_RolesRoomSettingsTab ul {
+ margin-bottom: 0;
+}
+
+.mx_RolesRoomSettingsTab_unbanBtn {
+ margin-right: 10px;
+ margin-bottom: 5px;
+}
\ No newline at end of file
diff --git a/src/components/views/dialogs/RoomSettingsDialog.js b/src/components/views/dialogs/RoomSettingsDialog.js
index f00a2a0121..6be2676d72 100644
--- a/src/components/views/dialogs/RoomSettingsDialog.js
+++ b/src/components/views/dialogs/RoomSettingsDialog.js
@@ -18,8 +18,10 @@ import React from 'react';
import PropTypes from 'prop-types';
import {Tab, TabbedView} from "../../structures/TabbedView";
import {_t, _td} from "../../../languageHandler";
+import AdvancedRoomSettingsTab from "../settings/tabs/AdvancedRoomSettingsTab";
import AccessibleButton from "../elements/AccessibleButton";
import dis from '../../../dispatcher';
+import RolesRoomSettingsTab from "../settings/tabs/RolesRoomSettingsTab";
import GeneralRoomSettingsTab from "../settings/tabs/GeneralRoomSettingsTab";
import SecurityRoomSettingsTab from "../settings/tabs/SecurityRoomSettingsTab";
@@ -74,12 +76,12 @@ export default class RoomSettingsDialog extends React.Component {
tabs.push(new Tab(
_td("Roles & Permissions"),
"mx_RoomSettingsDialog_rolesIcon",
-
Roles Test
,
+ ,
));
tabs.push(new Tab(
_td("Advanced"),
"mx_RoomSettingsDialog_warningIcon",
- Advanced Test
,
+ ,
));
tabs.push(new Tab(
_td("Visit old settings"),
diff --git a/src/components/views/settings/tabs/AdvancedRoomSettingsTab.js b/src/components/views/settings/tabs/AdvancedRoomSettingsTab.js
new file mode 100644
index 0000000000..9b99622516
--- /dev/null
+++ b/src/components/views/settings/tabs/AdvancedRoomSettingsTab.js
@@ -0,0 +1,104 @@
+/*
+Copyright 2019 New Vector Ltd
+
+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 from 'react';
+import PropTypes from 'prop-types';
+import {_t} from "../../../../languageHandler";
+import MatrixClientPeg from "../../../../MatrixClientPeg";
+import sdk from "../../../../index";
+import AccessibleButton from "../../elements/AccessibleButton";
+import Modal from "../../../../Modal";
+
+export default class AdvancedRoomSettingsTab extends React.Component {
+ static propTypes = {
+ roomId: PropTypes.string.isRequired,
+ };
+
+ constructor() {
+ super();
+
+ this.state = {
+ // This is eventually set to the value of room.getRecommendedVersion()
+ upgradeRecommendation: null,
+ };
+ }
+
+ componentWillMount() {
+ // we handle lack of this object gracefully later, so don't worry about it failing here.
+ MatrixClientPeg.get().getRoom(this.props.roomId).getRecommendedVersion().then((v) => {
+ this.setState({upgradeRecommendation: v});
+ });
+ }
+
+ _upgradeRoom = (e) => {
+ const RoomUpgradeDialog = sdk.getComponent('dialogs.RoomUpgradeDialog');
+ const room = MatrixClientPeg.get().getRoom(this.props.roomId);
+ Modal.createTrackedDialog('Upgrade Room Version', '', RoomUpgradeDialog, {room: room});
+ };
+
+ _openDevtools = (e) => {
+ const DevtoolsDialog = sdk.getComponent('dialogs.DevtoolsDialog');
+ Modal.createDialog(DevtoolsDialog, {roomId: this.props.roomId});
+ };
+
+ render() {
+ const client = MatrixClientPeg.get();
+ const room = client.getRoom(this.props.roomId);
+
+ let unfederatableSection;
+ const createEvent = room.currentState.getStateEvents('m.room.create', '');
+ if (createEvent && createEvent.getContent()['m.federate'] === false) {
+ unfederatableSection = {_t('This room is not accessible by remote Matrix servers')}
;
+ }
+
+ let roomUpgradeButton;
+ if (this.state.upgradeRecommendation && this.state.upgradeRecommendation.needsUpgrade) {
+ roomUpgradeButton = (
+
+ {_t("Upgrade room to version %(ver)s", {ver: this.state.upgradeRecommendation.version})}
+
+ );
+ }
+
+ return (
+
+
{_t("Advanced")}
+
+
{_t("Room information")}
+
+ {_t("Internal room ID:")}
+ {this.props.roomId}
+
+ {unfederatableSection}
+
+
+
{_t("Room version")}
+
+ {_t("Room version:")}
+ {room.getVersion()}
+
+ {roomUpgradeButton}
+
+
+
{_t("Developer options")}
+
+ {_t("Open Devtools")}
+
+
+
+ );
+ }
+}
diff --git a/src/components/views/settings/tabs/RolesRoomSettingsTab.js b/src/components/views/settings/tabs/RolesRoomSettingsTab.js
new file mode 100644
index 0000000000..776ce9f01a
--- /dev/null
+++ b/src/components/views/settings/tabs/RolesRoomSettingsTab.js
@@ -0,0 +1,319 @@
+/*
+Copyright 2019 New Vector Ltd
+
+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 from 'react';
+import PropTypes from 'prop-types';
+import {_t, _td} from "../../../../languageHandler";
+import MatrixClientPeg from "../../../../MatrixClientPeg";
+import sdk from "../../../../index";
+import AccessibleButton from "../../elements/AccessibleButton";
+import Modal from "../../../../Modal";
+
+const plEventsToLabels = {
+ // These will be translated for us later.
+ "m.room.avatar": _td("To change the room's avatar, you must be a"),
+ "m.room.name": _td("To change the room's name, you must be a"),
+ "m.room.canonical_alias": _td("To change the room's main address, you must be a"),
+ "m.room.history_visibility": _td("To change the room's history visibility, you must be a"),
+ "m.room.power_levels": _td("To change the permissions in the room, you must be a"),
+ "m.room.topic": _td("To change the topic, you must be a"),
+
+ "im.vector.modular.widgets": _td("To modify widgets in the room, you must be a"),
+};
+
+const plEventsToShow = {
+ // If an event is listed here, it will be shown in the PL settings. Defaults will be calculated.
+ "m.room.avatar": {isState: true},
+ "m.room.name": {isState: true},
+ "m.room.canonical_alias": {isState: true},
+ "m.room.history_visibility": {isState: true},
+ "m.room.power_levels": {isState: true},
+ "m.room.topic": {isState: true},
+
+ "im.vector.modular.widgets": {isState: true},
+};
+
+// parse a string as an integer; if the input is undefined, or cannot be parsed
+// as an integer, return a default.
+function parseIntWithDefault(val, def) {
+ const res = parseInt(val);
+ return isNaN(res) ? def : res;
+}
+
+export class BannedUser extends React.Component {
+ static propTypes = {
+ canUnban: PropTypes.bool,
+ member: PropTypes.object.isRequired, // js-sdk RoomMember
+ by: PropTypes.string.isRequired,
+ reason: PropTypes.string,
+ onUnbanned: PropTypes.func.isRequired,
+ };
+
+ _onUnbanClick = (e) => {
+ MatrixClientPeg.get().unban(this.props.member.roomId, this.props.member.userId).then(() => {
+ this.props.onUnbanned();
+ }).catch((err) => {
+ const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
+ console.error("Failed to unban: " + err);
+ Modal.createTrackedDialog('Failed to unban', '', ErrorDialog, {
+ title: _t('Error'),
+ description: _t('Failed to unban'),
+ });
+ });
+ };
+
+ render() {
+ let unbanButton;
+
+ if (this.props.canUnban) {
+ unbanButton = (
+
+ { _t('Unban') }
+
+ );
+ }
+
+ const userId = this.props.member.name === this.props.member.userId ? null : this.props.member.userId;
+ return (
+
+ {unbanButton}
+
+ { this.props.member.name } {userId}
+ {this.props.reason ? " " + _t('Reason') + ": " + this.props.reason : ""}
+
+
+ );
+ }
+}
+
+export default class RolesRoomSettingsTab extends React.Component {
+ static propTypes = {
+ roomId: PropTypes.string.isRequired,
+ };
+
+ _populateDefaultPlEvents(eventsSection, stateLevel, eventsLevel) {
+ for (const desiredEvent of Object.keys(plEventsToShow)) {
+ if (!(desiredEvent in eventsSection)) {
+ eventsSection[desiredEvent] = (plEventsToShow[desiredEvent].isState ? stateLevel : eventsLevel);
+ }
+ }
+ }
+
+ render() {
+ const PowerSelector = sdk.getComponent('elements.PowerSelector');
+
+ const client = MatrixClientPeg.get();
+ const room = client.getRoom(this.props.roomId);
+ const plContent = room.currentState.getStateEvents('m.room.power_levels', '').getContent() || {};
+ const canChangeLevels = room.currentState.mayClientSendStateEvent('m.room.power_levels', client);
+
+ const powerLevelDescriptors = {
+ "users_default": {
+ desc: _t('The default role for new room members is'),
+ defaultValue: 0,
+ },
+ "events_default": {
+ desc: _t('To send messages, you must be a'),
+ defaultValue: 0,
+ },
+ "invite": {
+ desc: _t('To invite users into the room, you must be a'),
+ defaultValue: 50,
+ },
+ "state_default": {
+ desc: _t('To configure the room, you must be a'),
+ defaultValue: 50,
+ },
+ "kick": {
+ desc: _t('To kick users, you must be a'),
+ defaultValue: 50,
+ },
+ "ban": {
+ desc: _t('To ban users, you must be a'),
+ defaultValue: 50,
+ },
+ "redact": {
+ desc: _t('To remove other users\' messages, you must be a'),
+ defaultValue: 50,
+ },
+ "notifications.room": {
+ desc: _t('To notify everyone in the room, you must be a'),
+ defaultValue: 50,
+ },
+ };
+
+ const eventsLevels = plContent.events || {};
+ const userLevels = plContent.users || {};
+ const banLevel = parseIntWithDefault(plContent.ban, powerLevelDescriptors.ban.defaultValue);
+ const defaultUserLevel = parseIntWithDefault(
+ plContent.users_default,
+ powerLevelDescriptors.users_default.defaultValue,
+ );
+
+ let currentUserLevel = userLevels[client.getUserId()];
+ if (currentUserLevel === undefined) {
+ currentUserLevel = defaultUserLevel;
+ }
+
+ this._populateDefaultPlEvents(
+ eventsLevels,
+ parseIntWithDefault(plContent.state_default, powerLevelDescriptors.state_default.defaultValue),
+ parseIntWithDefault(plContent.events_default, powerLevelDescriptors.events_default.defaultValue),
+ );
+
+ let privilegedUsersSection = {_t('No users have specific privileges in this room')}
;
+ let mutedUsersSection;
+ if (Object.keys(userLevels).length) {
+ const privilegedUsers = [];
+ const mutedUsers = [];
+
+ Object.keys(userLevels).forEach(function(user) {
+ if (userLevels[user] > defaultUserLevel) { // privileged
+ privilegedUsers.push(
+ { _t("%(user)s is a %(userRole)s", {
+ user: user,
+ userRole: ,
+ }) }
+ );
+ } else if (userLevels[user] < defaultUserLevel) { // muted
+ mutedUsers.push(
+ { _t("%(user)s is a %(userRole)s", {
+ user: user,
+ userRole: ,
+ }) }
+ );
+ }
+ });
+
+ // comparator for sorting PL users lexicographically on PL descending, MXID ascending. (case-insensitive)
+ const comparator = (a, b) => {
+ const plDiff = userLevels[b.key] - userLevels[a.key];
+ return plDiff !== 0 ? plDiff : a.key.toLocaleLowerCase().localeCompare(b.key.toLocaleLowerCase());
+ };
+
+ privilegedUsers.sort(comparator);
+ mutedUsers.sort(comparator);
+
+ if (privilegedUsers.length) {
+ privilegedUsersSection =
+
+
{ _t('Privileged Users') }
+
+
;
+ }
+ if (mutedUsers.length) {
+ mutedUsersSection =
+
+
{ _t('Muted Users') }
+
+
;
+ }
+ }
+
+ const banned = room.getMembersWithMembership("ban");
+ let bannedUsersSection;
+ if (banned.length) {
+ const canBanUsers = currentUserLevel >= banLevel;
+ bannedUsersSection =
+
+
{ _t('Banned users') }
+
+ {banned.map((member) => {
+ const banEvent = member.events.member.getContent();
+ const sender = room.getMember(member.events.member.getSender());
+ let bannedBy = member.events.member.getSender(); // start by falling back to mxid
+ if (sender) bannedBy = sender.name;
+ return (
+
+ );
+ })}
+
+
;
+ }
+
+ const powerSelectors = Object.keys(powerLevelDescriptors).map((key, index) => {
+ const descriptor = powerLevelDescriptors[key];
+
+ const keyPath = key.split('.');
+ let currentObj = plContent;
+ for (const prop of keyPath) {
+ if (currentObj === undefined) {
+ break;
+ }
+ currentObj = currentObj[prop];
+ }
+
+ const value = parseIntWithDefault(currentObj, descriptor.defaultValue);
+ return ;
+ });
+
+ const eventPowerSelectors = Object.keys(eventsLevels).map(function(eventType, i) {
+ let label = plEventsToLabels[eventType];
+ if (label) {
+ label = _t(label);
+ } else {
+ label = _t(
+ "To send events of type , you must be a", {},
+ { 'eventType': { eventType }
},
+ );
+ }
+ return (
+
+ );
+ });
+
+ return (
+
+
{_t("Roles & Permissions")}
+ {privilegedUsersSection}
+ {mutedUsersSection}
+ {bannedUsersSection}
+
+ {_t("Permissions")}
+ {powerSelectors}
+ {eventPowerSelectors}
+
+
+ );
+ }
+}
diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json
index 722631c23b..71e6cd6b51 100644
--- a/src/i18n/strings/en_EN.json
+++ b/src/i18n/strings/en_EN.json
@@ -430,6 +430,15 @@
"Upload profile picture": "Upload profile picture",
"Display Name": "Display Name",
"Save": "Save",
+ "This room is not accessible by remote Matrix servers": "This room is not accessible by remote Matrix servers",
+ "Upgrade room to version %(ver)s": "Upgrade room to version %(ver)s",
+ "Advanced": "Advanced",
+ "Room information": "Room information",
+ "Internal room ID:": "Internal room ID:",
+ "Room version": "Room version",
+ "Room version:": "Room version:",
+ "Developer options": "Developer options",
+ "Open Devtools": "Open Devtools",
"Flair": "Flair",
"General": "General",
"Room Addresses": "Room Addresses",
@@ -466,7 +475,6 @@
"matrix-react-sdk version:": "matrix-react-sdk version:",
"riot-web version:": "riot-web version:",
"olm version:": "olm version:",
- "Advanced": "Advanced",
"Homeserver is": "Homeserver is",
"Identity Server is": "Identity Server is",
"Access Token:": "Access Token:",
@@ -481,28 +489,32 @@
"Room list": "Room list",
"Timeline": "Timeline",
"Autocomplete delay (ms)": "Autocomplete delay (ms)",
- "End-to-end encryption is in beta and may not be reliable": "End-to-end encryption is in beta and may not be reliable",
- "You should not yet trust it to secure data": "You should not yet trust it to secure data",
- "Devices will not yet be able to decrypt history from before they joined the room": "Devices will not yet be able to decrypt history from before they joined the room",
- "Once encryption is enabled for a room it cannot be turned off again (for now)": "Once encryption is enabled for a room it cannot be turned off again (for now)",
- "Encrypted messages will not be visible on clients that do not yet implement encryption": "Encrypted messages will not be visible on clients that do not yet implement encryption",
- "Guests cannot join this room even if explicitly invited.": "Guests cannot join this room even if explicitly invited.",
- "Click here to fix": "Click here to fix",
- "To link to this room, please add an alias.": "To link to this room, please add an alias.",
- "Only people who have been invited": "Only people who have been invited",
- "Anyone who knows the room's link, apart from guests": "Anyone who knows the room's link, apart from guests",
- "Anyone who knows the room's link, including guests": "Anyone who knows the room's link, including guests",
- "Changes to who can read history will only apply to future messages in this room. The visibility of existing history will be unchanged.": "Changes to who can read history will only apply to future messages in this room. The visibility of existing history will be unchanged.",
- "Anyone": "Anyone",
- "Members only (since the point in time of selecting this option)": "Members only (since the point in time of selecting this option)",
- "Members only (since they were invited)": "Members only (since they were invited)",
- "Members only (since they joined)": "Members only (since they joined)",
- "Once enabled, encryption cannot be disabled.": "Once enabled, encryption cannot be disabled.",
- "Encrypted": "Encrypted",
- "Security & Privacy": "Security & Privacy",
- "Encryption": "Encryption",
- "Who can access this room?": "Who can access this room?",
- "Who can read history?": "Who can read history?",
+ "To change the room's avatar, you must be a": "To change the room's avatar, you must be a",
+ "To change the room's name, you must be a": "To change the room's name, you must be a",
+ "To change the room's main address, you must be a": "To change the room's main address, you must be a",
+ "To change the room's history visibility, you must be a": "To change the room's history visibility, you must be a",
+ "To change the permissions in the room, you must be a": "To change the permissions in the room, you must be a",
+ "To change the topic, you must be a": "To change the topic, you must be a",
+ "To modify widgets in the room, you must be a": "To modify widgets in the room, you must be a",
+ "Failed to unban": "Failed to unban",
+ "Unban": "Unban",
+ "Banned by %(displayName)s": "Banned by %(displayName)s",
+ "The default role for new room members is": "The default role for new room members is",
+ "To send messages, you must be a": "To send messages, you must be a",
+ "To invite users into the room, you must be a": "To invite users into the room, you must be a",
+ "To configure the room, you must be a": "To configure the room, you must be a",
+ "To kick users, you must be a": "To kick users, you must be a",
+ "To ban users, you must be a": "To ban users, you must be a",
+ "To remove other users' messages, you must be a": "To remove other users' messages, you must be a",
+ "To notify everyone in the room, you must be a": "To notify everyone in the room, you must be a",
+ "No users have specific privileges in this room": "No users have specific privileges in this room",
+ "%(user)s is a %(userRole)s": "%(user)s is a %(userRole)s",
+ "Privileged Users": "Privileged Users",
+ "Muted Users": "Muted Users",
+ "Banned users": "Banned users",
+ "To send events of type , you must be a": "To send events of type , you must be a",
+ "Roles & Permissions": "Roles & Permissions",
+ "Permissions": "Permissions",
"Unignore": "Unignore",
"": "",
"Import E2E room keys": "Import E2E room keys",
@@ -564,7 +576,6 @@
"Disinvite this user?": "Disinvite this user?",
"Kick this user?": "Kick this user?",
"Failed to kick": "Failed to kick",
- "Unban": "Unban",
"Ban": "Ban",
"Unban this user?": "Unban this user?",
"Ban this user?": "Ban this user?",
@@ -707,15 +718,6 @@
"If you log out or use another device, you'll lose your secure message history. To prevent this, set up Secure Message Recovery.": "If you log out or use another device, you'll lose your secure message history. To prevent this, set up Secure Message Recovery.",
"Secure Message Recovery": "Secure Message Recovery",
"Don't ask again": "Don't ask again",
- "To change the room's avatar, you must be a": "To change the room's avatar, you must be a",
- "To change the room's name, you must be a": "To change the room's name, you must be a",
- "To change the room's main address, you must be a": "To change the room's main address, you must be a",
- "To change the room's history visibility, you must be a": "To change the room's history visibility, you must be a",
- "To change the permissions in the room, you must be a": "To change the permissions in the room, you must be a",
- "To change the topic, you must be a": "To change the topic, you must be a",
- "To modify widgets in the room, you must be a": "To modify widgets in the room, you must be a",
- "Failed to unban": "Failed to unban",
- "Banned by %(displayName)s": "Banned by %(displayName)s",
"Privacy warning": "Privacy warning",
"Changes to who can read history will only apply to future messages in this room": "Changes to who can read history will only apply to future messages in this room",
"The visibility of existing history will be unchanged": "The visibility of existing history will be unchanged",
@@ -725,28 +727,21 @@
"(warning: cannot be disabled again!)": "(warning: cannot be disabled again!)",
"Encryption is enabled in this room": "Encryption is enabled in this room",
"Encryption is not enabled in this room": "Encryption is not enabled in this room",
- "The default role for new room members is": "The default role for new room members is",
- "To send messages, you must be a": "To send messages, you must be a",
- "To invite users into the room, you must be a": "To invite users into the room, you must be a",
- "To configure the room, you must be a": "To configure the room, you must be a",
- "To kick users, you must be a": "To kick users, you must be a",
- "To ban users, you must be a": "To ban users, you must be a",
- "To remove other users' messages, you must be a": "To remove other users' messages, you must be a",
- "To notify everyone in the room, you must be a": "To notify everyone in the room, you must be a",
- "No users have specific privileges in this room": "No users have specific privileges in this room",
- "%(user)s is a %(userRole)s": "%(user)s is a %(userRole)s",
- "Privileged Users": "Privileged Users",
- "Muted Users": "Muted Users",
- "Banned users": "Banned users",
- "This room is not accessible by remote Matrix servers": "This room is not accessible by remote Matrix servers",
"Favourite": "Favourite",
"Tagged as: ": "Tagged as: ",
"To link to a room it must have an address.": "To link to a room it must have an address.",
- "To send events of type , you must be a": "To send events of type , you must be a",
- "Upgrade room to version %(ver)s": "Upgrade room to version %(ver)s",
- "Open Devtools": "Open Devtools",
+ "Guests cannot join this room even if explicitly invited.": "Guests cannot join this room even if explicitly invited.",
+ "Click here to fix": "Click here to fix",
+ "Who can access this room?": "Who can access this room?",
+ "Only people who have been invited": "Only people who have been invited",
+ "Anyone who knows the room's link, apart from guests": "Anyone who knows the room's link, apart from guests",
+ "Anyone who knows the room's link, including guests": "Anyone who knows the room's link, including guests",
"Publish this room to the public in %(domain)s's room directory?": "Publish this room to the public in %(domain)s's room directory?",
- "Permissions": "Permissions",
+ "Who can read history?": "Who can read history?",
+ "Anyone": "Anyone",
+ "Members only (since the point in time of selecting this option)": "Members only (since the point in time of selecting this option)",
+ "Members only (since they were invited)": "Members only (since they were invited)",
+ "Members only (since they joined)": "Members only (since they joined)",
"Internal room ID: ": "Internal room ID: ",
"Room version number: ": "Room version number: ",
"Add a topic": "Add a topic",
@@ -1068,7 +1063,6 @@
"To help avoid duplicate issues, please view existing issues first (and add a +1) or create a new issue if you can't find it.": "To help avoid duplicate issues, please view existing issues first (and add a +1) or create a new issue if you can't find it.",
"Report bugs & give feedback": "Report bugs & give feedback",
"Go back": "Go back",
- "Roles & Permissions": "Roles & Permissions",
"Visit old settings": "Visit old settings",
"Failed to upgrade room": "Failed to upgrade room",
"The room upgrade could not be completed": "The room upgrade could not be completed",