Files
mattermost/webapp/components/sidebar.jsx

829 lines
30 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.
2016-03-14 08:50:46 -04:00
import $ from 'jquery';
import ReactDOM from 'react-dom';
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 StatusIcon from './status_icon.jsx';
2016-03-14 08:50:46 -04:00
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';
2016-03-14 08:50:46 -04:00
import * as AsyncClient from 'utils/async_client.jsx';
import * as Utils from 'utils/utils.jsx';
import * as ChannelActions from 'actions/channel_actions.jsx';
2016-03-14 08:50:46 -04:00
import Constants from 'utils/constants.jsx';
2016-03-14 08:50:46 -04:00
import {FormattedMessage, FormattedHTMLMessage} from 'react-intl';
2015-10-30 11:35:16 -04:00
const Preferences = Constants.Preferences;
const TutorialSteps = Constants.TutorialSteps;
2016-03-14 08:50:46 -04:00
import {Tooltip, OverlayTrigger} from 'react-bootstrap';
import loadingGif from 'images/load.gif';
import React from 'react';
2016-06-22 10:30:01 -04:00
import {browserHistory, Link} from 'react-router/es6';
2016-03-14 08:50:46 -04:00
import favicon from 'images/favicon/favicon-16x16.png';
import redFavicon from 'images/favicon/redfavicon-16x16.png';
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.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.navigateChannelShortcut = this.navigateChannelShortcut.bind(this);
this.navigateUnreadChannelShortcut = this.navigateUnreadChannelShortcut.bind(this);
this.getDisplayedChannels = this.getDisplayedChannels.bind(this);
this.updateScrollbarOnChannelChange = this.updateScrollbarOnChannelChange.bind(this);
this.isLeaving = new Map();
const state = this.getStateFromStores();
state.newChannelModalType = '';
2015-10-16 10:15:52 -04:00
state.showDirectChannelsModal = false;
state.loadingDMChannel = -1;
this.state = state;
2015-10-25 15:12:36 +01:00
}
2016-06-15 05:01:31 -07: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};
}
2016-06-15 05:01:31 -07:00
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 = [];
const directNonTeamChannels = [];
for (const [name, value] of preferences) {
if (value !== 'true') {
2015-11-03 15:11:16 -05:00
continue;
}
2015-06-14 23:53:32 -08:00
const teammateId = name;
let directChannel = channels.find(Utils.isDirectChannelForUser.bind(null, teammateId));
// a direct channel doesn't exist yet so create a fake one
if (directChannel == null) {
directChannel = {
name: Utils.getDirectChannelName(currentUserId, teammateId),
last_post_at: 0,
total_msg_count: 0,
type: Constants.DM_CHANNEL,
fake: true
};
} else {
directChannel = JSON.parse(JSON.stringify(directChannel));
2015-06-14 23:53:32 -08:00
}
directChannel.display_name = Utils.displayUsername(teammateId);
directChannel.teammate_id = teammateId;
directChannel.status = UserStore.getStatus(teammateId);
if (UserStore.hasTeamProfile(teammateId)) {
directChannels.push(directChannel);
} else {
directNonTeamChannels.push(directChannel);
}
2015-06-14 23:53:32 -08:00
}
directChannels.sort(this.sortChannelsByDisplayName);
directNonTeamChannels.sort(this.sortChannelsByDisplayName);
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,
directNonTeamChannels,
unreadCounts: JSON.parse(JSON.stringify(ChannelStore.getUnreadCounts())),
2016-02-08 07:26:10 -05:00
showTutorialTip: tutorialStep === TutorialSteps.CHANNEL_POPOVER,
currentTeam: TeamStore.getCurrent(),
currentUser: UserStore.getCurrentUser(),
townSquare: ChannelStore.getByName(Constants.DEFAULT_CHANNEL),
offTopic: ChannelStore.getByName(Constants.OFFTOPIC_CHANNEL)
};
}
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();
document.addEventListener('keydown', this.navigateChannelShortcut);
document.addEventListener('keydown', this.navigateUnreadChannelShortcut);
}
2016-06-15 05:01:31 -07:00
shouldComponentUpdate(nextProps, nextState) {
if (!Utils.areObjectsEqual(nextState, this.state)) {
return true;
}
return false;
}
2016-06-15 05:01:31 -07:00
componentDidUpdate(prevProps, prevState) {
2015-06-14 23:53:32 -08:00
this.updateTitle();
this.updateUnreadIndicators();
2016-03-23 04:06:08 +05:00
if (!Utils.isMobile()) {
$('.sidebar--left .nav-pills__container').perfectScrollbar();
}
2016-05-31 05:13:43 -07:00
// reset the scrollbar upon switching teams
if (this.state.currentTeam !== prevState.currentTeam) {
this.refs.container.scrollTop = 0;
$('.nav-pills__container').perfectScrollbar('update');
}
// close the LHS on mobile when you change channels
if (this.state.activeId !== prevState.activeId) {
$('.app__body .inner-wrap').removeClass('move--right');
$('.app__body .sidebar--left').removeClass('move--right');
}
}
2016-06-15 05:01:31 -07:00
componentWillUnmount() {
ChannelStore.removeChangeListener(this.onChange);
UserStore.removeChangeListener(this.onChange);
UserStore.removeStatusesChangeListener(this.onChange);
TeamStore.removeChangeListener(this.onChange);
PreferenceStore.removeChangeListener(this.onChange);
document.removeEventListener('keydown', this.navigateChannelShortcut);
document.removeEventListener('keydown', this.navigateUnreadChannelShortcut);
}
2016-06-15 05:01:31 -07:00
onChange() {
this.setState(this.getStateFromStores());
}
2016-06-15 05:01:31 -07:00
updateTitle() {
const channel = ChannelStore.getCurrent();
2016-02-08 07:26:10 -05:00
if (channel && this.state.currentTeam) {
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') {
2016-02-01 08:01:49 -05:00
const teammate = Utils.getDirectTeammate(channel.id);
if (teammate != null) {
currentChannelName = teammate.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 ? '* ' : '';
2016-02-08 07:26:10 -05:00
document.title = mentionTitle + unreadTitle + currentChannelName + ' - ' + this.state.currentTeam.display_name + ' ' + currentSiteName;
2015-06-14 23:53:32 -08:00
}
}
2016-06-15 05:01:31 -07:00
onScroll() {
this.updateUnreadIndicators();
}
2016-06-15 05:01:31 -07:00
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
});
}
2016-06-15 05:01:31 -07:00
updateScrollbarOnChannelChange(channel) {
const curChannel = this.refs[channel.name].getBoundingClientRect();
if ((curChannel.top - Constants.CHANNEL_SCROLL_ADJUSTMENT < 0) || (curChannel.top + curChannel.height > this.refs.container.getBoundingClientRect().height)) {
this.refs.container.scrollTop = this.refs.container.scrollTop + (curChannel.top - Constants.CHANNEL_SCROLL_ADJUSTMENT);
$('.nav-pills__container').perfectScrollbar('update');
}
}
2016-06-15 05:01:31 -07:00
navigateChannelShortcut(e) {
if (e.altKey && !e.shiftKey && (e.keyCode === Constants.KeyCodes.UP || e.keyCode === Constants.KeyCodes.DOWN)) {
e.preventDefault();
const allChannels = this.getDisplayedChannels();
const curChannelId = this.state.activeId;
let curIndex = -1;
for (let i = 0; i < allChannels.length; i++) {
if (allChannels[i].id === curChannelId) {
curIndex = i;
}
}
let nextChannel = allChannels[curIndex];
let nextIndex = curIndex;
if (e.keyCode === Constants.KeyCodes.DOWN) {
nextIndex = curIndex + 1;
} else if (e.keyCode === Constants.KeyCodes.UP) {
nextIndex = curIndex - 1;
}
nextChannel = allChannels[Utils.mod(nextIndex, allChannels.length)];
ChannelActions.goToChannel(nextChannel);
this.updateScrollbarOnChannelChange(nextChannel);
}
}
2016-06-15 05:01:31 -07:00
navigateUnreadChannelShortcut(e) {
if (e.altKey && e.shiftKey && (e.keyCode === Constants.KeyCodes.UP || e.keyCode === Constants.KeyCodes.DOWN)) {
e.preventDefault();
const allChannels = this.getDisplayedChannels();
const curChannelId = this.state.activeId;
let curIndex = -1;
for (let i = 0; i < allChannels.length; i++) {
if (allChannels[i].id === curChannelId) {
curIndex = i;
}
}
let nextChannel = allChannels[curIndex];
let nextIndex = curIndex;
let count = 0;
let increment = 0;
if (e.keyCode === Constants.KeyCodes.UP) {
increment = -1;
} else if (e.keyCode === Constants.KeyCodes.DOWN) {
increment = 1;
}
let unreadCounts = ChannelStore.getUnreadCount(allChannels[nextIndex].id);
while (count < allChannels.length && unreadCounts.msgs === 0 && unreadCounts.mentions === 0) {
nextIndex += increment;
count++;
nextIndex = Utils.mod(nextIndex, allChannels.length);
unreadCounts = ChannelStore.getUnreadCount(allChannels[nextIndex].id);
}
if (unreadCounts.msgs !== 0 || unreadCounts.mentions !== 0) {
nextChannel = allChannels[nextIndex];
ChannelActions.goToChannel(nextChannel);
this.updateScrollbarOnChannelChange(nextChannel);
}
}
}
2016-06-15 05:01:31 -07:00
getDisplayedChannels() {
return this.state.publicChannels.concat(this.state.privateChannels).concat(this.state.directChannels).concat(this.state.directNonTeamChannels);
}
2016-06-15 05:01:31 -07:00
handleLeaveDirectChannel(e, channel) {
e.preventDefault();
if (!this.isLeaving.get(channel.id)) {
this.isLeaving.set(channel.id, true);
AsyncClient.savePreference(
Constants.Preferences.CATEGORY_DIRECT_CHANNEL_SHOW,
channel.teammate_id,
'false',
() => {
this.isLeaving.set(channel.id, false);
},
() => {
this.isLeaving.set(channel.id, false);
}
);
this.setState(this.getStateFromStores());
}
if (channel.id === this.state.activeId) {
browserHistory.push('/' + this.state.currentTeam.name + '/channels/town-square');
}
}
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});
}
2016-06-15 05:01:31 -07:00
hideNewChannelModal() {
this.setState({newChannelModalType: ''});
}
showMoreDirectChannelsModal() {
this.setState({showDirectChannelsModal: true});
}
2016-06-15 05:01:31 -07:00
hideMoreDirectChannelsModal() {
this.setState({showDirectChannelsModal: false});
}
openLeftSidebar() {
if (Utils.isMobile()) {
setTimeout(() => {
document.querySelector('.app__body .inner-wrap').classList.add('move--right');
document.querySelector('.app__body .sidebar--left').classList.add('move--right');
});
}
}
2015-10-30 11:35:16 -04:00
createTutorialTip() {
const screens = [];
let townSquareDisplayName = Constants.DEFAULT_CHANNEL_UI_NAME;
if (this.state.townSquare) {
townSquareDisplayName = this.state.townSquare.display_name;
}
let offTopicDisplayName = Constants.OFFTOPIC_CHANNEL_UI_NAME;
if (this.state.offTopic) {
offTopicDisplayName = this.state.offTopic.display_name;
}
2015-10-30 11:35:16 -04:00
screens.push(
<div>
<FormattedHTMLMessage
id='sidebar.tutorialScreen1'
defaultMessage='<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>
<FormattedHTMLMessage
id='sidebar.tutorialScreen2'
defaultMessage='<h4>"{townsquare}" and "{offtopic}" channels</h4>
<p>Here are two public channels to start:</p>
<p><strong>{townsquare}</strong> is a place for team-wide communication. Everyone in your team is a member of this channel.</p>
<p><strong>{offtopic}</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>'
values={{
townsquare: townSquareDisplayName,
offtopic: offTopicDisplayName
}}
/>
2015-10-30 11:35:16 -04:00
</div>
);
screens.push(
<div>
<FormattedHTMLMessage
id='sidebar.tutorialScreen3'
defaultMessage='<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'
2016-03-14 08:50:46 -04:00
src={loadingGif}
/>
);
}
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
}
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>;
} else {
// set up status icon for direct message channels (status is null for other channel types)
icon = <StatusIcon status={channel.status}/>;
2015-12-15 21:12:19 +05:00
}
let closeButton = null;
2015-10-16 10:15:52 -04:00
const removeTooltip = (
<Tooltip id='remove-dm-tooltip'>
<FormattedMessage
id='sidebar.removeList'
defaultMessage='Remove from list'
/>
</Tooltip>
2015-10-16 10:15:52 -04:00
);
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
onClick={(e) => handleClose(e, channel)}
className='btn-close'
>
{'×'}
</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();
this.openLeftSidebar();
2015-10-30 11:35:16 -04:00
}
let link = '';
if (channel.fake) {
2016-03-29 16:51:27 -04:00
link = '/' + this.state.currentTeam.name + '/channels/' + channel.name + '?fakechannel=' + encodeURIComponent(JSON.stringify(channel));
} else {
2016-03-29 16:51:27 -04:00
link = '/' + this.state.currentTeam.name + '/channels/' + channel.name;
}
return (
<li
key={channel.name}
ref={channel.name}
className={linkClass}
>
<Link
to={link}
className={rowClass}
>
2015-12-15 21:12:19 +05:00
{icon}
{channel.display_name}
{badge}
{closeButton}
</Link>
2015-10-30 11:35:16 -04:00
{tutorialTip}
</li>
);
}
render() {
2016-02-08 07:26:10 -05:00
// Check if we have all info needed to render
if (this.state.currentTeam == null || this.state.currentUser == null) {
return (<div/>);
}
2016-05-04 10:10:19 -04:00
this.lastBadgesActive = this.badgesActive;
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
const directMessageNonTeamItems = this.state.directNonTeamChannels.map((channel, index, arr) => {
return this.createChannelElement(channel, index, arr, this.handleLeaveDirectChannel);
});
let directDivider;
if (directMessageNonTeamItems.length !== 0) {
directDivider =
(<div className='sidebar__divider'>
<div className='sidebar__divider__text'>
<FormattedMessage
id='sidebar.otherMembers'
2016-05-12 09:31:08 -03:00
defaultMessage='Outside this team'
/>
</div>
</div>);
}
// update the favicon to show if there are any notifications
2016-05-04 10:10:19 -04:00
if (this.lastBadgesActive !== this.badgesActive) {
var link = document.createElement('link');
link.type = 'image/x-icon';
link.rel = 'shortcut icon';
link.id = 'favicon';
if (this.badgesActive) {
link.href = redFavicon;
} else {
link.href = favicon;
}
var head = document.getElementsByTagName('head')[0];
var oldLink = document.getElementById('favicon');
if (oldLink) {
head.removeChild(oldLink);
}
head.appendChild(link);
2015-06-14 23:53:32 -08:00
}
var directMessageMore = (
<li key='more'>
<a
href='#'
onClick={this.showMoreDirectChannelsModal}
>
<FormattedMessage
id='sidebar.moreElips'
defaultMessage='More...'
/>
</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' >
<FormattedMessage
id='sidebar.createChannel'
defaultMessage='Create new channel'
/>
</Tooltip>
2015-10-16 10:15:52 -04:00
);
const createGroupTootlip = (
<Tooltip id='new-group-tooltip'>
<FormattedMessage
id='sidebar.createGroup'
defaultMessage='Create new group'
/>
</Tooltip>
);
const above = (
<FormattedMessage
id='sidebar.unreadAbove'
defaultMessage='Unread post(s) above'
/>
);
const below = (
<FormattedMessage
id='sidebar.unreadBelow'
defaultMessage='Unread post(s) below'
/>
2015-10-16 10:15:52 -04:00
);
2015-10-14 17:04:13 +05:00
const isAdmin = TeamStore.isTeamAdminForCurrentTeam() || UserStore.isSystemAdminForCurrentUser();
const isSystemAdmin = UserStore.isSystemAdminForCurrentUser();
let createPublicChannelIcon = (
<OverlayTrigger
delayShow={500}
placement='top'
overlay={createChannelTootlip}
>
<a
className='add-channel-btn'
href='#'
onClick={this.showNewChannelModal.bind(this, Constants.OPEN_CHANNEL)}
>
{'+'}
</a>
</OverlayTrigger>
);
let createPrivateChannelIcon = (
<OverlayTrigger
delayShow={500}
placement='top'
overlay={createGroupTootlip}
>
<a
className='add-channel-btn'
href='#'
onClick={this.showNewChannelModal.bind(this, Constants.PRIVATE_CHANNEL)}
>
{'+'}
</a>
</OverlayTrigger>
);
if (global.window.mm_license.IsLicensed === 'true') {
if (global.window.mm_config.RestrictPublicChannelManagement === Constants.PERMISSIONS_SYSTEM_ADMIN && !isSystemAdmin) {
createPublicChannelIcon = null;
} else if (global.window.mm_config.RestrictPublicChannelManagement === Constants.PERMISSIONS_TEAM_ADMIN && !isAdmin) {
createPublicChannelIcon = null;
}
if (global.window.mm_config.RestrictPrivateChannelManagement === Constants.PERMISSIONS_SYSTEM_ADMIN && !isSystemAdmin) {
createPrivateChannelIcon = null;
} else if (global.window.mm_config.RestrictPrivateChannelManagement === Constants.PERMISSIONS_TEAM_ADMIN && !isAdmin) {
createPrivateChannelIcon = null;
}
}
2015-06-14 23:53:32 -08:00
return (
2016-02-08 07:26:10 -05:00
<div
className='sidebar--left'
id='sidebar-left'
key='sidebar-left'
2016-02-08 07:26:10 -05:00
>
<NewChannelFlow
show={showChannelModal}
channelType={this.state.newChannelModalType}
onModalDismissed={this.hideNewChannelModal}
/>
<MoreDirectChannels
show={this.state.showDirectChannelsModal}
onModalDismissed={this.hideMoreDirectChannelsModal}
/>
<SidebarHeader
2016-02-08 07:26:10 -05:00
teamDisplayName={this.state.currentTeam.display_name}
teamName={this.state.currentTeam.name}
teamType={this.state.currentTeam.type}
currentUser={this.state.currentUser}
/>
2015-06-14 23:53:32 -08:00
<UnreadChannelIndicator
show={this.state.showTopUnread}
extraClass='nav-pills__unread-indicator-top'
text={above}
/>
<UnreadChannelIndicator
show={this.state.showBottomUnread}
extraClass='nav-pills__unread-indicator-bottom'
text={below}
/>
<div
ref='container'
className='nav-pills__container'
onScroll={this.onScroll}
>
<ul className='nav nav-pills nav-stacked'>
<li>
<h4>
<FormattedMessage
id='sidebar.channels'
defaultMessage='Channels'
/>
{createPublicChannelIcon}
</h4>
</li>
{publicChannelItems}
<li>
<a
href='#'
className='nav-more'
onClick={this.showMoreChannelsModal}
>
<FormattedMessage
id='sidebar.moreElips'
defaultMessage='More...'
/>
</a>
</li>
2015-06-14 23:53:32 -08:00
</ul>
<ul className='nav nav-pills nav-stacked'>
<li>
<h4>
<FormattedMessage
id='sidebar.pg'
defaultMessage='Private Groups'
/>
{createPrivateChannelIcon}
</h4>
</li>
2015-06-14 23:53:32 -08:00
{privateChannelItems}
</ul>
<ul className='nav nav-pills nav-stacked'>
<li>
<h4>
<FormattedMessage
id='sidebar.direct'
defaultMessage='Direct Messages'
/>
</h4>
</li>
2015-06-14 23:53:32 -08:00
{directMessageItems}
{directDivider}
{directMessageNonTeamItems}
{directMessageMore}
2015-06-14 23:53:32 -08:00
</ul>
</div>
</div>
);
}
}