2016-02-08 21:04:54 +03:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
|
2017-05-23 17:16:31 +03:00
|
|
|
import React from 'react';
|
2017-06-07 13:40:46 +03:00
|
|
|
import { _t, _tJsx } from '../../languageHandler';
|
2017-05-23 17:16:31 +03:00
|
|
|
import sdk from '../../index';
|
|
|
|
import WhoIsTyping from '../../WhoIsTyping';
|
|
|
|
import MatrixClientPeg from '../../MatrixClientPeg';
|
|
|
|
import MemberAvatar from '../views/avatars/MemberAvatar';
|
2017-01-20 19:51:35 +03:00
|
|
|
|
2017-01-23 18:01:39 +03:00
|
|
|
const HIDE_DEBOUNCE_MS = 10000;
|
|
|
|
const STATUS_BAR_HIDDEN = 0;
|
|
|
|
const STATUS_BAR_EXPANDED = 1;
|
|
|
|
const STATUS_BAR_EXPANDED_LARGE = 2;
|
|
|
|
|
2016-02-08 21:04:54 +03:00
|
|
|
module.exports = React.createClass({
|
|
|
|
displayName: 'RoomStatusBar',
|
|
|
|
|
|
|
|
propTypes: {
|
|
|
|
// the room this statusbar is representing.
|
|
|
|
room: React.PropTypes.object.isRequired,
|
2016-07-15 18:10:27 +03:00
|
|
|
|
2016-02-08 21:04:54 +03:00
|
|
|
// the number of messages which have arrived since we've been scrolled up
|
|
|
|
numUnreadMessages: React.PropTypes.number,
|
|
|
|
|
2017-02-03 03:40:40 +03:00
|
|
|
// string to display when there are messages in the room which had errors on send
|
|
|
|
unsentMessageError: React.PropTypes.string,
|
2016-02-08 21:04:54 +03:00
|
|
|
|
|
|
|
// this is true if we are fully scrolled-down, and are looking at
|
|
|
|
// the end of the live timeline.
|
|
|
|
atEndOfLiveTimeline: React.PropTypes.bool,
|
|
|
|
|
|
|
|
// true if there is an active call in this room (means we show
|
|
|
|
// the 'Active Call' text in the status bar if there is nothing
|
|
|
|
// more interesting)
|
|
|
|
hasActiveCall: React.PropTypes.bool,
|
2016-02-09 14:36:57 +03:00
|
|
|
|
2017-01-24 19:01:39 +03:00
|
|
|
// Number of names to display in typing indication. E.g. set to 3, will
|
|
|
|
// result in "X, Y, Z and 100 others are typing."
|
|
|
|
whoIsTypingLimit: React.PropTypes.number,
|
|
|
|
|
2016-02-09 14:36:57 +03:00
|
|
|
// callback for when the user clicks on the 'resend all' button in the
|
|
|
|
// 'unsent messages' bar
|
|
|
|
onResendAllClick: React.PropTypes.func,
|
|
|
|
|
2016-03-21 19:49:07 +03:00
|
|
|
// callback for when the user clicks on the 'cancel all' button in the
|
|
|
|
// 'unsent messages' bar
|
|
|
|
onCancelAllClick: React.PropTypes.func,
|
|
|
|
|
2016-02-09 14:36:57 +03:00
|
|
|
// callback for when the user clicks on the 'scroll to bottom' button
|
|
|
|
onScrollToBottomClick: React.PropTypes.func,
|
2016-02-23 14:06:16 +03:00
|
|
|
|
|
|
|
// callback for when we do something that changes the size of the
|
|
|
|
// status bar. This is used to trigger a re-layout in the parent
|
|
|
|
// component.
|
|
|
|
onResize: React.PropTypes.func,
|
2017-01-23 18:01:39 +03:00
|
|
|
|
|
|
|
// callback for when the status bar can be hidden from view, as it is
|
|
|
|
// not displaying anything
|
|
|
|
onHidden: React.PropTypes.func,
|
2017-02-02 20:48:47 +03:00
|
|
|
|
2017-01-23 18:01:39 +03:00
|
|
|
// callback for when the status bar is displaying something and should
|
|
|
|
// be visible
|
|
|
|
onVisible: React.PropTypes.func,
|
2016-02-08 21:04:54 +03:00
|
|
|
},
|
|
|
|
|
2017-01-24 20:16:26 +03:00
|
|
|
getDefaultProps: function() {
|
|
|
|
return {
|
2017-02-09 13:30:06 +03:00
|
|
|
whoIsTypingLimit: 3,
|
2017-01-24 20:16:26 +03:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2016-02-08 21:04:54 +03:00
|
|
|
getInitialState: function() {
|
|
|
|
return {
|
|
|
|
syncState: MatrixClientPeg.get().getSyncState(),
|
2017-02-09 13:01:51 +03:00
|
|
|
usersTyping: WhoIsTyping.usersTypingApartFromMe(this.props.room),
|
2016-02-08 21:04:54 +03:00
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
componentWillMount: function() {
|
|
|
|
MatrixClientPeg.get().on("sync", this.onSyncStateChange);
|
2016-02-23 19:17:50 +03:00
|
|
|
MatrixClientPeg.get().on("RoomMember.typing", this.onRoomMemberTyping);
|
2016-02-08 21:04:54 +03:00
|
|
|
|
2017-03-28 11:30:41 +03:00
|
|
|
this._checkSize();
|
|
|
|
},
|
2017-01-23 18:01:39 +03:00
|
|
|
|
2017-03-28 11:30:41 +03:00
|
|
|
componentDidUpdate: function() {
|
|
|
|
this._checkSize();
|
2016-02-23 14:06:16 +03:00
|
|
|
},
|
|
|
|
|
2016-02-08 21:04:54 +03:00
|
|
|
componentWillUnmount: function() {
|
2016-02-16 00:50:39 +03:00
|
|
|
// we may have entirely lost our client as we're logging out before clicking login on the guest bar...
|
2016-02-23 19:17:50 +03:00
|
|
|
var client = MatrixClientPeg.get();
|
|
|
|
if (client) {
|
|
|
|
client.removeListener("sync", this.onSyncStateChange);
|
|
|
|
client.removeListener("RoomMember.typing", this.onRoomMemberTyping);
|
2016-02-16 00:50:39 +03:00
|
|
|
}
|
2016-02-08 21:04:54 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
onSyncStateChange: function(state, prevState) {
|
|
|
|
if (state === "SYNCING" && prevState === "SYNCING") {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
this.setState({
|
|
|
|
syncState: state
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2016-02-23 19:17:50 +03:00
|
|
|
onRoomMemberTyping: function(ev, member) {
|
|
|
|
this.setState({
|
2017-09-15 05:25:02 +03:00
|
|
|
usersTyping: WhoIsTyping.usersTypingApartFromMeAndIgnored(this.props.room),
|
2016-02-23 19:17:50 +03:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2017-03-28 11:30:41 +03:00
|
|
|
// Check whether current size is greater than 0, if yes call props.onVisible
|
|
|
|
_checkSize: function () {
|
|
|
|
if (this.props.onVisible && this._getSize()) {
|
|
|
|
this.props.onVisible();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-01-23 18:01:39 +03:00
|
|
|
// We don't need the actual height - just whether it is likely to have
|
|
|
|
// changed - so we use '0' to indicate normal size, and other values to
|
|
|
|
// indicate other sizes.
|
2017-03-28 11:30:41 +03:00
|
|
|
_getSize: function() {
|
|
|
|
if (this.state.syncState === "ERROR" ||
|
|
|
|
(this.state.usersTyping.length > 0) ||
|
|
|
|
this.props.numUnreadMessages ||
|
|
|
|
!this.props.atEndOfLiveTimeline ||
|
2017-07-13 12:16:59 +03:00
|
|
|
this.props.hasActiveCall
|
2017-02-22 17:03:30 +03:00
|
|
|
) {
|
2017-01-23 18:01:39 +03:00
|
|
|
return STATUS_BAR_EXPANDED;
|
2017-03-28 11:30:41 +03:00
|
|
|
} else if (this.props.unsentMessageError) {
|
2017-01-23 18:01:39 +03:00
|
|
|
return STATUS_BAR_EXPANDED_LARGE;
|
2016-02-23 14:06:16 +03:00
|
|
|
}
|
2017-01-23 18:01:39 +03:00
|
|
|
return STATUS_BAR_HIDDEN;
|
|
|
|
},
|
2016-02-23 14:06:16 +03:00
|
|
|
|
2016-02-09 17:42:32 +03:00
|
|
|
// return suitable content for the image on the left of the status bar.
|
|
|
|
//
|
|
|
|
// if wantPlaceholder is true, we include a "..." placeholder if
|
|
|
|
// there is nothing better to put in.
|
|
|
|
_getIndicator: function(wantPlaceholder) {
|
|
|
|
if (this.props.numUnreadMessages) {
|
|
|
|
return (
|
|
|
|
<div className="mx_RoomStatusBar_scrollDownIndicator"
|
|
|
|
onClick={ this.props.onScrollToBottomClick }>
|
|
|
|
<img src="img/newmessages.svg" width="24" height="24"
|
|
|
|
alt=""/>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!this.props.atEndOfLiveTimeline) {
|
|
|
|
return (
|
|
|
|
<div className="mx_RoomStatusBar_scrollDownIndicator"
|
|
|
|
onClick={ this.props.onScrollToBottomClick }>
|
|
|
|
<img src="img/scrolldown.svg" width="24" height="24"
|
2017-05-23 17:16:31 +03:00
|
|
|
alt={ _t("Scroll to bottom of page") }
|
|
|
|
title={ _t("Scroll to bottom of page") }/>
|
2016-02-09 17:42:32 +03:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.props.hasActiveCall) {
|
2017-02-20 02:43:55 +03:00
|
|
|
var TintableSvg = sdk.getComponent("elements.TintableSvg");
|
2016-02-09 17:42:32 +03:00
|
|
|
return (
|
2017-02-20 02:43:55 +03:00
|
|
|
<TintableSvg src="img/sound-indicator.svg" width="23" height="20"/>
|
2016-02-09 17:42:32 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-02-18 21:16:48 +03:00
|
|
|
if (this.state.syncState === "ERROR") {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2016-02-09 17:42:32 +03:00
|
|
|
if (wantPlaceholder) {
|
|
|
|
return (
|
2017-01-20 19:51:35 +03:00
|
|
|
<div className="mx_RoomStatusBar_typingIndicatorAvatars">
|
2017-01-24 19:04:37 +03:00
|
|
|
{this._renderTypingIndicatorAvatars(this.props.whoIsTypingLimit)}
|
2016-03-28 17:09:36 +03:00
|
|
|
</div>
|
2016-02-09 17:42:32 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
|
2017-01-20 19:51:35 +03:00
|
|
|
_renderTypingIndicatorAvatars: function(limit) {
|
2017-02-09 13:01:51 +03:00
|
|
|
let users = this.state.usersTyping;
|
2017-01-20 19:51:35 +03:00
|
|
|
|
2017-02-09 13:30:06 +03:00
|
|
|
let othersCount = 0;
|
|
|
|
if (users.length > limit) {
|
|
|
|
othersCount = users.length - limit + 1;
|
|
|
|
users = users.slice(0, limit - 1);
|
|
|
|
}
|
2017-01-20 19:51:35 +03:00
|
|
|
|
2017-02-15 19:29:08 +03:00
|
|
|
const avatars = users.map((u) => {
|
2017-01-20 19:51:35 +03:00
|
|
|
return (
|
|
|
|
<MemberAvatar
|
|
|
|
key={u.userId}
|
|
|
|
member={u}
|
|
|
|
width={24}
|
|
|
|
height={24}
|
|
|
|
resizeMethod="crop"
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
if (othersCount > 0) {
|
|
|
|
avatars.push(
|
2017-02-02 21:47:15 +03:00
|
|
|
<span className="mx_RoomStatusBar_typingIndicatorRemaining" key="others">
|
2017-01-20 19:51:35 +03:00
|
|
|
+{othersCount}
|
|
|
|
</span>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return avatars;
|
|
|
|
},
|
2016-02-09 17:42:32 +03:00
|
|
|
|
|
|
|
// return suitable content for the main (text) part of the status bar.
|
|
|
|
_getContent: function() {
|
2016-08-11 05:25:12 +03:00
|
|
|
const EmojiText = sdk.getComponent('elements.EmojiText');
|
2016-02-08 21:04:54 +03:00
|
|
|
|
|
|
|
// no conn bar trumps unread count since you can't get unread messages
|
|
|
|
// without a connection! (technically may already have some but meh)
|
|
|
|
// It also trumps the "some not sent" msg since you can't resend without
|
|
|
|
// a connection!
|
|
|
|
if (this.state.syncState === "ERROR") {
|
|
|
|
return (
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar">
|
2016-02-08 21:04:54 +03:00
|
|
|
<img src="img/warning.svg" width="24" height="23" title="/!\ " alt="/!\ "/>
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar_title">
|
2017-05-23 17:16:31 +03:00
|
|
|
{_t('Connectivity to the server has been lost.')}
|
2016-02-09 17:42:32 +03:00
|
|
|
</div>
|
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar_desc">
|
2017-05-23 17:16:31 +03:00
|
|
|
{_t('Sent messages will be stored until your connection has returned.')}
|
2016-02-08 21:04:54 +03:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-02-03 03:40:40 +03:00
|
|
|
if (this.props.unsentMessageError) {
|
2016-02-08 21:04:54 +03:00
|
|
|
return (
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar">
|
2016-02-08 21:04:54 +03:00
|
|
|
<img src="img/warning.svg" width="24" height="23" title="/!\ " alt="/!\ "/>
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar_title">
|
2017-02-03 03:40:40 +03:00
|
|
|
{ this.props.unsentMessageError }
|
2016-02-09 17:42:32 +03:00
|
|
|
</div>
|
|
|
|
<div className="mx_RoomStatusBar_connectionLostBar_desc">
|
2017-06-07 13:40:46 +03:00
|
|
|
{_tJsx("<a>Resend all</a> or <a>cancel all</a> now. You can also select individual messages to resend or cancel.",
|
|
|
|
[/<a>(.*?)<\/a>/, /<a>(.*?)<\/a>/],
|
|
|
|
[
|
|
|
|
(sub) => <a className="mx_RoomStatusBar_resend_link" key="resend" onClick={ this.props.onResendAllClick }>{sub}</a>,
|
|
|
|
(sub) => <a className="mx_RoomStatusBar_resend_link" key="cancel" onClick={ this.props.onCancelAllClick }>{sub}</a>,
|
|
|
|
]
|
|
|
|
)}
|
2016-02-08 21:04:54 +03:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
// unread count trumps who is typing since the unread count is only
|
|
|
|
// set when you've scrolled up
|
2016-02-09 15:40:11 +03:00
|
|
|
if (this.props.numUnreadMessages) {
|
2017-06-07 13:40:46 +03:00
|
|
|
// MUST use var name "count" for pluralization to kick in
|
|
|
|
var unreadMsgs = _t("%(count)s new messages", {count: this.props.numUnreadMessages});
|
2016-02-08 21:04:54 +03:00
|
|
|
|
|
|
|
return (
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_unreadMessagesBar"
|
|
|
|
onClick={ this.props.onScrollToBottomClick }>
|
2016-02-08 21:04:54 +03:00
|
|
|
{unreadMsgs}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-02-09 13:01:51 +03:00
|
|
|
const typingString = WhoIsTyping.whoIsTypingString(
|
|
|
|
this.state.usersTyping,
|
|
|
|
this.props.whoIsTypingLimit
|
|
|
|
);
|
2016-02-08 21:04:54 +03:00
|
|
|
if (typingString) {
|
|
|
|
return (
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_typingBar">
|
2016-08-09 22:01:51 +03:00
|
|
|
<EmojiText>{typingString}</EmojiText>
|
2016-02-08 21:04:54 +03:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.props.hasActiveCall) {
|
|
|
|
return (
|
2016-02-09 17:42:32 +03:00
|
|
|
<div className="mx_RoomStatusBar_callBar">
|
2017-05-23 17:16:31 +03:00
|
|
|
<b>{_t('Active call')}</b>
|
2016-02-08 21:04:54 +03:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-02-09 17:42:32 +03:00
|
|
|
return null;
|
2016-02-08 21:04:54 +03:00
|
|
|
},
|
2016-02-09 17:42:32 +03:00
|
|
|
|
|
|
|
|
|
|
|
render: function() {
|
|
|
|
var content = this._getContent();
|
2017-02-09 13:01:51 +03:00
|
|
|
var indicator = this._getIndicator(this.state.usersTyping.length > 0);
|
2016-02-09 17:42:32 +03:00
|
|
|
|
|
|
|
return (
|
|
|
|
<div className="mx_RoomStatusBar">
|
|
|
|
<div className="mx_RoomStatusBar_indicator">
|
|
|
|
{indicator}
|
|
|
|
</div>
|
|
|
|
{content}
|
|
|
|
</div>
|
|
|
|
);
|
2016-07-15 18:10:27 +03:00
|
|
|
},
|
2016-02-08 21:04:54 +03:00
|
|
|
});
|