mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
257 lines
12 KiB
JavaScript
257 lines
12 KiB
JavaScript
// Copyright (c) 2015 Spinpunch, Inc. All Rights Reserved.
|
|
// See License.txt for license information.
|
|
|
|
|
|
var utils = require('../utils/utils.jsx');
|
|
var client = require('../utils/client.jsx');
|
|
var AsyncClient = require('../utils/async_client.jsx');
|
|
var UserStore = require('../stores/user_store.jsx');
|
|
var ChannelStore = require('../stores/channel_store.jsx');
|
|
var TeamStore = require('../stores/team_store.jsx');
|
|
|
|
var UserProfile = require('./user_profile.jsx');
|
|
var MessageWrapper = require('./message_wrapper.jsx');
|
|
|
|
var Constants = require('../utils/constants.jsx');
|
|
var ActionTypes = Constants.ActionTypes;
|
|
var AppDispatcher = require('../dispatcher/app_dispatcher.jsx');
|
|
|
|
function getCountsStateFromStores() {
|
|
var count = 0;
|
|
var channels = ChannelStore.getAll();
|
|
var members = ChannelStore.getAllMembers();
|
|
|
|
channels.forEach(function(channel) {
|
|
var channelMember = members[channel.id];
|
|
if (channel.type === 'D') {
|
|
count += channel.total_msg_count - channelMember.msg_count;
|
|
} else {
|
|
if (channelMember.mention_count > 0) {
|
|
count += channelMember.mention_count;
|
|
} else if (channelMember.notify_level !== "quiet" && channel.total_msg_count - channelMember.msg_count > 0) {
|
|
count += 1;
|
|
}
|
|
}
|
|
});
|
|
|
|
return { count: count };
|
|
}
|
|
|
|
var NotifyCounts = React.createClass({
|
|
componentDidMount: function() {
|
|
ChannelStore.addChangeListener(this._onChange);
|
|
},
|
|
componentWillUnmount: function() {
|
|
ChannelStore.removeChangeListener(this._onChange);
|
|
},
|
|
_onChange: function() {
|
|
var newState = getCountsStateFromStores();
|
|
if (!utils.areStatesEqual(newState, this.state)) {
|
|
this.setState(newState);
|
|
}
|
|
},
|
|
getInitialState: function() {
|
|
return getCountsStateFromStores();
|
|
},
|
|
render: function() {
|
|
if (this.state.count) {
|
|
return <span className="badge badge-notify">{ this.state.count }</span>;
|
|
} else {
|
|
return null;
|
|
}
|
|
}
|
|
});
|
|
|
|
function getStateFromStores() {
|
|
return {
|
|
channel: ChannelStore.getCurrent(),
|
|
member: ChannelStore.getCurrentMember(),
|
|
users: ChannelStore.getCurrentExtraInfo().members
|
|
};
|
|
}
|
|
|
|
module.exports = React.createClass({
|
|
displayName: 'Navbar',
|
|
|
|
componentDidMount: function() {
|
|
ChannelStore.addChangeListener(this._onChange);
|
|
ChannelStore.addExtraInfoChangeListener(this._onChange);
|
|
$('.inner__wrap').click(this.hideSidebars);
|
|
|
|
$('body').on('click.infopopover', function(e) {
|
|
if ($(e.target).attr('data-toggle') !== 'popover'
|
|
&& $(e.target).parents('.popover.in').length === 0) {
|
|
$('.info-popover').popover('hide');
|
|
}
|
|
});
|
|
|
|
},
|
|
componentWillUnmount: function() {
|
|
ChannelStore.removeChangeListener(this._onChange);
|
|
},
|
|
handleSubmit: function(e) {
|
|
e.preventDefault();
|
|
},
|
|
handleLeave: function(e) {
|
|
client.leaveChannel(this.state.channel.id,
|
|
function(data, text, req) {
|
|
AsyncClient.getChannels(true);
|
|
window.location.href = TeamStore.getCurrentTeamUrl() + '/channels/town-square';
|
|
}.bind(this),
|
|
function(err) {
|
|
AsyncClient.dispatchError(err, "handleLeave");
|
|
}
|
|
);
|
|
},
|
|
hideSidebars: function(e) {
|
|
var windowWidth = $(window).outerWidth();
|
|
if(windowWidth <= 768) {
|
|
AppDispatcher.handleServerAction({
|
|
type: ActionTypes.RECIEVED_SEARCH,
|
|
results: null
|
|
});
|
|
|
|
AppDispatcher.handleServerAction({
|
|
type: ActionTypes.RECIEVED_POST_SELECTED,
|
|
results: null
|
|
});
|
|
|
|
if (e.target.className != 'navbar-toggle' && e.target.className != 'icon-bar') {
|
|
$('.inner__wrap').removeClass('move--right move--left move--left-small');
|
|
$('.sidebar--left').removeClass('move--right');
|
|
$('.sidebar--right').removeClass('move--left');
|
|
$('.sidebar--menu').removeClass('move--left');
|
|
}
|
|
}
|
|
},
|
|
toggleLeftSidebar: function() {
|
|
$('.inner__wrap').toggleClass('move--right');
|
|
$('.sidebar--left').toggleClass('move--right');
|
|
},
|
|
toggleRightSidebar: function() {
|
|
$('.inner__wrap').toggleClass('move--left-small');
|
|
$('.sidebar--menu').toggleClass('move--left');
|
|
},
|
|
_onChange: function() {
|
|
this.setState(getStateFromStores());
|
|
$("#navbar .navbar-brand .description").popover({placement : 'bottom', trigger: 'click', html: true});
|
|
},
|
|
getInitialState: function() {
|
|
return getStateFromStores();
|
|
},
|
|
render: function() {
|
|
|
|
var currentId = UserStore.getCurrentId();
|
|
var popoverContent = "";
|
|
var channelTitle = this.props.teamDisplayName;
|
|
var isAdmin = false;
|
|
var isDirect = false;
|
|
var description = ""
|
|
var channel = this.state.channel;
|
|
|
|
if (channel) {
|
|
description = utils.textToJsx(channel.description, {"singleline": true, "noMentionHighlight": true});
|
|
popoverContent = React.renderToString(<MessageWrapper message={channel.description}/>);
|
|
isAdmin = this.state.member.roles.indexOf("admin") > -1;
|
|
|
|
if (channel.type === 'O') {
|
|
channelTitle = channel.display_name;
|
|
} else if (channel.type === 'P') {
|
|
channelTitle = channel.display_name;
|
|
} else if (channel.type === 'D') {
|
|
isDirect = true;
|
|
if (this.state.users.length > 1) {
|
|
if (this.state.users[0].id === currentId) {
|
|
channelTitle = <UserProfile userId={this.state.users[1].id} />;
|
|
} else {
|
|
channelTitle = <UserProfile userId={this.state.users[0].id} />;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (channel.description.length == 0) {
|
|
popoverContent = React.renderToString(<div>No channel description yet. <br /><a href='#' data-toggle='modal' data-desc={channel.description} data-title={channel.display_name} data-channelid={channel.id} data-target='#edit_channel'>Click here</a> to add one.</div>);
|
|
}
|
|
}
|
|
|
|
var navbar_collapse_button = currentId != null ? null :
|
|
<button type="button" className="navbar-toggle" data-toggle="collapse" data-target="#navbar-collapse-1">
|
|
<span className="sr-only">Toggle sidebar</span>
|
|
<span className="icon-bar"></span>
|
|
<span className="icon-bar"></span>
|
|
<span className="icon-bar"></span>
|
|
</button>;
|
|
var sidebar_collapse_button = currentId == null ? null :
|
|
<button type="button" className="navbar-toggle" data-toggle="collapse" data-target="#sidebar-nav" onClick={this.toggleLeftSidebar}>
|
|
<span className="sr-only">Toggle sidebar</span>
|
|
<span className="icon-bar"></span>
|
|
<span className="icon-bar"></span>
|
|
<span className="icon-bar"></span>
|
|
<NotifyCounts />
|
|
</button>;
|
|
var right_sidebar_collapse_button= currentId == null ? null :
|
|
<button type="button" className="navbar-toggle menu-toggle pull-right" data-toggle="collapse" data-target="#sidebar-nav" onClick={this.toggleRightSidebar}>
|
|
<span dangerouslySetInnerHTML={{__html: Constants.MENU_ICON }} />
|
|
</button>;
|
|
|
|
|
|
return (
|
|
<nav className="navbar navbar-default navbar-fixed-top" role="navigation">
|
|
<div className="container-fluid theme">
|
|
<div className="navbar-header">
|
|
{ navbar_collapse_button }
|
|
{ sidebar_collapse_button }
|
|
{ right_sidebar_collapse_button }
|
|
{ !isDirect && channel ?
|
|
<div className="navbar-brand">
|
|
<div className="dropdown">
|
|
<div data-toggle="popover" data-content={popoverContent} className="description info-popover"></div>
|
|
<a href="#" className="dropdown-toggle theme" type="button" id="channel_header_dropdown" data-toggle="dropdown" aria-expanded="true">
|
|
<span className="heading">{channelTitle} </span>
|
|
<span className="glyphicon glyphicon-chevron-down header-dropdown__icon"></span>
|
|
</a>
|
|
<ul className="dropdown-menu" role="menu" aria-labelledby="channel_header_dropdown">
|
|
{ !ChannelStore.isDefault(channel) ?
|
|
<li role="presentation"><a role="menuitem" data-toggle="modal" data-target="#channel_invite" href="#">Add Members</a></li>
|
|
: null
|
|
}
|
|
{ isAdmin && !ChannelStore.isDefault(channel) ?
|
|
<li role="presentation"><a role="menuitem" data-toggle="modal" data-target="#channel_members" href="#">Manage Members</a></li>
|
|
: null
|
|
}
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#edit_channel" data-desc={channel.description} data-title={channel.display_name} data-channelid={channel.id}>Set Channel Description...</a></li>
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#channel_notifications" data-title={channel.display_name} data-channelid={channel.id}>Notification Preferences</a></li>
|
|
{ isAdmin && !ChannelStore.isDefault(channel) ?
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#rename_channel" data-display={channel.display_name} data-name={channel.name} data-channelid={channel.id}>Rename Channel...</a></li>
|
|
: null
|
|
}
|
|
{ isAdmin && !ChannelStore.isDefault(channel) ?
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#delete_channel" data-title={channel.display_name} data-channelid={channel.id}>Delete Channel...</a></li>
|
|
: null
|
|
}
|
|
{ !ChannelStore.isDefault(channel) ?
|
|
<li role="presentation"><a role="menuitem" href="#" onClick={this.handleLeave}>Leave Channel</a></li>
|
|
: null
|
|
}
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
: null
|
|
}
|
|
{ isDirect && channel ?
|
|
<div className="navbar-brand">
|
|
<a href="#" className="heading">{ channelTitle }</a>
|
|
</div>
|
|
: null }
|
|
{ !channel ?
|
|
<div className="navbar-brand">
|
|
<a href="/" className="heading">{ channelTitle }</a>
|
|
</div>
|
|
: "" }
|
|
</div>
|
|
</div>
|
|
</nav>
|
|
);
|
|
}
|
|
});
|