diff --git a/res/css/views/rooms/_EntityTile.scss b/res/css/views/rooms/_EntityTile.scss
index 2b6b31acb48..a2867de3a7c 100644
--- a/res/css/views/rooms/_EntityTile.scss
+++ b/res/css/views/rooms/_EntityTile.scss
@@ -1,5 +1,6 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
+Copyright 2020 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.
@@ -19,6 +20,15 @@ limitations under the License.
align-items: center;
color: $primary-fg-color;
cursor: pointer;
+
+ .mx_E2EIcon {
+ margin: 0;
+ position: absolute;
+ bottom: 2px;
+ right: 7px;
+ height: 15px;
+ width: 15px;
+ }
}
.mx_EntityTile:hover {
@@ -30,7 +40,7 @@ limitations under the License.
content: "";
position: absolute;
top: calc(50% - 8px); // center
- right: 10px;
+ right: -8px;
mask: url('$(res)/img/member_chevron.png');
mask-repeat: no-repeat;
width: 16px;
@@ -64,14 +74,6 @@ limitations under the License.
position: relative;
}
-.mx_EntityTile_power {
- position: absolute;
- width: 16px;
- height: 17px;
- top: 0px;
- right: 6px;
-}
-
.mx_EntityTile_name,
.mx_GroupRoomTile_name {
flex: 1 1 0;
@@ -83,6 +85,7 @@ limitations under the License.
.mx_EntityTile_details {
overflow: hidden;
+ flex: 1;
}
.mx_EntityTile_ellipsis .mx_EntityTile_name {
@@ -112,10 +115,6 @@ limitations under the License.
opacity: 0.25;
}
-.mx_EntityTile:not(.mx_EntityTile_noHover):hover .mx_EntityTile_name {
- font-size: 13px;
-}
-
.mx_EntityTile_subtext {
font-size: 11px;
opacity: 0.5;
@@ -123,3 +122,17 @@ limitations under the License.
white-space: nowrap;
text-overflow: clip;
}
+
+.mx_EntityTile_power {
+ padding-inline-start: 6px;
+ font-size: 10px;
+ color: $notice-secondary-color;
+ max-width: 6em;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+}
+
+.mx_EntityTile:hover .mx_EntityTile_power {
+ display: none;
+}
diff --git a/res/img/admin.svg b/res/img/admin.svg
deleted file mode 100644
index 7ea7459304d..00000000000
--- a/res/img/admin.svg
+++ /dev/null
@@ -1,17 +0,0 @@
-
-
diff --git a/res/img/mod.svg b/res/img/mod.svg
deleted file mode 100644
index 847baf98f94..00000000000
--- a/res/img/mod.svg
+++ /dev/null
@@ -1,16 +0,0 @@
-
-
diff --git a/src/Entities.js b/src/Entities.js
deleted file mode 100644
index 872a837f3a6..00000000000
--- a/src/Entities.js
+++ /dev/null
@@ -1,137 +0,0 @@
-/*
-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.
-*/
-
-import * as sdk from './index';
-
-function isMatch(query, name, uid) {
- query = query.toLowerCase();
- name = name.toLowerCase();
- uid = uid.toLowerCase();
-
- // direct prefix matches
- if (name.indexOf(query) === 0 || uid.indexOf(query) === 0) {
- return true;
- }
-
- // strip @ on uid and try matching again
- if (uid.length > 1 && uid[0] === "@" && uid.substring(1).indexOf(query) === 0) {
- return true;
- }
-
- // split spaces in name and try matching constituent parts
- const parts = name.split(" ");
- for (let i = 0; i < parts.length; i++) {
- if (parts[i].indexOf(query) === 0) {
- return true;
- }
- }
- return false;
-}
-
-/*
- * Converts various data models to Entity objects.
- *
- * Entity objects provide an interface for UI components to use to display
- * members in a data-agnostic way. This means they don't need to care if the
- * underlying data model is a RoomMember, User or 3PID data structure, it just
- * cares about rendering.
- */
-
-class Entity {
- constructor(model) {
- this.model = model;
- }
-
- getJsx() {
- return null;
- }
-
- matches(queryString) {
- return false;
- }
-}
-
-class MemberEntity extends Entity {
- getJsx() {
- const MemberTile = sdk.getComponent("rooms.MemberTile");
- return (
-
- );
- }
-
- matches(queryString) {
- return isMatch(queryString, this.model.name, this.model.userId);
- }
-}
-
-class UserEntity extends Entity {
- constructor(model, showInviteButton, inviteFn) {
- super(model);
- this.showInviteButton = Boolean(showInviteButton);
- this.inviteFn = inviteFn;
- this.onClick = this.onClick.bind(this);
- }
-
- onClick() {
- if (this.inviteFn) {
- this.inviteFn(this.model.userId);
- }
- }
-
- getJsx() {
- const UserTile = sdk.getComponent("rooms.UserTile");
- return (
-
- );
- }
-
- matches(queryString) {
- const name = this.model.displayName || this.model.userId;
- return isMatch(queryString, name, this.model.userId);
- }
-}
-
-export function newEntity(jsx, matchFn) {
- const entity = new Entity();
- entity.getJsx = function() {
- return jsx;
- };
- entity.matches = matchFn;
- return entity;
-}
-
-/**
- * @param {RoomMember[]} members
- * @return {Entity[]}
- */
-export function fromRoomMembers(members) {
- return members.map(function(m) {
- return new MemberEntity(m);
- });
-}
-
-/**
- * @param {User[]} users
- * @param {boolean} showInviteButton
- * @param {Function} inviteFn Called with the user ID.
- * @return {Entity[]}
- */
-export function fromUsers(users, showInviteButton, inviteFn) {
- return users.map(function(u) {
- return new UserEntity(u, showInviteButton, inviteFn);
- });
-}
diff --git a/src/components/views/rooms/EntityTile.js b/src/components/views/rooms/EntityTile.js
index 57db1ac2409..fd92cb8e577 100644
--- a/src/components/views/rooms/EntityTile.js
+++ b/src/components/views/rooms/EntityTile.js
@@ -1,6 +1,7 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
Copyright 2018 New Vector Ltd
+Copyright 2020 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.
@@ -22,7 +23,7 @@ import * as sdk from '../../../index';
import AccessibleButton from '../elements/AccessibleButton';
import { _t } from '../../../languageHandler';
import classNames from "classnames";
-
+import E2EIcon from './E2EIcon';
const PRESENCE_CLASS = {
"offline": "mx_EntityTile_offline",
@@ -30,7 +31,6 @@ const PRESENCE_CLASS = {
"unavailable": "mx_EntityTile_unavailable",
};
-
function presenceClassForMember(presenceState, lastActiveAgo, showPresence) {
if (showPresence === false) {
return 'mx_EntityTile_online_beenactive';
@@ -69,6 +69,7 @@ const EntityTile = createReactClass({
suppressOnHover: PropTypes.bool,
showPresence: PropTypes.bool,
subtextLabel: PropTypes.string,
+ e2eStatus: PropTypes.string,
},
getDefaultProps: function() {
@@ -156,18 +157,20 @@ const EntityTile = createReactClass({
);
}
- let power;
+ let powerLabel;
const powerStatus = this.props.powerStatus;
if (powerStatus) {
- const src = {
- [EntityTile.POWER_STATUS_MODERATOR]: require("../../../../res/img/mod.svg"),
- [EntityTile.POWER_STATUS_ADMIN]: require("../../../../res/img/admin.svg"),
- }[powerStatus];
- const alt = {
- [EntityTile.POWER_STATUS_MODERATOR]: _t("Moderator"),
+ const powerText = {
+ [EntityTile.POWER_STATUS_MODERATOR]: _t("Mod"),
[EntityTile.POWER_STATUS_ADMIN]: _t("Admin"),
}[powerStatus];
- power = ;
+ powerLabel =
{powerText}
;
+ }
+
+ let e2eIcon;
+ const { e2eStatus } = this.props;
+ if (e2eStatus) {
+ e2eIcon = ;
}
const BaseAvatar = sdk.getComponent('avatars.BaseAvatar');
@@ -181,9 +184,10 @@ const EntityTile = createReactClass({
onClick={this.props.onClick}>
{ av }
- { power }
+ { e2eIcon }
{ nameEl }
+ { powerLabel }
{ inviteButton }
@@ -194,5 +198,4 @@ const EntityTile = createReactClass({
EntityTile.POWER_STATUS_MODERATOR = "moderator";
EntityTile.POWER_STATUS_ADMIN = "admin";
-
export default EntityTile;
diff --git a/src/components/views/rooms/MemberTile.js b/src/components/views/rooms/MemberTile.js
index 95e54953397..649e1b42779 100644
--- a/src/components/views/rooms/MemberTile.js
+++ b/src/components/views/rooms/MemberTile.js
@@ -1,6 +1,6 @@
/*
Copyright 2015, 2016 OpenMarket Ltd
-Copyright 2019 The Matrix.org Foundation C.I.C.
+Copyright 2019, 2020 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.
@@ -22,6 +22,7 @@ import createReactClass from 'create-react-class';
import * as sdk from "../../../index";
import dis from "../../../dispatcher";
import { _t } from '../../../languageHandler';
+import { MatrixClientPeg } from "../../../MatrixClientPeg";
export default createReactClass({
displayName: 'MemberTile',
@@ -40,29 +41,101 @@ export default createReactClass({
getInitialState: function() {
return {
statusMessage: this.getStatusMessage(),
+ isRoomEncrypted: false,
+ e2eStatus: null,
};
},
componentDidMount() {
- if (!SettingsStore.isFeatureEnabled("feature_custom_status")) {
- return;
+ const cli = MatrixClientPeg.get();
+
+ if (SettingsStore.isFeatureEnabled("feature_custom_status")) {
+ const { user } = this.props.member;
+ if (user) {
+ user.on("User._unstable_statusMessage", this._onStatusMessageCommitted);
+ }
}
- const { user } = this.props.member;
- if (!user) {
- return;
+
+ if (SettingsStore.isFeatureEnabled("feature_cross_signing")) {
+ const { roomId } = this.props.member;
+ if (roomId) {
+ const isRoomEncrypted = cli.isRoomEncrypted(roomId);
+ this.setState({
+ isRoomEncrypted,
+ });
+ if (isRoomEncrypted) {
+ cli.on("userTrustStatusChanged", this.onUserTrustStatusChanged);
+ this.updateE2EStatus();
+ } else {
+ // Listen for room to become encrypted
+ cli.on("RoomState.events", this.onRoomStateEvents);
+ }
+ }
}
- user.on("User._unstable_statusMessage", this._onStatusMessageCommitted);
},
componentWillUnmount() {
+ const cli = MatrixClientPeg.get();
+
const { user } = this.props.member;
- if (!user) {
+ if (user) {
+ user.removeListener(
+ "User._unstable_statusMessage",
+ this._onStatusMessageCommitted,
+ );
+ }
+
+ if (cli) {
+ cli.removeListener("RoomState.events", this.onRoomStateEvents);
+ cli.removeListener("userTrustStatusChanged", this.onUserTrustStatusChanged);
+ }
+ },
+
+ onRoomStateEvents: function(ev) {
+ if (ev.getType() !== "m.room.encryption") return;
+ const { roomId } = this.props.member;
+ if (ev.getRoomId() !== roomId) return;
+
+ // The room is encrypted now.
+ const cli = MatrixClientPeg.get();
+ cli.removeListener("RoomState.events", this.onRoomStateEvents);
+ this.setState({
+ isRoomEncrypted: true,
+ });
+ this.updateE2EStatus();
+ },
+
+ onUserTrustStatusChanged: function(userId, trustStatus) {
+ if (userId !== this.props.member.userId) return;
+ this.updateE2EStatus();
+ },
+
+ updateE2EStatus: async function() {
+ const cli = MatrixClientPeg.get();
+ const { userId } = this.props.member;
+ const isMe = userId === cli.getUserId();
+ const userVerified = cli.checkUserTrust(userId).isCrossSigningVerified();
+ if (!userVerified) {
+ this.setState({
+ e2eStatus: "normal",
+ });
return;
}
- user.removeListener(
- "User._unstable_statusMessage",
- this._onStatusMessageCommitted,
- );
+
+ const devices = await cli.getStoredDevicesForUser(userId);
+ const anyDeviceUnverified = devices.some(device => {
+ const { deviceId } = device;
+ // For your own devices, we use the stricter check of cross-signing
+ // verification to encourage everyone to trust their own devices via
+ // cross-signing so that other users can then safely trust you.
+ // For other people's devices, the more general verified check that
+ // includes locally verified devices can be used.
+ const deviceTrust = cli.checkDeviceTrust(userId, deviceId);
+ return isMe ? !deviceTrust.isCrossSigningVerified() : !deviceTrust.isVerified();
+ });
+ this.setState({
+ e2eStatus: anyDeviceUnverified ? "warning" : "verified",
+ });
},
getStatusMessage() {
@@ -94,6 +167,12 @@ export default createReactClass({
) {
return true;
}
+ if (
+ nextState.isRoomEncrypted !== this.state.isRoomEncrypted ||
+ nextState.e2eStatus !== this.state.e2eStatus
+ ) {
+ return true;
+ }
return false;
},
@@ -153,14 +232,26 @@ export default createReactClass({
const powerStatus = powerStatusMap.get(powerLevel);
+ let e2eStatus;
+ if (this.state.isRoomEncrypted) {
+ e2eStatus = this.state.e2eStatus;
+ }
+
return (
-
);
},
diff --git a/src/components/views/rooms/UserTile.js b/src/components/views/rooms/UserTile.js
deleted file mode 100644
index 01bbe7d4d13..00000000000
--- a/src/components/views/rooms/UserTile.js
+++ /dev/null
@@ -1,53 +0,0 @@
-/*
-Copyright 2015, 2016 OpenMarket Ltd
-Copyright 2019 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 from 'react';
-import PropTypes from 'prop-types';
-import createReactClass from 'create-react-class';
-import * as Avatar from '../../../Avatar';
-import * as sdk from "../../../index";
-
-export default createReactClass({
- displayName: 'UserTile',
-
- propTypes: {
- user: PropTypes.any.isRequired, // User
- },
-
- render: function() {
- const EntityTile = sdk.getComponent("rooms.EntityTile");
- const user = this.props.user;
- const name = user.displayName || user.userId;
- let active = -1;
-
- // FIXME: make presence data update whenever User.presence changes...
- active = user.lastActiveAgo ?
- (Date.now() - (user.lastPresenceTs - user.lastActiveAgo)) : -1;
-
- const BaseAvatar = sdk.getComponent('avatars.BaseAvatar');
- const avatarJsx = (
-
- );
-
- return (
-
- );
- },
-});
diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json
index 6e58a762830..dde006bbd28 100644
--- a/src/i18n/strings/en_EN.json
+++ b/src/i18n/strings/en_EN.json
@@ -909,6 +909,7 @@
"Some sessions in this encrypted room are not trusted": "Some sessions in this encrypted room are not trusted",
"All sessions in this encrypted room are trusted": "All sessions in this encrypted room are trusted",
"Edit message": "Edit message",
+ "Mod": "Mod",
"This event could not be displayed": "This event could not be displayed",
"%(senderName)s sent an image": "%(senderName)s sent an image",
"%(senderName)s sent a video": "%(senderName)s sent a video",
diff --git a/test/components/views/groups/GroupMemberList-test.js b/test/components/views/groups/GroupMemberList-test.js
index 867190f6f41..39720177cc0 100644
--- a/test/components/views/groups/GroupMemberList-test.js
+++ b/test/components/views/groups/GroupMemberList-test.js
@@ -112,7 +112,9 @@ describe("GroupMemberList", function() {
const memberList = ReactTestUtils.findRenderedDOMComponentWithClass(root, "mx_MemberList_joined");
const memberListElement = ReactDOM.findDOMNode(memberList);
expect(memberListElement).toBeTruthy();
- expect(memberListElement.textContent).toBe("Test");
+ const userNameElement = memberListElement.querySelector(".mx_EntityTile_name");
+ expect(userNameElement).toBeTruthy();
+ expect(userNameElement.textContent).toBe("Test");
});
httpBackend.when("GET", "/groups/" + groupIdEncoded + "/summary").respond(200, summaryResponse);