2015-11-27 14:50:33 +03:00
|
|
|
/*
|
2016-01-07 07:06:39 +03:00
|
|
|
Copyright 2015, 2016 OpenMarket Ltd
|
2017-09-11 18:59:09 +03:00
|
|
|
Copyright 2017 New Vector Ltd
|
2018-06-16 11:10:13 +03:00
|
|
|
Copyright 2018 Michael Telatynski <7t3chguy@gmail.com>
|
2019-06-19 13:48:47 +03:00
|
|
|
Copyright 2019 The Matrix.org Foundation C.I.C.
|
2015-11-27 14:50:33 +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.
|
|
|
|
*/
|
|
|
|
|
2019-12-10 03:08:45 +03:00
|
|
|
import React from 'react';
|
2017-12-26 04:03:18 +03:00
|
|
|
import PropTypes from 'prop-types';
|
2019-09-06 17:04:46 +03:00
|
|
|
import createReactClass from 'create-react-class';
|
2018-06-16 11:10:13 +03:00
|
|
|
import classNames from 'classnames';
|
2018-03-14 17:29:55 +03:00
|
|
|
import dis from '../../../dispatcher';
|
2019-12-21 00:13:46 +03:00
|
|
|
import {MatrixClientPeg} from '../../../MatrixClientPeg';
|
2016-10-02 14:57:45 +03:00
|
|
|
import DMRoomMap from '../../../utils/DMRoomMap';
|
2019-12-20 04:19:56 +03:00
|
|
|
import * as sdk from '../../../index';
|
2019-12-03 02:21:59 +03:00
|
|
|
import {ContextMenu, ContextMenuButton, toRightOf} from '../../structures/ContextMenu';
|
2018-06-16 11:10:13 +03:00
|
|
|
import * as RoomNotifs from '../../../RoomNotifs';
|
|
|
|
import * as FormattingUtils from '../../../utils/FormattingUtils';
|
2017-09-11 18:59:09 +03:00
|
|
|
import ActiveRoomObserver from '../../../ActiveRoomObserver';
|
2019-01-17 12:29:37 +03:00
|
|
|
import RoomViewStore from '../../../stores/RoomViewStore';
|
2018-12-19 01:11:08 +03:00
|
|
|
import SettingsStore from "../../../settings/SettingsStore";
|
2019-09-30 18:04:43 +03:00
|
|
|
import {_t} from "../../../languageHandler";
|
2020-01-16 04:45:16 +03:00
|
|
|
import {RovingTabIndexWrapper} from "../../../accessibility/RovingTabIndex";
|
2020-01-21 20:17:40 +03:00
|
|
|
import E2EIcon from './E2EIcon';
|
2020-01-24 15:04:08 +03:00
|
|
|
import InviteOnlyIcon from './InviteOnlyIcon';
|
2020-01-23 14:09:52 +03:00
|
|
|
// eslint-disable-next-line camelcase
|
2020-01-21 20:17:40 +03:00
|
|
|
import rate_limited_func from '../../../ratelimitedfunc';
|
2015-11-27 14:50:33 +03:00
|
|
|
|
2019-12-20 03:45:24 +03:00
|
|
|
export default createReactClass({
|
2015-11-27 14:50:33 +03:00
|
|
|
displayName: 'RoomTile',
|
|
|
|
|
|
|
|
propTypes: {
|
2017-12-26 04:03:18 +03:00
|
|
|
onClick: PropTypes.func,
|
2015-11-27 14:50:33 +03:00
|
|
|
|
2017-12-26 04:03:18 +03:00
|
|
|
room: PropTypes.object.isRequired,
|
|
|
|
collapsed: PropTypes.bool.isRequired,
|
|
|
|
unread: PropTypes.bool.isRequired,
|
|
|
|
highlight: PropTypes.bool.isRequired,
|
2018-03-05 15:36:02 +03:00
|
|
|
// If true, apply mx_RoomTile_transparent class
|
|
|
|
transparent: PropTypes.bool,
|
2017-12-26 04:03:18 +03:00
|
|
|
isInvite: PropTypes.bool.isRequired,
|
|
|
|
incomingCall: PropTypes.object,
|
2015-11-27 14:50:33 +03:00
|
|
|
},
|
|
|
|
|
2016-09-09 18:15:01 +03:00
|
|
|
getDefaultProps: function() {
|
|
|
|
return {
|
|
|
|
isDragging: false,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2015-11-27 14:50:33 +03:00
|
|
|
getInitialState: function() {
|
2020-01-10 20:22:09 +03:00
|
|
|
const joinRules = this.props.room.currentState.getStateEvents("m.room.join_rules", "");
|
2020-01-10 20:36:33 +03:00
|
|
|
const joinRule = joinRules && joinRules.getContent().join_rule;
|
2020-01-10 19:33:08 +03:00
|
|
|
|
2017-11-16 16:19:36 +03:00
|
|
|
return ({
|
2020-01-10 20:22:09 +03:00
|
|
|
joinRule,
|
2017-10-11 19:56:17 +03:00
|
|
|
hover: false,
|
|
|
|
badgeHover: false,
|
2019-12-10 03:08:45 +03:00
|
|
|
contextMenuPosition: null, // DOM bounding box, null if non-shown
|
2018-03-14 17:29:55 +03:00
|
|
|
roomName: this.props.room.name,
|
2016-08-18 16:00:14 +03:00
|
|
|
notifState: RoomNotifs.getRoomNotifsState(this.props.room.roomId),
|
2018-03-14 17:29:55 +03:00
|
|
|
notificationCount: this.props.room.getUnreadNotificationCount(),
|
2019-01-17 12:29:37 +03:00
|
|
|
selected: this.props.room.roomId === RoomViewStore.getRoomId(),
|
2019-01-15 18:20:43 +03:00
|
|
|
statusMessage: this._getStatusMessage(),
|
2020-01-21 20:17:40 +03:00
|
|
|
e2eStatus: null,
|
2016-07-18 18:10:07 +03:00
|
|
|
});
|
2015-11-27 14:50:33 +03:00
|
|
|
},
|
|
|
|
|
2019-01-15 18:20:43 +03:00
|
|
|
_shouldShowStatusMessage() {
|
|
|
|
if (!SettingsStore.isFeatureEnabled("feature_custom_status")) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
const isInvite = this.props.room.getMyMembership() === "invite";
|
|
|
|
const isJoined = this.props.room.getMyMembership() === "join";
|
|
|
|
const looksLikeDm = this.props.room.getInvitedAndJoinedMemberCount() === 2;
|
|
|
|
return !isInvite && isJoined && looksLikeDm;
|
|
|
|
},
|
|
|
|
|
|
|
|
_getStatusMessageUser() {
|
2019-06-05 00:34:05 +03:00
|
|
|
if (!MatrixClientPeg.get()) return null; // We've probably been logged out
|
|
|
|
|
2019-01-15 18:20:43 +03:00
|
|
|
const selfId = MatrixClientPeg.get().getUserId();
|
|
|
|
const otherMember = this.props.room.currentState.getMembersExcept([selfId])[0];
|
|
|
|
if (!otherMember) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return otherMember.user;
|
|
|
|
},
|
|
|
|
|
|
|
|
_getStatusMessage() {
|
|
|
|
const statusUser = this._getStatusMessageUser();
|
|
|
|
if (!statusUser) {
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
return statusUser._unstable_statusMessage;
|
|
|
|
},
|
|
|
|
|
2020-01-21 20:17:40 +03:00
|
|
|
onRoomStateMember: function(ev, state, member) {
|
|
|
|
// we only care about leaving users
|
|
|
|
// because trust state will change if someone joins a megolm session anyway
|
|
|
|
if (member.membership !== "leave") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// ignore members in other rooms
|
|
|
|
if (member.roomId !== this.props.room.roomId) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this._updateE2eStatus();
|
|
|
|
},
|
|
|
|
|
|
|
|
onUserVerificationChanged: function(userId, _trustStatus) {
|
|
|
|
if (!this.props.room.getMember(userId)) {
|
|
|
|
// Not in this room
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this._updateE2eStatus();
|
|
|
|
},
|
|
|
|
|
|
|
|
onRoomTimeline: function(ev, room) {
|
|
|
|
if (!room) return;
|
|
|
|
if (room.roomId != this.props.room.roomId) return;
|
|
|
|
if (ev.getType() !== "m.room.encryption") return;
|
|
|
|
MatrixClientPeg.get().removeListener("Room.timeline", this.onRoomTimeline);
|
|
|
|
this.onFindingRoomToBeEncrypted();
|
|
|
|
},
|
|
|
|
|
2020-01-23 14:09:52 +03:00
|
|
|
onFindingRoomToBeEncrypted: function() {
|
2020-01-21 20:17:40 +03:00
|
|
|
const cli = MatrixClientPeg.get();
|
|
|
|
cli.on("RoomState.members", this.onRoomStateMember);
|
|
|
|
cli.on("userTrustStatusChanged", this.onUserVerificationChanged);
|
|
|
|
|
|
|
|
this._updateE2eStatus();
|
|
|
|
},
|
|
|
|
|
|
|
|
_updateE2eStatus: async function() {
|
|
|
|
const cli = MatrixClientPeg.get();
|
|
|
|
if (!cli.isRoomEncrypted(this.props.room.roomId)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!SettingsStore.isFeatureEnabled("feature_cross_signing")) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Duplication between here and _updateE2eStatus in RoomView
|
|
|
|
const e2eMembers = await this.props.room.getEncryptionTargetMembers();
|
|
|
|
const verified = [];
|
|
|
|
const unverified = [];
|
|
|
|
e2eMembers.map(({userId}) => userId)
|
|
|
|
.filter((userId) => userId !== cli.getUserId())
|
|
|
|
.forEach((userId) => {
|
|
|
|
(cli.checkUserTrust(userId).isCrossSigningVerified() ?
|
2020-01-23 14:09:52 +03:00
|
|
|
verified : unverified).push(userId);
|
2020-01-21 20:17:40 +03:00
|
|
|
});
|
|
|
|
|
|
|
|
/* Check all verified user devices. */
|
|
|
|
for (const userId of verified) {
|
|
|
|
const devices = await cli.getStoredDevicesForUser(userId);
|
|
|
|
const allDevicesVerified = devices.every(({deviceId}) => {
|
|
|
|
return cli.checkDeviceTrust(userId, deviceId).isVerified();
|
|
|
|
});
|
|
|
|
if (!allDevicesVerified) {
|
|
|
|
this.setState({
|
|
|
|
e2eStatus: "warning",
|
|
|
|
});
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.setState({
|
|
|
|
e2eStatus: unverified.length === 0 ? "verified" : "normal",
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2018-03-14 17:29:55 +03:00
|
|
|
onRoomName: function(room) {
|
|
|
|
if (room !== this.props.room) return;
|
|
|
|
this.setState({
|
|
|
|
roomName: this.props.room.name,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2020-01-10 19:33:08 +03:00
|
|
|
onJoinRule: function(ev) {
|
2020-01-10 20:46:14 +03:00
|
|
|
if (ev.getType() !== "m.room.join_rules") return;
|
|
|
|
if (ev.getRoomId() !== this.props.room.roomId) return;
|
2020-01-13 14:18:24 +03:00
|
|
|
this.setState({ joinRule: ev.getContent().join_rule });
|
2020-01-10 19:33:08 +03:00
|
|
|
},
|
|
|
|
|
2017-05-16 18:11:01 +03:00
|
|
|
onAccountData: function(accountDataEvent) {
|
2018-06-16 11:10:13 +03:00
|
|
|
if (accountDataEvent.getType() === 'm.push_rules') {
|
2017-05-16 18:11:01 +03:00
|
|
|
this.setState({
|
|
|
|
notifState: RoomNotifs.getRoomNotifsState(this.props.room.roomId),
|
|
|
|
});
|
|
|
|
}
|
2017-04-19 02:13:01 +03:00
|
|
|
},
|
|
|
|
|
2018-03-14 17:29:55 +03:00
|
|
|
onAction: function(payload) {
|
|
|
|
switch (payload.action) {
|
|
|
|
// XXX: slight hack in order to zero the notification count when a room
|
|
|
|
// is read. Ideally this state would be given to this via props (as we
|
|
|
|
// do with `unread`). This is still better than forceUpdating the entire
|
|
|
|
// RoomList when a room is read.
|
|
|
|
case 'on_room_read':
|
|
|
|
if (payload.roomId !== this.props.room.roomId) break;
|
|
|
|
this.setState({
|
|
|
|
notificationCount: this.props.room.getUnreadNotificationCount(),
|
|
|
|
});
|
2019-01-12 03:24:06 +03:00
|
|
|
break;
|
|
|
|
// RoomTiles are one of the few components that may show custom status and
|
|
|
|
// also remain on screen while in Settings toggling the feature. This ensures
|
|
|
|
// you can clearly see the status hide and show when toggling the feature.
|
|
|
|
case 'feature_custom_status_changed':
|
|
|
|
this.forceUpdate();
|
|
|
|
break;
|
2018-03-14 17:29:55 +03:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-01-17 12:29:37 +03:00
|
|
|
_onActiveRoomChange: function() {
|
2017-09-11 18:59:09 +03:00
|
|
|
this.setState({
|
2019-01-17 12:29:37 +03:00
|
|
|
selected: this.props.room.roomId === RoomViewStore.getRoomId(),
|
2017-09-11 18:59:09 +03:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2019-10-08 14:10:37 +03:00
|
|
|
componentDidMount: function() {
|
2020-01-21 20:17:40 +03:00
|
|
|
/* We bind here rather than in the definition because otherwise we wind up with the
|
|
|
|
method only being callable once every 500ms across all instances, which would be wrong */
|
|
|
|
this._updateE2eStatus = rate_limited_func(this._updateE2eStatus, 500);
|
|
|
|
|
2019-10-08 14:10:37 +03:00
|
|
|
const cli = MatrixClientPeg.get();
|
|
|
|
cli.on("accountData", this.onAccountData);
|
|
|
|
cli.on("Room.name", this.onRoomName);
|
2020-01-10 19:33:08 +03:00
|
|
|
cli.on("RoomState.events", this.onJoinRule);
|
2020-01-21 20:17:40 +03:00
|
|
|
if (cli.isRoomEncrypted(this.props.room.roomId)) {
|
|
|
|
this.onFindingRoomToBeEncrypted();
|
|
|
|
} else {
|
|
|
|
cli.on("Room.timeline", this.onRoomTimeline);
|
|
|
|
}
|
2017-09-11 18:59:09 +03:00
|
|
|
ActiveRoomObserver.addListener(this.props.room.roomId, this._onActiveRoomChange);
|
2018-03-14 17:29:55 +03:00
|
|
|
this.dispatcherRef = dis.register(this.onAction);
|
2019-01-15 18:20:43 +03:00
|
|
|
|
|
|
|
if (this._shouldShowStatusMessage()) {
|
|
|
|
const statusUser = this._getStatusMessageUser();
|
|
|
|
if (statusUser) {
|
2019-10-08 14:10:37 +03:00
|
|
|
statusUser.on("User._unstable_statusMessage", this._onStatusMessageCommitted);
|
2019-01-15 18:20:43 +03:00
|
|
|
}
|
|
|
|
}
|
2017-04-18 21:28:24 +03:00
|
|
|
},
|
|
|
|
|
2017-05-16 18:11:01 +03:00
|
|
|
componentWillUnmount: function() {
|
2017-10-11 19:56:17 +03:00
|
|
|
const cli = MatrixClientPeg.get();
|
2017-05-16 18:11:01 +03:00
|
|
|
if (cli) {
|
|
|
|
MatrixClientPeg.get().removeListener("accountData", this.onAccountData);
|
2018-03-14 17:29:55 +03:00
|
|
|
MatrixClientPeg.get().removeListener("Room.name", this.onRoomName);
|
2020-01-10 19:33:08 +03:00
|
|
|
cli.removeListener("RoomState.events", this.onJoinRule);
|
2020-01-21 20:17:40 +03:00
|
|
|
cli.removeListener("RoomState.members", this.onRoomStateMember);
|
|
|
|
cli.removeListener("userTrustStatusChanged", this.onUserVerificationChanged);
|
|
|
|
cli.removeListener("Room.timeline", this.onRoomTimeline);
|
2017-05-16 18:11:01 +03:00
|
|
|
}
|
2017-09-11 18:59:09 +03:00
|
|
|
ActiveRoomObserver.removeListener(this.props.room.roomId, this._onActiveRoomChange);
|
2018-03-14 17:29:55 +03:00
|
|
|
dis.unregister(this.dispatcherRef);
|
2019-01-15 18:20:43 +03:00
|
|
|
|
|
|
|
if (this._shouldShowStatusMessage()) {
|
|
|
|
const statusUser = this._getStatusMessageUser();
|
|
|
|
if (statusUser) {
|
|
|
|
statusUser.removeListener(
|
|
|
|
"User._unstable_statusMessage",
|
|
|
|
this._onStatusMessageCommitted,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
2018-03-14 17:29:55 +03:00
|
|
|
},
|
|
|
|
|
2018-03-20 20:19:49 +03:00
|
|
|
componentWillReceiveProps: function(props) {
|
|
|
|
// XXX: This could be a lot better - this makes the assumption that
|
|
|
|
// the notification count may have changed when the properties of
|
|
|
|
// the room tile change.
|
|
|
|
this.setState({
|
|
|
|
notificationCount: this.props.room.getUnreadNotificationCount(),
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2018-03-14 17:29:55 +03:00
|
|
|
// Do a simple shallow comparison of props and state to avoid unnecessary
|
|
|
|
// renders. The assumption made here is that only state and props are used
|
|
|
|
// in rendering this component and children.
|
|
|
|
//
|
|
|
|
// RoomList is frequently made to forceUpdate, so this decreases number of
|
|
|
|
// RoomTile renderings.
|
|
|
|
shouldComponentUpdate: function(newProps, newState) {
|
|
|
|
if (Object.keys(newProps).some((k) => newProps[k] !== this.props[k])) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (Object.keys(newState).some((k) => newState[k] !== this.state[k])) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
2017-04-17 22:58:43 +03:00
|
|
|
},
|
|
|
|
|
2019-01-15 18:20:43 +03:00
|
|
|
_onStatusMessageCommitted() {
|
|
|
|
// The status message `User` object has observed a message change.
|
|
|
|
this.setState({
|
|
|
|
statusMessage: this._getStatusMessage(),
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2017-04-20 16:16:45 +03:00
|
|
|
onClick: function(ev) {
|
2017-03-03 16:48:37 +03:00
|
|
|
if (this.props.onClick) {
|
2017-04-20 16:16:45 +03:00
|
|
|
this.props.onClick(this.props.room.roomId, ev);
|
2017-03-03 16:48:37 +03:00
|
|
|
}
|
2015-11-27 14:50:33 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
onMouseEnter: function() {
|
2017-10-11 19:56:17 +03:00
|
|
|
this.setState( { hover: true });
|
2016-09-11 02:30:43 +03:00
|
|
|
this.badgeOnMouseEnter();
|
2015-11-27 14:50:33 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
onMouseLeave: function() {
|
2017-10-11 19:56:17 +03:00
|
|
|
this.setState( { hover: false });
|
2016-09-11 02:30:43 +03:00
|
|
|
this.badgeOnMouseLeave();
|
2015-11-27 14:50:33 +03:00
|
|
|
},
|
|
|
|
|
2016-07-18 18:10:07 +03:00
|
|
|
badgeOnMouseEnter: function() {
|
2016-09-11 02:30:43 +03:00
|
|
|
// Only allow non-guests to access the context menu
|
2016-08-02 16:46:47 +03:00
|
|
|
// and only change it if it needs to change
|
|
|
|
if (!MatrixClientPeg.get().isGuest() && !this.state.badgeHover) {
|
2017-10-11 19:56:17 +03:00
|
|
|
this.setState( { badgeHover: true } );
|
2016-07-27 13:58:40 +03:00
|
|
|
}
|
2016-07-18 18:10:07 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
badgeOnMouseLeave: function() {
|
2017-10-11 19:56:17 +03:00
|
|
|
this.setState( { badgeHover: false } );
|
2016-07-18 18:10:07 +03:00
|
|
|
},
|
|
|
|
|
2019-12-10 03:08:45 +03:00
|
|
|
_showContextMenu: function(boundingClientRect) {
|
2018-06-18 14:05:57 +03:00
|
|
|
// Only allow non-guests to access the context menu
|
|
|
|
if (MatrixClientPeg.get().isGuest()) return;
|
|
|
|
|
2019-11-11 20:53:17 +03:00
|
|
|
const state = {
|
2019-12-10 03:08:45 +03:00
|
|
|
contextMenuPosition: boundingClientRect,
|
2019-11-11 20:53:17 +03:00
|
|
|
};
|
|
|
|
|
2018-06-18 14:05:57 +03:00
|
|
|
// If the badge is clicked, then no longer show tooltip
|
|
|
|
if (this.props.collapsed) {
|
2019-11-11 20:53:17 +03:00
|
|
|
state.hover = false;
|
2018-06-18 14:05:57 +03:00
|
|
|
}
|
|
|
|
|
2019-11-11 20:53:17 +03:00
|
|
|
this.setState(state);
|
|
|
|
},
|
2018-06-18 14:05:57 +03:00
|
|
|
|
2019-12-10 03:08:45 +03:00
|
|
|
onContextMenuButtonClick: function(e) {
|
|
|
|
// Prevent the RoomTile onClick event firing as well
|
|
|
|
e.stopPropagation();
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
|
|
this._showContextMenu(e.target.getBoundingClientRect());
|
|
|
|
},
|
|
|
|
|
|
|
|
onContextMenu: function(e) {
|
|
|
|
// Prevent the native context menu
|
|
|
|
e.preventDefault();
|
|
|
|
|
|
|
|
this._showContextMenu({
|
|
|
|
right: e.clientX,
|
|
|
|
top: e.clientY,
|
|
|
|
height: 0,
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2019-11-11 20:53:17 +03:00
|
|
|
closeMenu: function() {
|
|
|
|
this.setState({
|
2019-12-10 03:08:45 +03:00
|
|
|
contextMenuPosition: null,
|
2019-11-11 20:53:17 +03:00
|
|
|
});
|
|
|
|
this.props.refreshSubList();
|
2016-07-21 19:44:31 +03:00
|
|
|
},
|
|
|
|
|
2015-11-27 14:50:33 +03:00
|
|
|
render: function() {
|
2018-08-02 12:42:05 +03:00
|
|
|
const isInvite = this.props.room.getMyMembership() === "invite";
|
2019-02-11 17:40:17 +03:00
|
|
|
const notificationCount = this.props.notificationCount;
|
2016-01-22 20:22:49 +03:00
|
|
|
// var highlightCount = this.props.room.getUnreadNotificationCount("highlight");
|
|
|
|
|
2019-06-19 13:46:24 +03:00
|
|
|
const notifBadges = notificationCount > 0 && RoomNotifs.shouldShowNotifBadge(this.state.notifState);
|
|
|
|
const mentionBadges = this.props.highlight && RoomNotifs.shouldShowMentionBadge(this.state.notifState);
|
2016-08-18 15:44:58 +03:00
|
|
|
const badges = notifBadges || mentionBadges;
|
2016-08-18 13:58:27 +03:00
|
|
|
|
2018-12-12 07:40:11 +03:00
|
|
|
let subtext = null;
|
2019-01-15 18:20:43 +03:00
|
|
|
if (this._shouldShowStatusMessage()) {
|
|
|
|
subtext = this.state.statusMessage;
|
2018-12-12 07:40:11 +03:00
|
|
|
}
|
|
|
|
|
2019-12-10 03:08:45 +03:00
|
|
|
const isMenuDisplayed = Boolean(this.state.contextMenuPosition);
|
|
|
|
|
2020-01-10 19:49:07 +03:00
|
|
|
const dmUserId = DMRoomMap.shared().getUserIdForRoomId(this.props.room.roomId);
|
|
|
|
|
2017-10-11 19:56:17 +03:00
|
|
|
const classes = classNames({
|
2015-11-27 14:50:33 +03:00
|
|
|
'mx_RoomTile': true,
|
2017-09-11 18:59:09 +03:00
|
|
|
'mx_RoomTile_selected': this.state.selected,
|
2017-05-16 18:11:01 +03:00
|
|
|
'mx_RoomTile_unread': this.props.unread,
|
2016-08-18 17:19:24 +03:00
|
|
|
'mx_RoomTile_unreadNotify': notifBadges,
|
|
|
|
'mx_RoomTile_highlight': mentionBadges,
|
2018-08-02 12:42:05 +03:00
|
|
|
'mx_RoomTile_invited': isInvite,
|
2019-12-10 03:08:45 +03:00
|
|
|
'mx_RoomTile_menuDisplayed': isMenuDisplayed,
|
2016-08-18 13:58:27 +03:00
|
|
|
'mx_RoomTile_noBadges': !badges,
|
2018-03-05 15:36:02 +03:00
|
|
|
'mx_RoomTile_transparent': this.props.transparent,
|
2018-12-12 22:57:48 +03:00
|
|
|
'mx_RoomTile_hasSubtext': subtext && !this.props.collapsed,
|
2015-11-27 14:50:33 +03:00
|
|
|
});
|
|
|
|
|
2017-10-11 19:56:17 +03:00
|
|
|
const avatarClasses = classNames({
|
2016-07-28 19:24:58 +03:00
|
|
|
'mx_RoomTile_avatar': true,
|
|
|
|
});
|
|
|
|
|
2017-10-11 19:56:17 +03:00
|
|
|
const badgeClasses = classNames({
|
2016-07-28 19:24:58 +03:00
|
|
|
'mx_RoomTile_badge': true,
|
2019-12-10 03:08:45 +03:00
|
|
|
'mx_RoomTile_badgeButton': this.state.badgeHover || isMenuDisplayed,
|
2016-07-28 19:24:58 +03:00
|
|
|
});
|
|
|
|
|
2018-03-14 17:29:55 +03:00
|
|
|
let name = this.state.roomName;
|
2018-09-18 02:16:25 +03:00
|
|
|
if (name == undefined || name == null) name = '';
|
2015-11-27 14:50:33 +03:00
|
|
|
name = name.replace(":", ":\u200b"); // add a zero-width space to allow linewrapping after the colon
|
2016-07-21 16:33:54 +03:00
|
|
|
|
2016-07-20 14:47:32 +03:00
|
|
|
|
2018-11-05 19:42:25 +03:00
|
|
|
let badge;
|
|
|
|
if (badges) {
|
2017-10-11 19:56:17 +03:00
|
|
|
const limitedCount = FormattingUtils.formatCount(notificationCount);
|
2018-11-05 19:42:25 +03:00
|
|
|
const badgeContent = notificationCount ? limitedCount : '!';
|
|
|
|
badge = <div className={badgeClasses}>{ badgeContent }</div>;
|
2016-07-20 14:47:32 +03:00
|
|
|
}
|
|
|
|
|
2017-10-11 19:56:17 +03:00
|
|
|
let label;
|
2018-12-12 07:40:11 +03:00
|
|
|
let subtextLabel;
|
2017-10-11 19:56:17 +03:00
|
|
|
let tooltip;
|
2015-11-27 14:50:33 +03:00
|
|
|
if (!this.props.collapsed) {
|
2017-10-11 19:56:17 +03:00
|
|
|
const nameClasses = classNames({
|
2016-07-28 19:24:58 +03:00
|
|
|
'mx_RoomTile_name': true,
|
|
|
|
'mx_RoomTile_invite': this.props.isInvite,
|
2019-12-10 03:08:45 +03:00
|
|
|
'mx_RoomTile_badgeShown': badges || this.state.badgeHover || isMenuDisplayed,
|
2016-07-28 19:24:58 +03:00
|
|
|
});
|
|
|
|
|
2018-12-12 07:40:11 +03:00
|
|
|
subtextLabel = subtext ? <span className="mx_RoomTile_subtext">{ subtext }</span> : null;
|
2020-01-22 13:36:20 +03:00
|
|
|
// XXX: this is a workaround for Firefox giving this div a tabstop :( [tabIndex]
|
|
|
|
label = <div title={name} className={nameClasses} tabIndex={-1} dir="auto">{ name }</div>;
|
2016-09-09 08:57:30 +03:00
|
|
|
} else if (this.state.hover) {
|
2019-02-01 02:36:19 +03:00
|
|
|
const Tooltip = sdk.getComponent("elements.Tooltip");
|
|
|
|
tooltip = <Tooltip className="mx_RoomTile_tooltip" label={this.props.room.name} dir="auto" />;
|
2015-11-27 14:50:33 +03:00
|
|
|
}
|
|
|
|
|
2016-09-15 16:39:34 +03:00
|
|
|
//var incomingCallBox;
|
|
|
|
//if (this.props.incomingCall) {
|
|
|
|
// var IncomingCallBox = sdk.getComponent("voip.IncomingCallBox");
|
|
|
|
// incomingCallBox = <IncomingCallBox incomingCall={ this.props.incomingCall }/>;
|
|
|
|
//}
|
2015-12-17 05:49:09 +03:00
|
|
|
|
2019-11-11 20:53:17 +03:00
|
|
|
const AccessibleButton = sdk.getComponent('elements.AccessibleButton');
|
|
|
|
|
2018-11-05 19:42:25 +03:00
|
|
|
let contextMenuButton;
|
|
|
|
if (!MatrixClientPeg.get().isGuest()) {
|
2019-11-11 20:53:17 +03:00
|
|
|
contextMenuButton = (
|
2019-11-28 21:16:59 +03:00
|
|
|
<ContextMenuButton
|
|
|
|
className="mx_RoomTile_menuButton"
|
|
|
|
label={_t("Options")}
|
2019-12-10 03:08:45 +03:00
|
|
|
isExpanded={isMenuDisplayed}
|
|
|
|
onClick={this.onContextMenuButtonClick} />
|
2019-11-11 20:53:17 +03:00
|
|
|
);
|
2018-11-05 19:42:25 +03:00
|
|
|
}
|
|
|
|
|
2017-10-11 19:56:17 +03:00
|
|
|
const RoomAvatar = sdk.getComponent('avatars.RoomAvatar');
|
2015-11-27 14:50:33 +03:00
|
|
|
|
2019-09-30 18:04:43 +03:00
|
|
|
let ariaLabel = name;
|
|
|
|
|
2018-06-16 11:10:13 +03:00
|
|
|
let dmIndicator;
|
2019-12-19 17:29:46 +03:00
|
|
|
let dmOnline;
|
2020-01-27 15:17:12 +03:00
|
|
|
/* Post-cross-signing we don't show DM indicators at all, instead relying on user
|
|
|
|
context to let them know when that is. */
|
|
|
|
if (dmUserId && !SettingsStore.isFeatureEnabled("feature_cross_signing")) {
|
2019-01-11 04:37:28 +03:00
|
|
|
dmIndicator = <img
|
|
|
|
src={require("../../../../res/img/icon_person.svg")}
|
|
|
|
className="mx_RoomTile_dm"
|
|
|
|
width="11"
|
|
|
|
height="13"
|
|
|
|
alt="dm"
|
|
|
|
/>;
|
2019-12-19 17:29:46 +03:00
|
|
|
|
2019-12-26 21:15:08 +03:00
|
|
|
const { room } = this.props;
|
2019-12-27 21:31:15 +03:00
|
|
|
const member = room.getMember(dmUserId);
|
2020-01-13 14:48:51 +03:00
|
|
|
if (
|
|
|
|
member && member.membership === "join" && room.getJoinedMemberCount() === 2 &&
|
|
|
|
SettingsStore.isFeatureEnabled("feature_presence_in_room_list")
|
|
|
|
) {
|
2019-12-26 21:10:52 +03:00
|
|
|
const UserOnlineDot = sdk.getComponent('rooms.UserOnlineDot');
|
|
|
|
dmOnline = <UserOnlineDot userId={dmUserId} />;
|
2019-12-19 17:29:46 +03:00
|
|
|
}
|
2016-09-26 20:02:14 +03:00
|
|
|
}
|
2016-09-09 16:36:51 +03:00
|
|
|
|
2019-10-03 11:35:39 +03:00
|
|
|
// The following labels are written in such a fashion to increase screen reader efficiency (speed).
|
2019-09-30 18:04:43 +03:00
|
|
|
if (notifBadges && mentionBadges && !isInvite) {
|
2019-10-03 11:35:39 +03:00
|
|
|
ariaLabel += " " + _t("%(count)s unread messages including mentions.", {
|
2019-09-30 18:04:43 +03:00
|
|
|
count: notificationCount,
|
|
|
|
});
|
|
|
|
} else if (notifBadges) {
|
2019-10-03 11:35:39 +03:00
|
|
|
ariaLabel += " " + _t("%(count)s unread messages.", { count: notificationCount });
|
2019-09-30 18:04:43 +03:00
|
|
|
} else if (mentionBadges && !isInvite) {
|
2019-10-03 11:35:39 +03:00
|
|
|
ariaLabel += " " + _t("Unread mentions.");
|
2019-10-29 19:34:56 +03:00
|
|
|
} else if (this.props.unread) {
|
|
|
|
ariaLabel += " " + _t("Unread messages.");
|
2019-09-30 18:04:43 +03:00
|
|
|
}
|
|
|
|
|
2019-11-11 20:53:17 +03:00
|
|
|
let contextMenu;
|
2019-12-10 03:08:45 +03:00
|
|
|
if (isMenuDisplayed) {
|
2019-11-11 20:53:17 +03:00
|
|
|
const RoomTileContextMenu = sdk.getComponent('context_menus.RoomTileContextMenu');
|
|
|
|
contextMenu = (
|
2019-12-10 03:08:45 +03:00
|
|
|
<ContextMenu {...toRightOf(this.state.contextMenuPosition)} onFinished={this.closeMenu}>
|
2019-11-11 20:53:17 +03:00
|
|
|
<RoomTileContextMenu room={this.props.room} onFinished={this.closeMenu} />
|
|
|
|
</ContextMenu>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-01-13 20:53:49 +03:00
|
|
|
let privateIcon = null;
|
|
|
|
if (SettingsStore.isFeatureEnabled("feature_cross_signing")) {
|
2020-01-24 15:04:08 +03:00
|
|
|
if (this.state.joinRule == "invite" && !dmUserId) {
|
|
|
|
privateIcon = <InviteOnlyIcon />;
|
|
|
|
}
|
2020-01-13 20:53:49 +03:00
|
|
|
}
|
|
|
|
|
2020-01-21 20:17:40 +03:00
|
|
|
let e2eIcon = null;
|
2020-01-24 15:59:46 +03:00
|
|
|
if (this.state.e2eStatus) {
|
2020-01-23 14:09:52 +03:00
|
|
|
e2eIcon = <E2EIcon status={this.state.e2eStatus} className="mx_RoomTile_e2eIcon" />;
|
2020-01-21 20:17:40 +03:00
|
|
|
}
|
|
|
|
|
2019-11-11 20:53:17 +03:00
|
|
|
return <React.Fragment>
|
2020-01-15 14:37:14 +03:00
|
|
|
<RovingTabIndexWrapper>
|
|
|
|
{({onFocus, isActive, ref}) =>
|
|
|
|
<AccessibleButton
|
|
|
|
onFocus={onFocus}
|
|
|
|
tabIndex={isActive ? 0 : -1}
|
|
|
|
inputRef={ref}
|
|
|
|
className={classes}
|
|
|
|
onClick={this.onClick}
|
|
|
|
onMouseEnter={this.onMouseEnter}
|
|
|
|
onMouseLeave={this.onMouseLeave}
|
|
|
|
onContextMenu={this.onContextMenu}
|
|
|
|
aria-label={ariaLabel}
|
|
|
|
aria-selected={this.state.selected}
|
|
|
|
role="treeitem"
|
|
|
|
>
|
|
|
|
<div className={avatarClasses}>
|
|
|
|
<div className="mx_RoomTile_avatar_container">
|
|
|
|
<RoomAvatar room={this.props.room} width={24} height={24} />
|
|
|
|
{ dmIndicator }
|
2020-01-21 20:17:40 +03:00
|
|
|
{ e2eIcon }
|
2020-01-15 14:37:14 +03:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
{ privateIcon }
|
|
|
|
<div className="mx_RoomTile_nameContainer">
|
|
|
|
<div className="mx_RoomTile_labelContainer">
|
|
|
|
{ label }
|
|
|
|
{ subtextLabel }
|
|
|
|
</div>
|
|
|
|
{ dmOnline }
|
|
|
|
{ contextMenuButton }
|
|
|
|
{ badge }
|
|
|
|
</div>
|
|
|
|
{ /* { incomingCallBox } */ }
|
|
|
|
{ tooltip }
|
|
|
|
</AccessibleButton>
|
|
|
|
}
|
|
|
|
</RovingTabIndexWrapper>
|
2019-11-11 20:53:17 +03:00
|
|
|
|
|
|
|
{ contextMenu }
|
|
|
|
</React.Fragment>;
|
2017-10-11 19:56:17 +03:00
|
|
|
},
|
2015-11-27 14:50:33 +03:00
|
|
|
});
|