mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
633 lines
23 KiB
JavaScript
633 lines
23 KiB
JavaScript
// Copyright (c) 2015 Mattermost, Inc. All Rights Reserved.
|
||
// See License.txt for license information.
|
||
|
||
const AsyncClient = require('../utils/async_client.jsx');
|
||
const BrowserStore = require('../stores/browser_store.jsx');
|
||
const ChannelStore = require('../stores/channel_store.jsx');
|
||
const Client = require('../utils/client.jsx');
|
||
const Constants = require('../utils/constants.jsx');
|
||
const PreferenceStore = require('../stores/preference_store.jsx');
|
||
const NewChannelFlow = require('./new_channel_flow.jsx');
|
||
const SearchBox = require('./search_bar.jsx');
|
||
const SidebarHeader = require('./sidebar_header.jsx');
|
||
const SocketStore = require('../stores/socket_store.jsx');
|
||
const TeamStore = require('../stores/team_store.jsx');
|
||
const UnreadChannelIndicator = require('./unread_channel_indicator.jsx');
|
||
const UserStore = require('../stores/user_store.jsx');
|
||
const Utils = require('../utils/utils.jsx');
|
||
|
||
export default class Sidebar extends React.Component {
|
||
constructor(props) {
|
||
super(props);
|
||
|
||
this.badgesActive = false;
|
||
this.firstUnreadChannel = null;
|
||
this.lastUnreadChannel = null;
|
||
|
||
this.getStateFromStores = this.getStateFromStores.bind(this);
|
||
|
||
this.onChange = this.onChange.bind(this);
|
||
this.onScroll = this.onScroll.bind(this);
|
||
this.onResize = this.onResize.bind(this);
|
||
this.updateUnreadIndicators = this.updateUnreadIndicators.bind(this);
|
||
this.handleLeaveDirectChannel = this.handleLeaveDirectChannel.bind(this);
|
||
this.createChannelElement = this.createChannelElement.bind(this);
|
||
|
||
const state = this.getStateFromStores();
|
||
state.modal = '';
|
||
state.loadingDMChannel = -1;
|
||
|
||
this.state = state;
|
||
}
|
||
getStateFromStores() {
|
||
const members = ChannelStore.getAllMembers();
|
||
var teamMemberMap = UserStore.getActiveOnlyProfiles();
|
||
var currentId = ChannelStore.getCurrentId();
|
||
|
||
var teammates = [];
|
||
for (var id in teamMemberMap) {
|
||
if (id === UserStore.getCurrentId()) {
|
||
continue;
|
||
}
|
||
teammates.push(teamMemberMap[id]);
|
||
}
|
||
|
||
const preferences = PreferenceStore.getPreferences('direct_channels', 'show_hide');
|
||
|
||
// Create lists of all read and unread direct channels
|
||
var visibleDirectChannels = [];
|
||
var hiddenDirectChannels = [];
|
||
for (var i = 0; i < teammates.length; i++) {
|
||
const teammate = teammates[i];
|
||
|
||
if (teammate.id === UserStore.getCurrentId()) {
|
||
continue;
|
||
}
|
||
|
||
var channelName = '';
|
||
if (teammate.id > UserStore.getCurrentId()) {
|
||
channelName = UserStore.getCurrentId() + '__' + teammate.id;
|
||
} else {
|
||
channelName = teammate.id + '__' + UserStore.getCurrentId();
|
||
}
|
||
|
||
let forceShow = false;
|
||
let channel = ChannelStore.getByName(channelName);
|
||
|
||
if (!channel) {
|
||
channel = {};
|
||
channel.fake = true;
|
||
channel.name = channelName;
|
||
channel.last_post_at = 0;
|
||
channel.total_msg_count = 0;
|
||
channel.type = 'D';
|
||
} else {
|
||
const member = members[channel.id];
|
||
const msgCount = channel.total_msg_count - member.msg_count;
|
||
|
||
forceShow = currentId === channel.id || msgCount > 0;
|
||
}
|
||
|
||
channel.display_name = teammate.username;
|
||
channel.teammate_id = teammate.id;
|
||
channel.status = UserStore.getStatus(teammate.id);
|
||
|
||
if (preferences.some((preference) => (preference.alt_id === teammate.id && preference.value !== 'false'))) {
|
||
visibleDirectChannels.push(channel);
|
||
} else if (forceShow) {
|
||
// make sure that unread direct channels are visible
|
||
const preference = PreferenceStore.setPreferenceWithAltId('direct_channels', 'show_hide', teammate.id, 'true');
|
||
AsyncClient.setPreferences([preference]);
|
||
|
||
visibleDirectChannels.push(channel);
|
||
} else {
|
||
hiddenDirectChannels.push(channel);
|
||
}
|
||
}
|
||
|
||
visibleDirectChannels.sort(this.sortChannelsByDisplayName);
|
||
hiddenDirectChannels.sort(this.sortChannelsByDisplayName);
|
||
|
||
return {
|
||
activeId: currentId,
|
||
channels: ChannelStore.getAll(),
|
||
members: members,
|
||
visibleDirectChannels: visibleDirectChannels,
|
||
hiddenDirectChannels: hiddenDirectChannels
|
||
};
|
||
}
|
||
|
||
componentDidMount() {
|
||
ChannelStore.addChangeListener(this.onChange);
|
||
UserStore.addChangeListener(this.onChange);
|
||
UserStore.addStatusesChangeListener(this.onChange);
|
||
TeamStore.addChangeListener(this.onChange);
|
||
SocketStore.addChangeListener(this.onSocketChange);
|
||
PreferenceStore.addChangeListener(this.onChange);
|
||
|
||
AsyncClient.getDirectChannels();
|
||
|
||
$('.nav-pills__container').perfectScrollbar();
|
||
|
||
this.updateTitle();
|
||
this.updateUnreadIndicators();
|
||
|
||
$(window).on('resize', this.onResize);
|
||
}
|
||
shouldComponentUpdate(nextProps, nextState) {
|
||
if (!Utils.areStatesEqual(nextProps, this.props)) {
|
||
return true;
|
||
}
|
||
|
||
if (!Utils.areStatesEqual(nextState, this.state)) {
|
||
return true;
|
||
}
|
||
return false;
|
||
}
|
||
componentDidUpdate() {
|
||
this.updateTitle();
|
||
this.updateUnreadIndicators();
|
||
}
|
||
componentWillUnmount() {
|
||
$(window).off('resize', this.onResize);
|
||
|
||
ChannelStore.removeChangeListener(this.onChange);
|
||
UserStore.removeChangeListener(this.onChange);
|
||
UserStore.removeStatusesChangeListener(this.onChange);
|
||
TeamStore.removeChangeListener(this.onChange);
|
||
SocketStore.removeChangeListener(this.onSocketChange);
|
||
PreferenceStore.removeChangeListener(this.onChange);
|
||
}
|
||
onChange() {
|
||
var newState = this.getStateFromStores();
|
||
if (!Utils.areStatesEqual(newState, this.state)) {
|
||
this.setState(newState);
|
||
}
|
||
}
|
||
onSocketChange(msg) {
|
||
if (msg.action === 'posted') {
|
||
if (ChannelStore.getCurrentId() === msg.channel_id) {
|
||
if (window.isActive) {
|
||
AsyncClient.updateLastViewedAt();
|
||
}
|
||
} else {
|
||
AsyncClient.getChannels();
|
||
}
|
||
|
||
if (UserStore.getCurrentId() !== msg.user_id) {
|
||
var mentions = [];
|
||
if (msg.props.mentions) {
|
||
mentions = JSON.parse(msg.props.mentions);
|
||
}
|
||
var channel = ChannelStore.get(msg.channel_id);
|
||
|
||
const user = UserStore.getCurrentUser();
|
||
const member = ChannelStore.getMember(msg.channel_id);
|
||
|
||
var notifyLevel = member && member.notify_props ? member.notify_props.desktop : 'default';
|
||
if (notifyLevel === 'default') {
|
||
notifyLevel = user.notify_props.desktop;
|
||
}
|
||
|
||
if (notifyLevel === 'none') {
|
||
return;
|
||
} else if (notifyLevel === 'mention' && mentions.indexOf(user.id) === -1 && channel.type !== 'D') {
|
||
return;
|
||
}
|
||
|
||
var username = 'Someone';
|
||
if (UserStore.hasProfile(msg.user_id)) {
|
||
username = UserStore.getProfile(msg.user_id).username;
|
||
}
|
||
|
||
var title = 'Posted';
|
||
if (channel) {
|
||
title = channel.display_name;
|
||
}
|
||
|
||
var repRegex = new RegExp('<br>', 'g');
|
||
var post = JSON.parse(msg.props.post);
|
||
var msgProps = msg.props;
|
||
var notifyText = post.message.replace(repRegex, '\n').replace(/\n+/g, ' ').replace('<mention>', '').replace('</mention>', '');
|
||
|
||
if (notifyText.length > 50) {
|
||
notifyText = notifyText.substring(0, 49) + '...';
|
||
}
|
||
|
||
if (notifyText.length === 0) {
|
||
if (msgProps.image) {
|
||
Utils.notifyMe(title, username + ' uploaded an image', channel);
|
||
} else if (msgProps.otherFile) {
|
||
Utils.notifyMe(title, username + ' uploaded a file', channel);
|
||
} else {
|
||
Utils.notifyMe(title, username + ' did something new', channel);
|
||
}
|
||
} else {
|
||
Utils.notifyMe(title, username + ' wrote: ' + notifyText, channel);
|
||
}
|
||
if (!user.notify_props || user.notify_props.desktop_sound === 'true') {
|
||
Utils.ding();
|
||
}
|
||
}
|
||
} else if (msg.action === 'viewed') {
|
||
if (ChannelStore.getCurrentId() !== msg.channel_id && UserStore.getCurrentId() === msg.user_id) {
|
||
AsyncClient.getChannel(msg.channel_id);
|
||
}
|
||
} else if (msg.action === 'user_added') {
|
||
if (UserStore.getCurrentId() === msg.user_id) {
|
||
AsyncClient.getChannel(msg.channel_id);
|
||
}
|
||
} else if (msg.action === 'user_removed') {
|
||
if (msg.user_id === UserStore.getCurrentId()) {
|
||
AsyncClient.getChannels(true);
|
||
|
||
if (msg.props.remover !== msg.user_id && msg.props.channel_id === ChannelStore.getCurrentId() && $('#removed_from_channel').length > 0) {
|
||
var sentState = {};
|
||
sentState.channelName = ChannelStore.getCurrent().display_name;
|
||
sentState.remover = UserStore.getProfile(msg.props.remover).username;
|
||
|
||
BrowserStore.setItem('channel-removed-state', sentState);
|
||
$('#removed_from_channel').modal('show');
|
||
}
|
||
}
|
||
}
|
||
}
|
||
updateTitle() {
|
||
const channel = ChannelStore.getCurrent();
|
||
if (channel) {
|
||
let currentSiteName = '';
|
||
if (global.window.config.SiteName != null) {
|
||
currentSiteName = global.window.config.SiteName;
|
||
}
|
||
|
||
let currentChannelName = channel.display_name;
|
||
if (channel.type === 'D') {
|
||
currentChannelName = Utils.getDirectTeammate(channel.id).username;
|
||
}
|
||
|
||
document.title = currentChannelName + ' - ' + this.props.teamDisplayName + ' ' + currentSiteName;
|
||
}
|
||
}
|
||
onScroll() {
|
||
this.updateUnreadIndicators();
|
||
}
|
||
onResize() {
|
||
this.updateUnreadIndicators();
|
||
}
|
||
updateUnreadIndicators() {
|
||
const container = $(React.findDOMNode(this.refs.container));
|
||
|
||
var showTopUnread = false;
|
||
var showBottomUnread = false;
|
||
|
||
if (this.firstUnreadChannel) {
|
||
var firstUnreadElement = $(React.findDOMNode(this.refs[this.firstUnreadChannel]));
|
||
|
||
if (firstUnreadElement.position().top + firstUnreadElement.height() < 0) {
|
||
showTopUnread = true;
|
||
}
|
||
}
|
||
|
||
if (this.lastUnreadChannel) {
|
||
var lastUnreadElement = $(React.findDOMNode(this.refs[this.lastUnreadChannel]));
|
||
|
||
if (lastUnreadElement.position().top > container.height()) {
|
||
showBottomUnread = true;
|
||
}
|
||
}
|
||
|
||
this.setState({
|
||
showTopUnread,
|
||
showBottomUnread
|
||
});
|
||
}
|
||
|
||
handleLeaveDirectChannel(channel) {
|
||
if (!channel.leaving) {
|
||
channel.leaving = true;
|
||
|
||
const preference = PreferenceStore.setPreferenceWithAltId('direct_channels', 'show_hide', channel.teammate_id, 'false');
|
||
AsyncClient.setPreferences(
|
||
[preference],
|
||
() => {
|
||
channel.leaving = false;
|
||
},
|
||
() => {
|
||
channel.leaving = 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);
|
||
}
|
||
|
||
createChannelElement(channel, index, arr, handleClose) {
|
||
var members = this.state.members;
|
||
var activeId = this.state.activeId;
|
||
var channelMember = members[channel.id];
|
||
var msgCount;
|
||
|
||
var linkClass = '';
|
||
if (channel.id === activeId) {
|
||
linkClass = 'active';
|
||
}
|
||
|
||
var unread = false;
|
||
if (channelMember) {
|
||
msgCount = channel.total_msg_count - channelMember.msg_count;
|
||
unread = (msgCount > 0 && channelMember.notify_props.mark_unread !== 'mention') || channelMember.mention_count > 0;
|
||
}
|
||
|
||
var titleClass = '';
|
||
if (unread) {
|
||
titleClass = 'unread-title';
|
||
|
||
if (channel.id !== activeId) {
|
||
if (!this.firstUnreadChannel) {
|
||
this.firstUnreadChannel = channel.name;
|
||
}
|
||
this.lastUnreadChannel = channel.name;
|
||
}
|
||
}
|
||
|
||
var badge = null;
|
||
if (channelMember) {
|
||
if (channel.type === 'D') {
|
||
// direct message channels show badges for any number of unread posts
|
||
msgCount = channel.total_msg_count - channelMember.msg_count;
|
||
if (msgCount > 0) {
|
||
badge = <span className='badge pull-right small'>{msgCount}</span>;
|
||
this.badgesActive = true;
|
||
}
|
||
} else if (channelMember.mention_count > 0) {
|
||
// public and private channels only show badges for mentions
|
||
badge = <span className='badge pull-right small'>{channelMember.mention_count}</span>;
|
||
this.badgesActive = true;
|
||
}
|
||
} else if (this.state.loadingDMChannel === index && channel.type === 'D') {
|
||
badge = (
|
||
<img
|
||
className='channel-loading-gif pull-right'
|
||
src='/static/images/load.gif'
|
||
/>
|
||
);
|
||
}
|
||
|
||
var badgeClass;
|
||
if (msgCount > 0) {
|
||
badgeClass = 'has-badge';
|
||
}
|
||
|
||
// 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}}
|
||
/>
|
||
);
|
||
}
|
||
|
||
// 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) {
|
||
if (!e.target.attributes.getNamedItem('data-close')) {
|
||
Utils.switchChannel(channel);
|
||
} else {
|
||
handleClose(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) {
|
||
handleClick = function clickHandler(e) {
|
||
e.preventDefault();
|
||
|
||
if (!e.target.attributes.getNamedItem('data-close')) {
|
||
this.setState({loadingDMChannel: index});
|
||
|
||
Client.createDirectChannel(channel, otherUserId,
|
||
function success(data) {
|
||
this.setState({loadingDMChannel: -1});
|
||
AsyncClient.getChannel(data.id);
|
||
Utils.switchChannel(data);
|
||
}.bind(this),
|
||
function error() {
|
||
this.setState({loadingDMChannel: -1});
|
||
window.location.href = TeamStore.getCurrentTeamUrl() + '/channels/' + channel.name;
|
||
}.bind(this)
|
||
);
|
||
} else {
|
||
handleClose(channel);
|
||
}
|
||
}.bind(this);
|
||
}
|
||
}
|
||
|
||
let closeButton = null;
|
||
if (handleClose) {
|
||
closeButton = (
|
||
<span
|
||
className='sidebar-channel__close'
|
||
data-close='true'
|
||
>
|
||
{'×'}
|
||
</span>
|
||
);
|
||
}
|
||
|
||
return (
|
||
<li
|
||
key={channel.name}
|
||
ref={channel.name}
|
||
className={linkClass}
|
||
>
|
||
<a
|
||
className={'sidebar-channel ' + titleClass + ' ' + badgeClass}
|
||
href={href}
|
||
onClick={handleClick}
|
||
>
|
||
{status}
|
||
{channel.display_name}
|
||
{badge}
|
||
{closeButton}
|
||
</a>
|
||
</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 publicChannels = this.state.channels.filter((channel) => channel.type === 'O');
|
||
const publicChannelItems = publicChannels.map(this.createChannelElement);
|
||
|
||
const privateChannels = this.state.channels.filter((channel) => channel.type === 'P');
|
||
const privateChannelItems = privateChannels.map(this.createChannelElement);
|
||
|
||
const directMessageItems = this.state.visibleDirectChannels.map((channel, index, arr) => {
|
||
return this.createChannelElement(channel, index, arr, this.handleLeaveDirectChannel);
|
||
});
|
||
|
||
// update the favicon to show if there are any notifications
|
||
var link = document.createElement('link');
|
||
link.type = 'image/x-icon';
|
||
link.rel = 'shortcut icon';
|
||
link.id = 'favicon';
|
||
if (this.badgesActive) {
|
||
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.hiddenDirectChannels.length > 0) {
|
||
directMessageMore = (
|
||
<li key='more'>
|
||
<a
|
||
key={`more${this.state.hiddenDirectChannels.length}`}
|
||
href='#'
|
||
data-toggle='modal'
|
||
className='nav-more'
|
||
data-target='#more_direct_channels'
|
||
data-channels={JSON.stringify(this.state.hiddenDirectChannels)}
|
||
>
|
||
{'More (' + this.state.hiddenDirectChannels.length + ')'}
|
||
</a>
|
||
</li>
|
||
);
|
||
}
|
||
|
||
let showChannelModal = false;
|
||
if (this.state.modal !== '') {
|
||
showChannelModal = true;
|
||
}
|
||
|
||
return (
|
||
<div>
|
||
<NewChannelFlow
|
||
show={showChannelModal}
|
||
channelType={this.state.modal}
|
||
onModalDismissed={() => this.setState({modal: ''})}
|
||
/>
|
||
<SidebarHeader
|
||
teamDisplayName={this.props.teamDisplayName}
|
||
teamName={this.props.teamName}
|
||
teamType={this.props.teamType}
|
||
/>
|
||
<SearchBox />
|
||
|
||
<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>
|
||
Channels
|
||
<a
|
||
className='add-channel-btn'
|
||
href='#'
|
||
onClick={() => this.setState({modal: 'O'})}
|
||
>
|
||
{'+'}
|
||
</a>
|
||
</h4>
|
||
</li>
|
||
{publicChannelItems}
|
||
<li>
|
||
<a
|
||
href='#'
|
||
data-toggle='modal'
|
||
className='nav-more'
|
||
data-target='#more_channels'
|
||
data-channeltype='O'
|
||
>
|
||
More...
|
||
</a>
|
||
</li>
|
||
</ul>
|
||
|
||
<ul className='nav nav-pills nav-stacked'>
|
||
<li>
|
||
<h4>
|
||
Private Groups
|
||
<a
|
||
className='add-channel-btn'
|
||
href='#'
|
||
onClick={() => this.setState({modal: 'P'})}
|
||
>
|
||
{'+'}
|
||
</a>
|
||
</h4>
|
||
</li>
|
||
{privateChannelItems}
|
||
</ul>
|
||
<ul className='nav nav-pills nav-stacked'>
|
||
<li><h4>Direct Messages</h4></li>
|
||
{directMessageItems}
|
||
{directMessageMore}
|
||
</ul>
|
||
</div>
|
||
</div>
|
||
);
|
||
}
|
||
}
|
||
|
||
Sidebar.defaultProps = {
|
||
teamType: '',
|
||
teamDisplayName: ''
|
||
};
|
||
Sidebar.propTypes = {
|
||
teamType: React.PropTypes.string,
|
||
teamDisplayName: React.PropTypes.string,
|
||
teamName: React.PropTypes.string
|
||
};
|