/* Copyright 2015, 2016 OpenMarket 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. */ var React = require('react'); var MatrixClientPeg = require('../../../MatrixClientPeg'); var Tinter = require('../../../Tinter'); var sdk = require('../../../index'); var Modal = require('../../../Modal'); var room_colors = [ // magic room default values courtesy of Ribot ["#76cfa6", "#eaf5f0"], ["#81bddb", "#eaf1f4"], ["#bd79cb", "#f3eaf5"], ["#c65d94", "#f5eaef"], ["#e55e5e", "#f5eaea"], ["#eca46f", "#f5eeea"], ["#dad658", "#f5f4ea"], ["#80c553", "#eef5ea"], ["#bb814e", "#eee8e3"], ["#595959", "#ececec"], ]; module.exports = React.createClass({ displayName: 'RoomSettings', propTypes: { room: React.PropTypes.object.isRequired, onSaveClick: React.PropTypes.func, onCancelClick: React.PropTypes.func, }, componentDidMount: function() { // XXX: dirty hack to gutwrench to focus on the invite box if (this.props.room.getJoinedMembers().length == 1) { var inviteBox = document.getElementById("mx_MemberList_invite"); if (inviteBox) setTimeout(function() { inviteBox.focus(); }, 0); } }, getInitialState: function() { // work out the initial color index var room_color_index = undefined; var color_scheme_event = this.props.room.getAccountData("org.matrix.room.color_scheme"); if (color_scheme_event) { var color_scheme = color_scheme_event.getContent(); if (color_scheme.primary_color) color_scheme.primary_color = color_scheme.primary_color.toLowerCase(); if (color_scheme.secondary_color) color_scheme.secondary_color = color_scheme.secondary_color.toLowerCase(); // XXX: we should validate these values for (var i = 0; i < room_colors.length; i++) { var room_color = room_colors[i]; if (room_color[0] === color_scheme.primary_color && room_color[1] === color_scheme.secondary_color) { room_color_index = i; break; } } if (room_color_index === undefined) { // append the unrecognised colours to our palette room_color_index = room_colors.length; room_colors[room_color_index] = [ color_scheme.primary_color, color_scheme.secondary_color ]; } } else { room_color_index = 0; } // get the aliases var aliases = {}; var domain = MatrixClientPeg.get().getDomain(); var alias_events = this.props.room.currentState.getStateEvents('m.room.aliases'); for (var i = 0; i < alias_events.length; i++) { aliases[alias_events[i].getStateKey()] = alias_events[i].getContent().aliases.slice(); // shallow copy } aliases[domain] = aliases[domain] || []; var tags = {}; Object.keys(this.props.room.tags).forEach(function(tagName) { tags[tagName] = {}; }); return { power_levels_changed: false, color_scheme_changed: false, color_scheme_index: room_color_index, aliases_changed: false, aliases: aliases, tags_changed: false, tags: tags, }; }, resetState: function() { this.set.state(this.getInitialState()); }, canGuestsJoin: function() { return this.refs.guests_join.checked; }, canGuestsRead: function() { return this.refs.guests_read.checked; }, getTopic: function() { return this.refs.topic ? this.refs.topic.value : ""; }, getJoinRules: function() { return this.refs.is_private.checked ? "invite" : "public"; }, getHistoryVisibility: function() { return this.refs.share_history.checked ? "shared" : "invited"; }, getPowerLevels: function() { if (!this.state.power_levels_changed) return undefined; var power_levels = this.props.room.currentState.getStateEvents('m.room.power_levels', ''); power_levels = power_levels.getContent(); var new_power_levels = { ban: parseInt(this.refs.ban.getValue()), kick: parseInt(this.refs.kick.getValue()), redact: parseInt(this.refs.redact.getValue()), invite: parseInt(this.refs.invite.getValue()), events_default: parseInt(this.refs.events_default.getValue()), state_default: parseInt(this.refs.state_default.getValue()), users_default: parseInt(this.refs.users_default.getValue()), users: power_levels.users, events: power_levels.events, }; return new_power_levels; }, getCanonicalAlias: function() { return this.refs.canonical_alias ? this.refs.canonical_alias.value : ""; }, getAliasOperations: function() { if (!this.state.aliases_changed) return undefined; // work out the delta from room state to UI state var ops = []; // calculate original ("old") aliases var oldAliases = {}; var aliases = this.state.aliases; var alias_events = this.props.room.currentState.getStateEvents('m.room.aliases'); for (var i = 0; i < alias_events.length; i++) { var domain = alias_events[i].getStateKey(); oldAliases[domain] = alias_events[i].getContent().aliases.slice(); // shallow copy } // work out whether any domains have entirely disappeared or appeared var domainDelta = {} Object.keys(oldAliases).forEach(function(domain) { domainDelta[domain] = domainDelta[domain] || 0; domainDelta[domain]--; }); Object.keys(aliases).forEach(function(domain) { domainDelta[domain] = domainDelta[domain] || 0; domainDelta[domain]++; }); Object.keys(domainDelta).forEach(function(domain) { switch (domainDelta[domain]) { case 1: // entirely new domain aliases[domain].forEach(function(alias) { ops.push({ type: "put", alias : alias }); }); break; case -1: // entirely removed domain oldAliases[domain].forEach(function(alias) { ops.push({ type: "delete", alias : alias }); }); break; case 0: // mix of aliases in this domain. // compare old & new aliases for this domain var delta = {}; oldAliases[domain].forEach(function(item) { delta[item] = delta[item] || 0; delta[item]--; }); aliases[domain].forEach(function(item) { delta[item] = delta[item] || 0; delta[item]++; }); Object.keys(delta).forEach(function(alias) { if (delta[alias] == 1) { ops.push({ type: "put", alias: alias }); } else if (delta[alias] == -1) { ops.push({ type: "delete", alias: alias }); } }); break; } }); return ops; }, onPowerLevelsChanged: function() { this.setState({ power_levels_changed: true }); }, getColorScheme: function() { if (!this.state.color_scheme_changed) return undefined; return { primary_color: room_colors[this.state.color_scheme_index][0], secondary_color: room_colors[this.state.color_scheme_index][1], }; }, onColorSchemeChanged: function(index) { // preview what the user just changed the scheme to. Tinter.tint(room_colors[index][0], room_colors[index][1]); this.setState({ color_scheme_changed: true, color_scheme_index: index, }); }, onAliasChanged: function(domain, index, alias) { if (alias === "") return; // hit the delete button to delete please var oldAlias; if (this.isAliasValid(alias)) { oldAlias = this.state.aliases[domain][index]; this.state.aliases[domain][index] = alias; this.setState({ aliases_changed : true }); } else { var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); Modal.createDialog(ErrorDialog, { title: "Invalid alias format", description: "'" + alias + "' is not a valid format for an alias", }); } }, onAliasDeleted: function(domain, index) { // XXX: can we edit state directly and then set, or should we copy it first? var alias = this.state.aliases[domain].splice(index, 1); this.setState({ aliases: this.state.aliases }); this.setState({ aliases_changed : true }); }, onAliasAdded: function(alias) { if (alias === "") return; // ignore attempts to create blank aliases if (alias === undefined) { alias = this.refs.add_alias ? this.refs.add_alias.getValue() : undefined; if (alias === undefined || alias === "") return; } if (this.isAliasValid(alias)) { var domain = alias.replace(/^.*?:/, ''); // XXX: do we need to deep copy aliases before editing it? this.state.aliases[domain] = this.state.aliases[domain] || []; this.state.aliases[domain].push(alias); this.setState({ aliases: this.state.aliases }); // reset the add field this.refs.add_alias.setValue(''); this.setState({ aliases_changed : true }); } else { var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog"); Modal.createDialog(ErrorDialog, { title: "Invalid alias format", description: "'" + alias + "' is not a valid format for an alias", }); } }, isAliasValid: function(alias) { // XXX: FIXME SPEC-1 return (alias.match(/^#([^\/:,]+?):(.+)$/) && encodeURI(alias) === alias); }, onTagChange: function(tagName, event) { if (event.target.checked) { if (tagName === 'm.favourite') { delete this.state.tags['m.lowpriority']; } else if (tagName === 'm.lowpriority') { delete this.state.tags['m.favourite']; } this.state.tags[tagName] = this.state.tags[tagName] || {}; } else { delete this.state.tags[tagName]; } // XXX: hacky say to deep-edit state this.setState({ tags: this.state.tags, tags_changed: true }); }, render: function() { // TODO: go through greying out things you don't have permission to change // (or turning them into informative stuff) var EditableText = sdk.getComponent('elements.EditableText'); var PowerSelector = sdk.getComponent('elements.PowerSelector'); var join_rule = this.props.room.currentState.getStateEvents('m.room.join_rules', ''); if (join_rule) join_rule = join_rule.getContent().join_rule; var history_visibility = this.props.room.currentState.getStateEvents('m.room.history_visibility', ''); if (history_visibility) history_visibility = history_visibility.getContent().history_visibility; var power_levels = this.props.room.currentState.getStateEvents('m.room.power_levels', ''); var guest_access = this.props.room.currentState.getStateEvents('m.room.guest_access', ''); if (guest_access) { guest_access = guest_access.getContent().guest_access; } var events_levels = (power_levels ? power_levels.events : {}) || {}; var user_id = MatrixClientPeg.get().credentials.userId; if (power_levels) { power_levels = power_levels.getContent(); var ban_level = parseInt(power_levels.ban); var kick_level = parseInt(power_levels.kick); var redact_level = parseInt(power_levels.redact); var invite_level = parseInt(power_levels.invite || 0); var send_level = parseInt(power_levels.events_default || 0); var state_level = parseInt(power_levels.state_default || 0); var default_user_level = parseInt(power_levels.users_default || 0); if (power_levels.ban == undefined) ban_level = 50; if (power_levels.kick == undefined) kick_level = 50; if (power_levels.redact == undefined) redact_level = 50; var user_levels = power_levels.users || {}; var current_user_level = user_levels[user_id]; if (current_user_level == undefined) current_user_level = default_user_level; var power_level_level = events_levels["m.room.power_levels"]; if (power_level_level == undefined) { power_level_level = state_level; } var can_change_levels = current_user_level >= power_level_level; } else { var ban_level = 50; var kick_level = 50; var redact_level = 50; var invite_level = 0; var send_level = 0; var state_level = 0; var default_user_level = 0; var user_levels = []; var events_levels = []; var current_user_level = 0; var power_level_level = 0; var can_change_levels = false; } var state_default = (parseInt(power_levels ? power_levels.state_default : 0) || 0); var room_aliases_level = state_default; if (events_levels['m.room.aliases'] !== undefined) { room_avatar_level = events_levels['m.room.aliases']; } var can_set_room_aliases = current_user_level >= room_aliases_level; var canonical_alias_level = state_default; if (events_levels['m.room.canonical_alias'] !== undefined) { room_avatar_level = events_levels['m.room.canonical_alias']; } var can_set_canonical_alias = current_user_level >= canonical_alias_level; var tag_level = state_default; if (events_levels['m.tag'] !== undefined) { tag_level = events_levels['m.tag']; } var can_set_tag = current_user_level >= tag_level; var self = this; var canonical_alias_event = this.props.room.currentState.getStateEvents('m.room.canonical_alias', ''); var canonical_alias = canonical_alias_event ? canonical_alias_event.getContent().alias : ""; var domain = MatrixClientPeg.get().getDomain(); var remote_domains = Object.keys(this.state.aliases).filter(function(alias) { return alias !== domain }); var remote_aliases_section; if (remote_domains.length) { remote_aliases_section =
This room can be found elsewhere as:
{ remote_domains.map(function(state_key, i) { self.state.aliases[state_key].map(function(alias, j) { return (
); }); })}
} var canonical_alias_section; if (can_set_canonical_alias) { canonical_alias_section = } else { canonical_alias_section = { canonical_alias || "not set" }; } var aliases_section =

Directory

{ this.state.aliases[domain].length ? "This room can be found on " + domain + " as:" : "This room is not findable on " + domain }
{ this.state.aliases[domain].map(function(alias, i) { var deleteButton; if (can_set_room_aliases) { deleteButton = Delete; } return (
{ deleteButton }
); })}
Add
{ remote_aliases_section }
The official way to refer to this room is: { canonical_alias_section }
; var room_colors_section =

Room Colour

{room_colors.map(function(room_color, i) { var selected; if (i === self.state.color_scheme_index) { selected =
./
} var boundClick = self.onColorSchemeChanged.bind(self, i) return (
{ selected }
); })}
; var user_levels_section; if (user_levels.length) { user_levels_section =
Users with specific roles are:
{Object.keys(user_levels).map(function(user, i) { return (
{ user } is a
); })}
; } else { user_levels_section =
No users have specific privileges in this room.
} var banned = this.props.room.getMembersWithMembership("ban"); var banned_users_section; if (banned.length) { banned_users_section =

Banned users

{banned.map(function(member, i) { return (
{member.userId}
); })}
; } var create_event = this.props.room.currentState.getStateEvents('m.room.create', ''); var unfederatable_section; if (create_event.getContent()["m.federate"] === false) { unfederatable_section =
Ths room is not accessible by remote Matrix servers.
} // TODO: support editing custom events_levels // TODO: support editing custom user_levels var tags = [ { name: "m.favourite", label: "Favourite", ref: "tag_favourite" }, { name: "m.lowpriority", label: "Low priority", ref: "tag_lowpriority" }, ]; Object.keys(this.state.tags).sort().forEach(function(tagName) { if (tagName !== 'm.favourite' && tagName !== 'm.lowpriority') { tags.push({ name: tagName, label: tagName }); } }); var tags_section =
This room is tagged as { can_set_tag ? tags.map(function(tag, i) { return (); }) : tags.map(function(tag) { return tag.label; }).join(", ") }
return (




{ tags_section } { room_colors_section } { aliases_section }

Permissions

The default role for new room members is
To send messages, you must be a
To invite users into the room, you must be a
To configure the room, you must be a
To kick users, you must be a
To ban users, you must be a
To redact messages, you must be a
{Object.keys(events_levels).map(function(event_type, i) { return (
To send events of type { event_type }, you must be a
); })} { unfederatable_section }

Users

Your role in this room is currently .
{ user_levels_section }
{ banned_users_section }

Advanced

This room's internal ID is { this.props.room.roomId }
); } });