Files
mattermost/web/react/components/sidebar.jsx

647 lines
23 KiB
React
Raw Normal View History

// Copyright (c) 2015 Mattermost, Inc. All Rights Reserved.
2015-06-14 23:53:32 -08:00
// See License.txt for license information.
import NewChannelFlow from './new_channel_flow.jsx';
import MoreDirectChannels from './more_direct_channels.jsx';
import SidebarHeader from './sidebar_header.jsx';
import UnreadChannelIndicator from './unread_channel_indicator.jsx';
import TutorialTip from './tutorial/tutorial_tip.jsx';
import ChannelStore from '../stores/channel_store.jsx';
import UserStore from '../stores/user_store.jsx';
import TeamStore from '../stores/team_store.jsx';
import PreferenceStore from '../stores/preference_store.jsx';
import * as AsyncClient from '../utils/async_client.jsx';
import * as Client from '../utils/client.jsx';
import * as Utils from '../utils/utils.jsx';
import Constants from '../utils/constants.jsx';
2015-10-30 11:35:16 -04:00
const Preferences = Constants.Preferences;
const TutorialSteps = Constants.TutorialSteps;
2015-10-14 17:04:13 +05:00
const Tooltip = ReactBootstrap.Tooltip;
const OverlayTrigger = ReactBootstrap.OverlayTrigger;
2015-06-14 23:53:32 -08:00
export default class Sidebar extends React.Component {
constructor(props) {
super(props);
2015-06-14 23:53:32 -08:00
this.badgesActive = false;
this.firstUnreadChannel = null;
this.lastUnreadChannel = null;
2015-06-14 23:53:32 -08:00
this.getStateFromStores = this.getStateFromStores.bind(this);
this.onChange = this.onChange.bind(this);
this.onScroll = this.onScroll.bind(this);
this.updateUnreadIndicators = this.updateUnreadIndicators.bind(this);
this.handleLeaveDirectChannel = this.handleLeaveDirectChannel.bind(this);
this.handleResize = this.handleResize.bind(this);
this.showMoreChannelsModal = this.showMoreChannelsModal.bind(this);
this.showNewChannelModal = this.showNewChannelModal.bind(this);
this.hideNewChannelModal = this.hideNewChannelModal.bind(this);
this.showMoreDirectChannelsModal = this.showMoreDirectChannelsModal.bind(this);
this.hideMoreDirectChannelsModal = this.hideMoreDirectChannelsModal.bind(this);
this.createChannelElement = this.createChannelElement.bind(this);
2015-10-25 15:12:36 +01:00
this.updateTitle = this.updateTitle.bind(this);
2015-06-14 23:53:32 -08:00
this.isLeaving = new Map();
const state = this.getStateFromStores();
state.newChannelModalType = '';
2015-10-16 10:15:52 -04:00
state.showDirectChannelsModal = false;
state.loadingDMChannel = -1;
state.windowWidth = Utils.windowWidth();
this.state = state;
2015-10-25 15:12:36 +01:00
}
getTotalUnreadCount() {
2015-10-25 15:12:36 +01:00
let msgs = 0;
let mentions = 0;
const unreadCounts = this.state.unreadCounts;
2015-10-25 15:12:36 +01:00
Object.keys(unreadCounts).forEach((chId) => {
msgs += unreadCounts[chId].msgs;
mentions += unreadCounts[chId].mentions;
2015-10-25 15:12:36 +01:00
});
return {msgs, mentions};
}
getStateFromStores() {
const members = ChannelStore.getAllMembers();
const currentChannelId = ChannelStore.getCurrentId();
const currentUserId = UserStore.getCurrentId();
const channels = Object.assign([], ChannelStore.getAll());
channels.sort((a, b) => a.display_name.localeCompare(b.display_name));
const publicChannels = channels.filter((channel) => channel.type === Constants.OPEN_CHANNEL);
const privateChannels = channels.filter((channel) => channel.type === Constants.PRIVATE_CHANNEL);
2015-06-14 23:53:32 -08:00
const preferences = PreferenceStore.getCategory(Constants.Preferences.CATEGORY_DIRECT_CHANNEL_SHOW);
const directChannels = [];
for (const preference of preferences) {
if (preference.value !== 'true') {
2015-11-03 15:11:16 -05:00
continue;
}
2015-06-14 23:53:32 -08:00
const teammateId = preference.name;
let directChannel = channels.find(Utils.isDirectChannelForUser.bind(null, teammateId));
// a direct channel doesn't exist yet so create a fake one
if (!directChannel) {
directChannel = {
name: Utils.getDirectChannelName(currentUserId, teammateId),
last_post_at: 0,
total_msg_count: 0,
type: Constants.DM_CHANNEL,
fake: true
};
2015-06-14 23:53:32 -08:00
}
directChannel.display_name = Utils.displayUsername(teammateId);
directChannel.teammate_id = teammateId;
directChannel.status = UserStore.getStatus(teammateId);
directChannels.push(directChannel);
2015-06-14 23:53:32 -08:00
}
directChannels.sort(this.sortChannelsByDisplayName);
const hiddenDirectChannelCount = UserStore.getActiveOnlyProfileList(true).length - directChannels.length;
2015-06-14 23:53:32 -08:00
const tutorialStep = PreferenceStore.getInt(Preferences.TUTORIAL_STEP, UserStore.getCurrentId(), 999);
2015-10-30 11:35:16 -04:00
return {
activeId: currentChannelId,
2015-10-07 13:07:59 -04:00
members,
publicChannels,
privateChannels,
directChannels,
2015-10-30 11:35:16 -04:00
hiddenDirectChannelCount,
unreadCounts: JSON.parse(JSON.stringify(ChannelStore.getUnreadCounts())),
showTutorialTip: tutorialStep === TutorialSteps.CHANNEL_POPOVER
};
}
componentDidMount() {
ChannelStore.addChangeListener(this.onChange);
UserStore.addChangeListener(this.onChange);
UserStore.addStatusesChangeListener(this.onChange);
TeamStore.addChangeListener(this.onChange);
PreferenceStore.addChangeListener(this.onChange);
2015-06-14 23:53:32 -08:00
this.updateTitle();
this.updateUnreadIndicators();
window.addEventListener('resize', this.handleResize);
2015-11-30 22:27:45 +05:00
if ($(window).width() > 768) {
$('.nav-pills__container').perfectScrollbar();
}
}
shouldComponentUpdate(nextProps, nextState) {
if (!Utils.areObjectsEqual(nextState, this.state)) {
return true;
}
return false;
}
componentDidUpdate() {
2015-06-14 23:53:32 -08:00
this.updateTitle();
this.updateUnreadIndicators();
}
componentWillUnmount() {
window.removeEventListener('resize', this.handleResize);
ChannelStore.removeChangeListener(this.onChange);
UserStore.removeChangeListener(this.onChange);
UserStore.removeStatusesChangeListener(this.onChange);
TeamStore.removeChangeListener(this.onChange);
PreferenceStore.removeChangeListener(this.onChange);
}
handleResize() {
this.setState({
windowWidth: Utils.windowWidth(),
windowHeight: Utils.windowHeight()
});
}
onChange() {
this.setState(this.getStateFromStores());
}
updateTitle() {
const channel = ChannelStore.getCurrent();
2015-06-14 23:53:32 -08:00
if (channel) {
let currentSiteName = '';
2015-10-16 09:10:54 -07:00
if (global.window.mm_config.SiteName != null) {
currentSiteName = global.window.mm_config.SiteName;
}
let currentChannelName = channel.display_name;
2015-06-14 23:53:32 -08:00
if (channel.type === 'D') {
currentChannelName = Utils.getDirectTeammate(channel.id).username;
2015-06-14 23:53:32 -08:00
}
const unread = this.getTotalUnreadCount();
2015-10-25 15:12:36 +01:00
const mentionTitle = unread.mentions > 0 ? '(' + unread.mentions + ') ' : '';
const unreadTitle = unread.msgs > 0 ? '* ' : '';
2015-10-27 15:48:51 -04:00
document.title = mentionTitle + unreadTitle + currentChannelName + ' - ' + TeamStore.getCurrent().display_name + ' ' + currentSiteName;
2015-06-14 23:53:32 -08:00
}
}
onScroll() {
this.updateUnreadIndicators();
}
updateUnreadIndicators() {
2015-10-15 12:07:06 -04:00
const container = $(ReactDOM.findDOMNode(this.refs.container));
var showTopUnread = false;
var showBottomUnread = false;
if (this.firstUnreadChannel) {
2015-10-15 12:07:06 -04:00
var firstUnreadElement = $(ReactDOM.findDOMNode(this.refs[this.firstUnreadChannel]));
if (firstUnreadElement.position().top + firstUnreadElement.height() < 0) {
showTopUnread = true;
}
}
if (this.lastUnreadChannel) {
2015-10-15 12:07:06 -04:00
var lastUnreadElement = $(ReactDOM.findDOMNode(this.refs[this.lastUnreadChannel]));
if (lastUnreadElement.position().top > container.height()) {
showBottomUnread = true;
}
}
this.setState({
showTopUnread,
showBottomUnread
});
}
handleLeaveDirectChannel(channel) {
if (!this.isLeaving.get(channel.id)) {
this.isLeaving.set(channel.id, true);
2015-10-13 11:52:17 -04:00
const preference = PreferenceStore.setPreference(Constants.Preferences.CATEGORY_DIRECT_CHANNEL_SHOW, channel.teammate_id, 'false');
2015-10-13 15:18:01 -04:00
// bypass AsyncClient since we've already saved the updated preferences
Client.savePreferences(
[preference],
() => {
this.isLeaving.set(channel.id, false);
},
() => {
this.isLeaving.set(channel.id, false);
}
);
this.setState(this.getStateFromStores());
}
if (channel.id === this.state.activeId) {
Utils.switchChannel(ChannelStore.getByName(Constants.DEFAULT_CHANNEL));
}
}
sortChannelsByDisplayName(a, b) {
return a.display_name.localeCompare(b.display_name);
}
showMoreChannelsModal() {
// manually show the modal because using data-toggle messes with keyboard focus when the modal is dismissed
$('#more_channels').modal({'data-channeltype': 'O'}).modal('show');
}
showNewChannelModal(type) {
this.setState({newChannelModalType: type});
}
hideNewChannelModal() {
this.setState({newChannelModalType: ''});
}
showMoreDirectChannelsModal() {
this.setState({showDirectChannelsModal: true});
}
hideMoreDirectChannelsModal() {
this.setState({showDirectChannelsModal: false});
}
2015-10-30 11:35:16 -04:00
createTutorialTip() {
const screens = [];
screens.push(
<div>
2015-11-02 11:40:53 +05:00
<h4>{'Channels'}</h4>
<p><strong>{'Channels'}</strong>{' organize conversations across different topics. Theyre open to everyone on your team. To send private communications use '}<strong>{'Direct Messages'}</strong>{' for a single person or '}<strong>{'Private Groups'}</strong>{' for multiple people.'}
</p>
2015-10-30 11:35:16 -04:00
</div>
);
screens.push(
<div>
2015-11-02 11:40:53 +05:00
<h4>{'"Town Square" and "Off-Topic" channels'}</h4>
<p>{'Here are two public channels to start:'}</p>
<p>
<strong>{'Town Square'}</strong>{' is a place for team-wide communication. Everyone in your team is a member of this channel.'}
</p>
<p>
<strong>{'Off-Topic'}</strong>{' is a place for fun and humor outside of work-related channels. You and your team can decide what other channels to create.'}
</p>
2015-10-30 11:35:16 -04:00
</div>
);
screens.push(
<div>
2015-11-02 11:40:53 +05:00
<h4>{'Creating and Joining Channels'}</h4>
<p>
{'Click '}<strong>{'"More..."'}</strong>{' to create a new channel or join an existing one.'}
</p>
<p>
{'You can also create a new channel or private group by clicking the '}<strong>{'"+" symbol'}</strong>{' next to the channel or private group header.'}
</p>
2015-10-30 11:35:16 -04:00
</div>
);
return (
<TutorialTip
placement='right'
screens={screens}
overlayClass='tip-overlay--sidebar'
2015-10-30 11:35:16 -04:00
/>
);
}
createChannelElement(channel, index, arr, handleClose) {
const members = this.state.members;
const activeId = this.state.activeId;
const channelMember = members[channel.id];
const unreadCount = this.state.unreadCounts[channel.id] || {msgs: 0, mentions: 0};
let msgCount;
2015-06-14 23:53:32 -08:00
let linkClass = '';
if (channel.id === activeId) {
linkClass = 'active';
}
2015-06-14 23:53:32 -08:00
let rowClass = 'sidebar-channel';
var unread = false;
if (channelMember) {
2015-10-25 15:12:36 +01:00
msgCount = unreadCount.msgs + unreadCount.mentions;
unread = msgCount > 0 || channelMember.mention_count > 0;
}
2015-06-14 23:53:32 -08:00
if (unread) {
rowClass += ' unread-title';
if (channel.id !== activeId) {
if (!this.firstUnreadChannel) {
this.firstUnreadChannel = channel.name;
}
this.lastUnreadChannel = channel.name;
2015-06-14 23:53:32 -08:00
}
}
2015-06-14 23:53:32 -08:00
var badge = null;
if (channelMember) {
2015-10-25 15:12:36 +01:00
if (unreadCount.mentions) {
badge = <span className='badge pull-right small'>{unreadCount.mentions}</span>;
this.badgesActive = true;
2015-06-14 23:53:32 -08:00
}
} else if (this.state.loadingDMChannel === index && channel.type === 'D') {
badge = (
<img
className='channel-loading-gif pull-right'
src='/static/images/load.gif'
/>
);
}
2015-06-14 23:53:32 -08:00
2015-09-16 20:32:10 +05:00
if (msgCount > 0) {
rowClass += ' has-badge';
2015-09-16 20:32:10 +05:00
}
// set up status icon for direct message channels
var status = null;
if (channel.type === 'D') {
var statusIcon = '';
if (channel.status === 'online') {
statusIcon = Constants.ONLINE_ICON_SVG;
} else if (channel.status === 'away') {
statusIcon = Constants.ONLINE_ICON_SVG;
} else {
statusIcon = Constants.OFFLINE_ICON_SVG;
}
status = (
<span
className='status'
dangerouslySetInnerHTML={{__html: statusIcon}}
/>
);
}
2015-12-15 21:12:19 +05:00
var icon = null;
if (channel.type === 'O') {
icon = <div className='status'><i className='fa fa-globe'></i></div>;
} else if (channel.type === 'P') {
icon = <div className='status'><i className='fa fa-lock'></i></div>;
}
// set up click handler to switch channels (or create a new channel for non-existant ones)
var handleClick = null;
var href = '#';
var teamURL = TeamStore.getCurrentTeamUrl();
if (!channel.fake) {
handleClick = function clickHandler(e) {
2015-10-07 13:07:59 -04:00
if (e.target.attributes.getNamedItem('data-close')) {
handleClose(channel);
2015-10-07 13:07:59 -04:00
} else {
Utils.switchChannel(channel);
}
e.preventDefault();
};
} else if (channel.fake && teamURL) {
// It's a direct message channel that doesn't exist yet so let's create it now
var otherUserId = Utils.getUserIdFromChannelName(channel);
if (this.state.loadingDMChannel === -1) {
2015-08-14 17:39:50 -07:00
handleClick = function clickHandler(e) {
e.preventDefault();
2015-10-07 13:07:59 -04:00
if (e.target.attributes.getNamedItem('data-close')) {
handleClose(channel);
} else {
this.setState({loadingDMChannel: index});
Client.createDirectChannel(channel, otherUserId,
2015-10-07 13:07:59 -04:00
(data) => {
this.setState({loadingDMChannel: -1});
AsyncClient.getChannel(data.id);
Utils.switchChannel(data);
2015-10-07 13:07:59 -04:00
},
() => {
this.setState({loadingDMChannel: -1});
window.location.href = TeamStore.getCurrentTeamUrl() + '/channels/' + channel.name;
2015-10-07 13:07:59 -04:00
}
);
}
2015-09-02 16:27:55 -04:00
}.bind(this);
2015-06-14 23:53:32 -08:00
}
2015-08-14 17:39:50 -07:00
}
2015-06-14 23:53:32 -08:00
let closeButton = null;
2015-10-16 10:15:52 -04:00
const removeTooltip = (
<Tooltip id='remove-dm-tooltip'>{'Remove from list'}</Tooltip>
);
if (handleClose && !badge) {
closeButton = (
2015-10-14 17:04:13 +05:00
<OverlayTrigger
2015-10-16 10:15:52 -04:00
delayShow={1000}
2015-10-14 17:04:13 +05:00
placement='top'
overlay={removeTooltip}
>
<span
2015-10-14 17:04:13 +05:00
className='btn-close'
data-close='true'
>
{'×'}
</span>
2015-10-14 17:04:13 +05:00
</OverlayTrigger>
);
rowClass += ' has-close';
}
2015-10-30 11:35:16 -04:00
let tutorialTip = null;
if (this.state.showTutorialTip && channel.name === Constants.DEFAULT_CHANNEL) {
tutorialTip = this.createTutorialTip();
}
return (
<li
key={channel.name}
ref={channel.name}
className={linkClass}
>
<a
className={rowClass}
href={href}
onClick={handleClick}
>
2015-12-15 21:12:19 +05:00
{icon}
{status}
{channel.display_name}
{badge}
{closeButton}
</a>
2015-10-30 11:35:16 -04:00
{tutorialTip}
</li>
);
}
render() {
this.badgesActive = false;
// keep track of the first and last unread channels so we can use them to set the unread indicators
this.firstUnreadChannel = null;
this.lastUnreadChannel = null;
// create elements for all 3 types of channels
const publicChannelItems = this.state.publicChannels.map(this.createChannelElement);
2015-06-14 23:53:32 -08:00
const privateChannelItems = this.state.privateChannels.map(this.createChannelElement);
const directMessageItems = this.state.directChannels.map((channel, index, arr) => {
return this.createChannelElement(channel, index, arr, this.handleLeaveDirectChannel);
});
2015-06-14 23:53:32 -08:00
// update the favicon to show if there are any notifications
2015-06-14 23:53:32 -08:00
var link = document.createElement('link');
link.type = 'image/x-icon';
link.rel = 'shortcut icon';
link.id = 'favicon';
if (this.badgesActive) {
2015-06-14 23:53:32 -08:00
link.href = '/static/images/redfavicon.ico';
} else {
link.href = '/static/images/favicon.ico';
}
var head = document.getElementsByTagName('head')[0];
var oldLink = document.getElementById('favicon');
if (oldLink) {
head.removeChild(oldLink);
}
head.appendChild(link);
var directMessageMore = null;
if (this.state.hiddenDirectChannelCount > 0) {
directMessageMore = (
<li key='more'>
<a
href='#'
onClick={this.showMoreDirectChannelsModal}
>
{'More (' + this.state.hiddenDirectChannelCount + ')'}
</a>
</li>
);
}
let showChannelModal = false;
if (this.state.newChannelModalType !== '') {
showChannelModal = true;
}
2015-10-16 10:15:52 -04:00
const createChannelTootlip = (
<Tooltip id='new-channel-tooltip' >{'Create new channel'}</Tooltip>
);
const createGroupTootlip = (
<Tooltip id='new-group-tooltip'>{'Create new group'}</Tooltip>
);
2015-10-14 17:04:13 +05:00
2015-06-14 23:53:32 -08:00
return (
<div>
<NewChannelFlow
show={showChannelModal}
channelType={this.state.newChannelModalType}
onModalDismissed={this.hideNewChannelModal}
/>
<MoreDirectChannels
show={this.state.showDirectChannelsModal}
onModalDismissed={this.hideMoreDirectChannelsModal}
/>
<SidebarHeader
2015-10-27 15:48:51 -04:00
teamDisplayName={TeamStore.getCurrent().display_name}
teamName={TeamStore.getCurrent().name}
teamType={TeamStore.getCurrent().type}
/>
2015-06-14 23:53:32 -08:00
<UnreadChannelIndicator
show={this.state.showTopUnread}
extraClass='nav-pills__unread-indicator-top'
text={'Unread post(s) above'}
/>
<UnreadChannelIndicator
show={this.state.showBottomUnread}
extraClass='nav-pills__unread-indicator-bottom'
text={'Unread post(s) below'}
/>
<div
ref='container'
className='nav-pills__container'
onScroll={this.onScroll}
>
<ul className='nav nav-pills nav-stacked'>
<li>
<h4>
2015-10-07 13:07:59 -04:00
{'Channels'}
2015-10-14 17:04:13 +05:00
<OverlayTrigger
2015-10-16 10:15:52 -04:00
delayShow={500}
2015-10-14 17:04:13 +05:00
placement='top'
overlay={createChannelTootlip}
>
<a
className='add-channel-btn'
href='#'
onClick={this.showNewChannelModal.bind(this, 'O')}
>
{'+'}
</a>
2015-10-14 17:04:13 +05:00
</OverlayTrigger>
</h4>
</li>
{publicChannelItems}
<li>
<a
href='#'
className='nav-more'
onClick={this.showMoreChannelsModal}
>
2015-10-07 13:07:59 -04:00
{'More...'}
</a>
</li>
2015-06-14 23:53:32 -08:00
</ul>
<ul className='nav nav-pills nav-stacked'>
<li>
<h4>
2015-10-07 13:07:59 -04:00
{'Private Groups'}
2015-10-14 17:04:13 +05:00
<OverlayTrigger
2015-10-16 10:15:52 -04:00
delayShow={500}
2015-10-14 17:04:13 +05:00
placement='top'
overlay={createGroupTootlip}
>
<a
className='add-channel-btn'
href='#'
onClick={this.showNewChannelModal.bind(this, 'P')}
>
{'+'}
</a>
2015-10-14 17:04:13 +05:00
</OverlayTrigger>
</h4>
</li>
2015-06-14 23:53:32 -08:00
{privateChannelItems}
</ul>
<ul className='nav nav-pills nav-stacked'>
2015-10-07 13:07:59 -04:00
<li><h4>{'Direct Messages'}</h4></li>
2015-06-14 23:53:32 -08:00
{directMessageItems}
{directMessageMore}
2015-06-14 23:53:32 -08:00
</ul>
</div>
</div>
);
}
}
Sidebar.defaultProps = {
};
Sidebar.propTypes = {
};