element-web/src/components/views/rooms/MemberInfo.js

752 lines
27 KiB
JavaScript
Raw Normal View History

2015-09-18 20:39:16 +03:00
/*
2016-01-07 07:06:39 +03:00
Copyright 2015, 2016 OpenMarket Ltd
2017-02-14 16:57:22 +03:00
Copyright 2017 Vector Creations Ltd
2015-09-18 20:39:16 +03:00
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.
*/
/*
* State vars:
* 'can': {
* kick: boolean,
* ban: boolean,
* mute: boolean,
* modifyLevel: boolean
* },
* 'muted': boolean,
* 'isTargetMod': boolean
*/
import React from 'react';
import classNames from 'classnames';
import dis from '../../../dispatcher';
import Modal from '../../../Modal';
import sdk from '../../../index';
import createRoom from '../../../createRoom';
import DMRoomMap from '../../../utils/DMRoomMap';
import Unread from '../../../Unread';
2017-02-14 17:33:21 +03:00
import { findReadReceiptFromUserId } from '../../../utils/Receipt';
import WithMatrixClient from '../../../wrappers/WithMatrixClient';
2017-01-25 01:41:52 +03:00
import AccessibleButton from '../elements/AccessibleButton';
2015-09-18 20:39:16 +03:00
module.exports = WithMatrixClient(React.createClass({
2015-11-26 20:49:39 +03:00
displayName: 'MemberInfo',
2015-09-18 20:39:16 +03:00
propTypes: {
matrixClient: React.PropTypes.object.isRequired,
member: React.PropTypes.object.isRequired,
},
getInitialState: function() {
return {
can: {
kick: false,
ban: false,
mute: false,
modifyLevel: false
},
muted: false,
isTargetMod: false,
updating: 0,
devicesLoading: true,
devices: null,
};
2015-09-18 20:39:16 +03:00
},
componentWillMount: function() {
this._cancelDeviceList = null;
// only display the devices list if our client supports E2E
this._enableDevices = this.props.matrixClient.isCryptoEnabled();
const cli = this.props.matrixClient;
cli.on("deviceVerificationChanged", this.onDeviceVerificationChanged);
cli.on("Room", this.onRoom);
cli.on("deleteRoom", this.onDeleteRoom);
cli.on("Room.timeline", this.onRoomTimeline);
cli.on("Room.name", this.onRoomName);
cli.on("Room.receipt", this.onRoomReceipt);
cli.on("RoomState.events", this.onRoomStateEvents);
cli.on("RoomMember.name", this.onRoomMemberName);
cli.on("accountData", this.onAccountData);
},
componentDidMount: function() {
this._updateStateForNewMember(this.props.member);
},
componentWillReceiveProps: function(newProps) {
if (this.props.member.userId != newProps.member.userId) {
this._updateStateForNewMember(newProps.member);
}
},
componentWillUnmount: function() {
var client = this.props.matrixClient;
if (client) {
client.removeListener("deviceVerificationChanged", this.onDeviceVerificationChanged);
client.removeListener("Room", this.onRoom);
client.removeListener("deleteRoom", this.onDeleteRoom);
client.removeListener("Room.timeline", this.onRoomTimeline);
client.removeListener("Room.name", this.onRoomName);
client.removeListener("Room.receipt", this.onRoomReceipt);
client.removeListener("RoomState.events", this.onRoomStateEvents);
client.removeListener("RoomMember.name", this.onRoomMemberName);
client.removeListener("accountData", this.onAccountData);
}
if (this._cancelDeviceList) {
this._cancelDeviceList();
}
},
_disambiguateDevices: function(devices) {
var names = Object.create(null);
for (var i = 0; i < devices.length; i++) {
var name = devices[i].getDisplayName();
var indexList = names[name] || [];
indexList.push(i);
names[name] = indexList;
}
for (name in names) {
if (names[name].length > 1) {
names[name].forEach((j)=>{
devices[j].ambiguous = true;
});
}
}
},
onDeviceVerificationChanged: function(userId, device) {
if (!this._enableDevices) {
return;
}
if (userId == this.props.member.userId) {
// no need to re-download the whole thing; just update our copy of
// the list.
var devices = this.props.matrixClient.getStoredDevicesForUser(userId);
this.setState({devices: devices});
}
},
onRoom: function(room) {
this.forceUpdate();
},
onDeleteRoom: function(roomId) {
this.forceUpdate();
},
onRoomTimeline: function(ev, room, toStartOfTimeline) {
if (toStartOfTimeline) return;
this.forceUpdate();
},
onRoomName: function(room) {
this.forceUpdate();
},
onRoomReceipt: function(receiptEvent, room) {
// because if we read a notification, it will affect notification count
// only bother updating if there's a receipt from us
2017-02-14 16:58:29 +03:00
if (findReadReceiptFromUserId(receiptEvent, this.props.matrixClient.credentials.userId)) {
this.forceUpdate();
}
},
onRoomStateEvents: function(ev, state) {
this.forceUpdate();
},
onRoomMemberName: function(ev, member) {
this.forceUpdate();
},
onAccountData: function(ev) {
if (ev.getType() == 'm.direct') {
this.forceUpdate();
}
},
_updateStateForNewMember: function(member) {
var newState = this._calculateOpsPermissions(member);
newState.devicesLoading = true;
newState.devices = null;
this.setState(newState);
if (this._cancelDeviceList) {
this._cancelDeviceList();
this._cancelDeviceList = null;
}
this._downloadDeviceList(member);
},
_downloadDeviceList: function(member) {
if (!this._enableDevices) {
return;
}
var cancelled = false;
this._cancelDeviceList = function() { cancelled = true; };
var client = this.props.matrixClient;
var self = this;
client.downloadKeys([member.userId], true).finally(function() {
self._cancelDeviceList = null;
}).done(function() {
if (cancelled) {
// we got cancelled - presumably a different user now
return;
}
var devices = client.getStoredDevicesForUser(member.userId);
self._disambiguateDevices(devices);
self.setState({devicesLoading: false, devices: devices});
}, function(err) {
console.log("Error downloading devices", err);
self.setState({devicesLoading: false});
});
},
2015-09-18 20:39:16 +03:00
onKick: function() {
const membership = this.props.member.membership;
const kickLabel = membership === "invite" ? "Disinvite" : "Kick";
const ConfirmUserActionDialog = sdk.getComponent("dialogs.ConfirmUserActionDialog");
Modal.createDialog(ConfirmUserActionDialog, {
member: this.props.member,
action: kickLabel,
askReason: membership == "join",
danger: true,
onFinished: (proceed, reason) => {
if (!proceed) return;
this.setState({ updating: this.state.updating + 1 });
this.props.matrixClient.kick(
this.props.member.roomId, this.props.member.userId,
reason || undefined
).then(function() {
// NO-OP; rely on the m.room.member event coming down else we could
// get out of sync if we force setState here!
console.log("Kick success");
}, function(err) {
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
2017-03-13 01:59:41 +03:00
console.error("Kick error: " + err);
Modal.createDialog(ErrorDialog, {
title: "Failed to kick",
description: ((err && err.message) ? err.message : "Operation failed"),
});
}
).finally(()=>{
this.setState({ updating: this.state.updating - 1 });
});
}
});
2015-09-18 20:39:16 +03:00
},
2017-02-14 20:29:40 +03:00
onBanOrUnban: function() {
const ConfirmUserActionDialog = sdk.getComponent("dialogs.ConfirmUserActionDialog");
Modal.createDialog(ConfirmUserActionDialog, {
member: this.props.member,
action: this.props.member.membership == 'ban' ? 'Unban' : 'Ban',
askReason: this.props.member.membership != 'ban',
danger: this.props.member.membership != 'ban',
onFinished: (proceed, reason) => {
if (!proceed) return;
this.setState({ updating: this.state.updating + 1 });
let promise;
if (this.props.member.membership == 'ban') {
promise = this.props.matrixClient.unban(
this.props.member.roomId, this.props.member.userId,
);
} else {
promise = this.props.matrixClient.ban(
this.props.member.roomId, this.props.member.userId,
reason || undefined
);
}
promise.then(
function() {
// NO-OP; rely on the m.room.member event coming down else we could
// get out of sync if we force setState here!
console.log("Ban success");
}, function(err) {
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
2017-03-13 01:59:41 +03:00
console.error("Ban error: " + err);
Modal.createDialog(ErrorDialog, {
2017-03-13 01:59:41 +03:00
title: "Error",
description: "Failed to ban user",
});
}
).finally(()=>{
this.setState({ updating: this.state.updating - 1 });
});
},
});
2015-09-18 20:39:16 +03:00
},
onMuteToggle: function() {
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
2015-09-18 20:39:16 +03:00
var roomId = this.props.member.roomId;
var target = this.props.member.userId;
var room = this.props.matrixClient.getRoom(roomId);
2015-09-18 20:39:16 +03:00
if (!room) {
return;
}
var powerLevelEvent = room.currentState.getStateEvents(
"m.room.power_levels", ""
);
if (!powerLevelEvent) {
return;
}
var isMuted = this.state.muted;
var powerLevels = powerLevelEvent.getContent();
var levelToSend = (
(powerLevels.events ? powerLevels.events["m.room.message"] : null) ||
powerLevels.events_default
);
var level;
if (isMuted) { // unmute
level = levelToSend;
}
else { // mute
level = levelToSend - 1;
}
2016-01-22 18:29:57 +03:00
level = parseInt(level);
2015-09-18 20:39:16 +03:00
2016-01-22 18:29:57 +03:00
if (level !== NaN) {
this.setState({ updating: this.state.updating + 1 });
this.props.matrixClient.setPowerLevel(roomId, target, level, powerLevelEvent).then(
2016-01-22 18:29:57 +03:00
function() {
// NO-OP; rely on the m.room.member event coming down else we could
// get out of sync if we force setState here!
console.log("Mute toggle success");
}, function(err) {
2017-03-13 01:59:41 +03:00
console.error("Mute error: " + err);
2016-01-22 18:29:57 +03:00
Modal.createDialog(ErrorDialog, {
2017-03-13 01:59:41 +03:00
title: "Error",
description: "Failed to mute user",
2016-01-22 18:29:57 +03:00
});
}
).finally(()=>{
this.setState({ updating: this.state.updating - 1 });
});
2016-01-22 18:29:57 +03:00
}
2015-09-18 20:39:16 +03:00
},
onModToggle: function() {
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
2015-09-18 20:39:16 +03:00
var roomId = this.props.member.roomId;
var target = this.props.member.userId;
var room = this.props.matrixClient.getRoom(roomId);
2015-09-18 20:39:16 +03:00
if (!room) {
return;
}
var powerLevelEvent = room.currentState.getStateEvents(
"m.room.power_levels", ""
);
if (!powerLevelEvent) {
return;
}
var me = room.getMember(this.props.matrixClient.credentials.userId);
2015-09-18 20:39:16 +03:00
if (!me) {
return;
}
var defaultLevel = powerLevelEvent.getContent().users_default;
var modLevel = me.powerLevel - 1;
2016-01-18 04:18:02 +03:00
if (modLevel > 50 && defaultLevel < 50) modLevel = 50; // try to stick with the vector level defaults
2015-09-18 20:39:16 +03:00
// toggle the level
var newLevel = this.state.isTargetMod ? defaultLevel : modLevel;
this.setState({ updating: this.state.updating + 1 });
this.props.matrixClient.setPowerLevel(roomId, target, parseInt(newLevel), powerLevelEvent).then(
function() {
// NO-OP; rely on the m.room.member event coming down else we could
// get out of sync if we force setState here!
console.log("Mod toggle success");
}, function(err) {
2016-03-22 16:19:29 +03:00
if (err.errcode == 'M_GUEST_ACCESS_FORBIDDEN') {
var NeedToRegisterDialog = sdk.getComponent("dialogs.NeedToRegisterDialog");
Modal.createDialog(NeedToRegisterDialog, {
title: "Please Register",
description: "This action cannot be performed by a guest user. Please register to be able to do this."
});
} else {
2017-03-13 01:59:41 +03:00
console.error("Toggle moderator error:" + err);
2016-03-22 16:19:29 +03:00
Modal.createDialog(ErrorDialog, {
2017-03-13 01:59:41 +03:00
title: "Error",
description: "Failed to toggle moderator status",
2016-03-22 16:19:29 +03:00
});
}
}
).finally(()=>{
this.setState({ updating: this.state.updating - 1 });
});
2015-09-18 20:39:16 +03:00
},
2016-03-21 03:49:18 +03:00
_applyPowerChange: function(roomId, target, powerLevel, powerLevelEvent) {
this.setState({ updating: this.state.updating + 1 });
this.props.matrixClient.setPowerLevel(roomId, target, parseInt(powerLevel), powerLevelEvent).then(
2016-03-21 03:49:18 +03:00
function() {
// NO-OP; rely on the m.room.member event coming down else we could
// get out of sync if we force setState here!
console.log("Power change success");
}, function(err) {
2017-01-13 18:17:34 +03:00
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
2017-03-13 01:59:41 +03:00
console.error("Failed to change power level " + err);
2016-03-21 03:49:18 +03:00
Modal.createDialog(ErrorDialog, {
2017-03-13 01:59:41 +03:00
title: "Error",
description: "Failed to change power level",
2016-03-21 03:49:18 +03:00
});
}
).finally(()=>{
this.setState({ updating: this.state.updating - 1 });
2017-01-13 18:17:34 +03:00
}).done();
2016-03-21 03:49:18 +03:00
},
2016-01-18 04:18:02 +03:00
onPowerChange: function(powerLevel) {
var ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
var roomId = this.props.member.roomId;
var target = this.props.member.userId;
var room = this.props.matrixClient.getRoom(roomId);
2016-03-21 03:49:18 +03:00
var self = this;
2016-01-18 04:18:02 +03:00
if (!room) {
return;
}
var powerLevelEvent = room.currentState.getStateEvents(
"m.room.power_levels", ""
);
if (!powerLevelEvent) {
return;
}
2016-03-21 03:49:18 +03:00
if (powerLevelEvent.getContent().users) {
var myPower = powerLevelEvent.getContent().users[this.props.matrixClient.credentials.userId];
2016-03-21 03:49:18 +03:00
if (parseInt(myPower) === parseInt(powerLevel)) {
var QuestionDialog = sdk.getComponent("dialogs.QuestionDialog");
Modal.createDialog(QuestionDialog, {
title: "Warning",
description:
<div>
You will not be able to undo this change as you are promoting the user to have the same power level as yourself.<br/>
Are you sure?
</div>,
button: "Continue",
onFinished: function(confirmed) {
if (confirmed) {
self._applyPowerChange(roomId, target, powerLevel, powerLevelEvent);
}
},
});
}
2016-03-21 03:49:18 +03:00
else {
this._applyPowerChange(roomId, target, powerLevel, powerLevelEvent);
}
}
else {
this._applyPowerChange(roomId, target, powerLevel, powerLevelEvent);
}
},
2016-01-18 04:18:02 +03:00
2016-09-09 18:15:01 +03:00
onNewDMClick: function() {
this.setState({ updating: this.state.updating + 1 });
createRoom({dmUserId: this.props.member.userId}).finally(() => {
2016-09-09 18:15:01 +03:00
this.setState({ updating: this.state.updating - 1 });
}).done();
2015-09-22 18:37:39 +03:00
},
onLeaveClick: function() {
dis.dispatch({
action: 'leave_room',
room_id: this.props.member.roomId,
2015-09-22 18:37:39 +03:00
});
2015-09-18 20:39:16 +03:00
},
_calculateOpsPermissions: function(member) {
const defaultPerms = {
2015-09-18 20:39:16 +03:00
can: {},
muted: false,
modifyLevel: false
};
const room = this.props.matrixClient.getRoom(member.roomId);
2015-09-18 20:39:16 +03:00
if (!room) {
return defaultPerms;
}
const powerLevels = room.currentState.getStateEvents(
2015-09-18 20:39:16 +03:00
"m.room.power_levels", ""
);
if (!powerLevels) {
return defaultPerms;
}
const me = room.getMember(this.props.matrixClient.credentials.userId);
if (!me) {
return defaultPerms;
}
const them = member;
2015-09-18 20:39:16 +03:00
return {
can: this._calculateCanPermissions(
me, them, powerLevels.getContent()
),
muted: this._isMuted(them, powerLevels.getContent()),
isTargetMod: them.powerLevel > powerLevels.getContent().users_default
};
},
_calculateCanPermissions: function(me, them, powerLevels) {
const can = {
2015-09-18 20:39:16 +03:00
kick: false,
ban: false,
mute: false,
modifyLevel: false
};
const canAffectUser = them.powerLevel < me.powerLevel;
2015-09-18 20:39:16 +03:00
if (!canAffectUser) {
//console.log("Cannot affect user: %s >= %s", them.powerLevel, me.powerLevel);
return can;
}
const editPowerLevel = (
2015-09-18 20:39:16 +03:00
(powerLevels.events ? powerLevels.events["m.room.power_levels"] : null) ||
powerLevels.state_default
);
const levelToSend = (
2016-01-18 04:40:19 +03:00
(powerLevels.events ? powerLevels.events["m.room.message"] : null) ||
powerLevels.events_default
);
2015-09-18 20:39:16 +03:00
can.kick = me.powerLevel >= powerLevels.kick;
can.ban = me.powerLevel >= powerLevels.ban;
can.mute = me.powerLevel >= editPowerLevel;
2016-01-18 04:40:19 +03:00
can.toggleMod = me.powerLevel > them.powerLevel && them.powerLevel >= levelToSend;
2015-09-18 20:39:16 +03:00
can.modifyLevel = me.powerLevel > them.powerLevel;
return can;
},
_isMuted: function(member, powerLevelContent) {
if (!powerLevelContent || !member) {
return false;
}
var levelToSend = (
(powerLevelContent.events ? powerLevelContent.events["m.room.message"] : null) ||
powerLevelContent.events_default
);
return member.powerLevel < levelToSend;
2015-11-26 20:49:39 +03:00
},
onCancel: function(e) {
dis.dispatch({
action: "view_user",
member: null
});
},
onMemberAvatarClick: function() {
2016-08-25 18:20:31 +03:00
var avatarUrl = this.props.member.user ? this.props.member.user.avatarUrl : this.props.member.events.member.getContent().avatar_url;
if(!avatarUrl) return;
2016-04-04 02:18:18 +03:00
var httpUrl = this.props.matrixClient.mxcUrlToHttp(avatarUrl);
var ImageView = sdk.getComponent("elements.ImageView");
var params = {
src: httpUrl,
name: this.props.member.name
};
Modal.createDialog(ImageView, params, "mx_Dialog_lightbox");
},
onRoomTileClick(roomId) {
dis.dispatch({
action: 'view_room',
room_id: roomId,
});
},
_renderDevices: function() {
if (!this._enableDevices) {
return null;
}
var devices = this.state.devices;
var MemberDeviceInfo = sdk.getComponent('rooms.MemberDeviceInfo');
var Spinner = sdk.getComponent("elements.Spinner");
var devComponents;
if (this.state.devicesLoading) {
// still loading
devComponents = <Spinner />;
} else if (devices === null) {
devComponents = "Unable to load device list";
} else if (devices.length === 0) {
2017-03-06 19:01:22 +03:00
devComponents = "No devices with registered encryption keys";
} else {
devComponents = [];
for (var i = 0; i < devices.length; i++) {
devComponents.push(<MemberDeviceInfo key={i}
userId={this.props.member.userId}
device={devices[i]}/>);
}
}
return (
<div>
<h3>Devices</h3>
<div className="mx_MemberInfo_devices">
{devComponents}
</div>
</div>
);
},
2015-11-26 20:49:39 +03:00
render: function() {
2016-01-18 04:18:02 +03:00
var startChat, kickButton, banButton, muteButton, giveModButton, spinner;
if (this.props.member.userId !== this.props.matrixClient.credentials.userId) {
const dmRoomMap = new DMRoomMap(this.props.matrixClient);
2016-09-09 18:15:01 +03:00
const dmRooms = dmRoomMap.getDMRoomsForUserId(this.props.member.userId);
const RoomTile = sdk.getComponent("rooms.RoomTile");
const tiles = [];
for (const roomId of dmRooms) {
const room = this.props.matrixClient.getRoom(roomId);
2016-09-09 18:15:01 +03:00
if (room) {
const me = room.getMember(this.props.matrixClient.credentials.userId);
2016-09-09 18:15:01 +03:00
const highlight = (
room.getUnreadNotificationCount('highlight') > 0 ||
me.membership == "invite"
);
tiles.push(
<RoomTile key={room.roomId} room={room}
collapsed={false}
selected={false}
unread={Unread.doesRoomHaveUnreadMessages(room)}
highlight={highlight}
isInvite={me.membership == "invite"}
onClick={this.onRoomTileClick}
2016-09-09 18:15:01 +03:00
/>
);
}
}
2016-09-09 18:15:01 +03:00
const labelClasses = classNames({
mx_MemberInfo_createRoom_label: true,
mx_RoomTile_name: true,
});
2017-01-13 19:25:26 +03:00
const startNewChat = <AccessibleButton
2016-09-09 18:15:01 +03:00
className="mx_MemberInfo_createRoom"
onClick={this.onNewDMClick}
>
<div className="mx_RoomTile_avatar">
<img src="img/create-big.svg" width="26" height="26" />
</div>
2016-09-20 17:06:48 +03:00
<div className={labelClasses}><i>Start new chat</i></div>
</AccessibleButton>;
2016-09-09 18:15:01 +03:00
startChat = <div>
2016-09-16 16:35:43 +03:00
<h3>Direct chats</h3>
2016-09-09 18:15:01 +03:00
{tiles}
{startNewChat}
</div>;
2015-11-26 20:49:39 +03:00
}
if (this.state.updating) {
2015-11-26 20:49:39 +03:00
var Loader = sdk.getComponent("elements.Spinner");
spinner = <Loader imgClassName="mx_ContextualMenu_spinner"/>;
}
if (this.state.can.kick) {
2017-01-25 01:41:52 +03:00
const membership = this.props.member.membership;
const kickLabel = membership === "invite" ? "Disinvite" : "Kick";
kickButton = (
<AccessibleButton className="mx_MemberInfo_field"
onClick={this.onKick}>
{kickLabel}
</AccessibleButton>
);
2015-11-26 20:49:39 +03:00
}
if (this.state.can.ban) {
let label = 'Ban';
if (this.props.member.membership == 'ban') {
label = 'Unban';
}
2017-01-25 01:41:52 +03:00
banButton = (
<AccessibleButton className="mx_MemberInfo_field"
2017-02-14 20:29:40 +03:00
onClick={this.onBanOrUnban}>
{label}
2017-01-25 01:41:52 +03:00
</AccessibleButton>
);
2015-11-26 20:49:39 +03:00
}
if (this.state.can.mute) {
2017-01-25 01:41:52 +03:00
const muteLabel = this.state.muted ? "Unmute" : "Mute";
muteButton = (
<AccessibleButton className="mx_MemberInfo_field"
onClick={this.onMuteToggle}>
{muteLabel}
</AccessibleButton>
);
2015-11-26 20:49:39 +03:00
}
2016-01-18 04:40:19 +03:00
if (this.state.can.toggleMod) {
2016-01-18 04:18:02 +03:00
var giveOpLabel = this.state.isTargetMod ? "Revoke Moderator" : "Make Moderator";
2017-01-13 19:25:26 +03:00
giveModButton = <AccessibleButton className="mx_MemberInfo_field" onClick={this.onModToggle}>
2015-11-26 20:49:39 +03:00
{giveOpLabel}
</AccessibleButton>;
2015-11-26 20:49:39 +03:00
}
2016-01-18 04:26:15 +03:00
// TODO: we should have an invite button if this MemberInfo is showing a user who isn't actually in the current room yet
// e.g. clicking on a linkified userid in a room
2016-01-18 04:18:02 +03:00
var adminTools;
if (kickButton || banButton || muteButton || giveModButton) {
adminTools =
2016-01-18 04:18:02 +03:00
<div>
<h3>Admin tools</h3>
<div className="mx_MemberInfo_buttons">
{muteButton}
{kickButton}
{banButton}
{giveModButton}
</div>
</div>;
2016-01-18 04:18:02 +03:00
}
const memberName = this.props.member.name;
2016-07-05 07:54:18 +03:00
2015-11-26 20:49:39 +03:00
var MemberAvatar = sdk.getComponent('avatars.MemberAvatar');
2016-01-18 04:18:02 +03:00
var PowerSelector = sdk.getComponent('elements.PowerSelector');
const EmojiText = sdk.getComponent('elements.EmojiText');
2015-11-26 20:49:39 +03:00
return (
<div className="mx_MemberInfo">
2017-01-13 19:25:26 +03:00
<AccessibleButton className="mx_MemberInfo_cancel" onClick={this.onCancel}> <img src="img/cancel.svg" width="18" height="18"/></AccessibleButton>
2015-11-26 20:49:39 +03:00
<div className="mx_MemberInfo_avatar">
<MemberAvatar onClick={this.onMemberAvatarClick} member={this.props.member} width={48} height={48} />
2015-11-26 20:49:39 +03:00
</div>
2016-01-18 04:18:02 +03:00
2016-08-11 13:34:14 +03:00
<EmojiText element="h2">{memberName}</EmojiText>
2016-01-18 04:18:02 +03:00
<div className="mx_MemberInfo_profile">
<div className="mx_MemberInfo_profileField">
{ this.props.member.userId }
</div>
<div className="mx_MemberInfo_profileField">
Level: <b><PowerSelector controlled={true} value={ parseInt(this.props.member.powerLevel) } disabled={ !this.state.can.modifyLevel } onChange={ this.onPowerChange }/></b>
2016-01-18 04:18:02 +03:00
</div>
2015-11-26 20:49:39 +03:00
</div>
2016-01-18 04:18:02 +03:00
2016-09-17 17:07:41 +03:00
{ adminTools }
2016-01-18 04:18:02 +03:00
{ startChat }
{ this._renderDevices() }
2016-01-18 04:18:02 +03:00
{ spinner }
2015-11-26 20:49:39 +03:00
</div>
);
2015-09-18 20:39:16 +03:00
}
}));