mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
228 lines
11 KiB
JavaScript
228 lines
11 KiB
JavaScript
// Copyright (c) 2015 Spinpunch, Inc. All Rights Reserved.
|
|
// See License.txt for license information.
|
|
|
|
var ChannelStore = require('../stores/channel_store.jsx');
|
|
var UserStore = require('../stores/user_store.jsx');
|
|
var PostStore = require('../stores/post_store.jsx');
|
|
var UserProfile = require( './user_profile.jsx' );
|
|
var NavbarSearchBox =require('./search_bar.jsx');
|
|
var AsyncClient = require('../utils/async_client.jsx');
|
|
var Client = require('../utils/client.jsx');
|
|
var utils = require('../utils/utils.jsx');
|
|
var MessageWrapper = require('./message_wrapper.jsx');
|
|
|
|
var AppDispatcher = require('../dispatcher/app_dispatcher.jsx');
|
|
var Constants = require('../utils/constants.jsx');
|
|
var ActionTypes = Constants.ActionTypes;
|
|
|
|
var ExtraMembers = React.createClass({
|
|
componentDidMount: function() {
|
|
var originalLeave = $.fn.popover.Constructor.prototype.leave;
|
|
$.fn.popover.Constructor.prototype.leave = function(obj) {
|
|
var self = obj instanceof this.constructor ? obj : $(obj.currentTarget)[this.type](this.getDelegateOptions()).data('bs.' + this.type);
|
|
originalLeave.call(this, obj);
|
|
|
|
if (obj.currentTarget && self.$tip) {
|
|
self.$tip.one('mouseenter', function() {
|
|
clearTimeout(self.timeout);
|
|
self.$tip.one('mouseleave', function() {
|
|
$.fn.popover.Constructor.prototype.leave.call(self, self);
|
|
});
|
|
})
|
|
}
|
|
};
|
|
|
|
$("#member_popover").popover({placement : 'bottom', trigger: 'click', html: true});
|
|
$('body').on('click', function (e) {
|
|
if ($(e.target.parentNode.parentNode)[0] !== $("#member_popover")[0] && $(e.target).parents('.popover.in').length === 0) {
|
|
$("#member_popover").popover('hide');
|
|
}
|
|
});
|
|
|
|
},
|
|
render: function() {
|
|
var count = this.props.members.length == 0 ? "-" : this.props.members.length;
|
|
count = this.props.members.length > 19 ? "20+" : count;
|
|
var data_content = "";
|
|
|
|
this.props.members.forEach(function(m) {
|
|
data_content += "<div style='white-space: nowrap'>" + m.username + "</div>";
|
|
});
|
|
|
|
return (
|
|
<div style={{"cursor" : "pointer"}} id="member_popover" data-toggle="popover" data-content={data_content} data-original-title="Members" >
|
|
<div id="member_tooltip" data-toggle="tooltip" title="View Channel Members">
|
|
{count} <span className="glyphicon glyphicon-user" aria-hidden="true"></span>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
});
|
|
|
|
function getStateFromStores() {
|
|
return {
|
|
channel: ChannelStore.getCurrent(),
|
|
memberChannel: ChannelStore.getCurrentMember(),
|
|
memberTeam: UserStore.getCurrentUser(),
|
|
users: ChannelStore.getCurrentExtraInfo().members,
|
|
search_visible: PostStore.getSearchResults() != null
|
|
};
|
|
}
|
|
|
|
module.exports = React.createClass({
|
|
componentDidMount: function() {
|
|
ChannelStore.addChangeListener(this._onChange);
|
|
ChannelStore.addExtraInfoChangeListener(this._onChange);
|
|
PostStore.addSearchChangeListener(this._onChange);
|
|
UserStore.addChangeListener(this._onChange);
|
|
},
|
|
componentWillUnmount: function() {
|
|
ChannelStore.removeChangeListener(this._onChange);
|
|
ChannelStore.removeExtraInfoChangeListener(this._onChange);
|
|
PostStore.removeSearchChangeListener(this._onChange);
|
|
UserStore.addChangeListener(this._onChange);
|
|
},
|
|
_onChange: function() {
|
|
var newState = getStateFromStores();
|
|
if (!utils.areStatesEqual(newState, this.state)) {
|
|
this.setState(newState);
|
|
}
|
|
$(".channel-header__info .description").popover({placement : 'bottom', trigger: 'hover', html: true, delay: {show: 500, hide: 500}});
|
|
},
|
|
getInitialState: function() {
|
|
return getStateFromStores();
|
|
},
|
|
handleLeave: function(e) {
|
|
var self = this;
|
|
Client.leaveChannel(this.state.channel.id,
|
|
function(data) {
|
|
var townsquare = ChannelStore.getByName('town-square');
|
|
utils.switchChannel(townsquare);
|
|
}.bind(this),
|
|
function(err) {
|
|
AsyncClient.dispatchError(err, "handleLeave");
|
|
}.bind(this)
|
|
);
|
|
},
|
|
searchMentions: function(e) {
|
|
e.preventDefault();
|
|
|
|
var user = UserStore.getCurrentUser();
|
|
|
|
var terms = "";
|
|
if (user.notify_props && user.notify_props.mention_keys) {
|
|
terms = UserStore.getCurrentMentionKeys().join(' ');
|
|
}
|
|
|
|
AppDispatcher.handleServerAction({
|
|
type: ActionTypes.RECIEVED_SEARCH_TERM,
|
|
term: terms,
|
|
do_search: false
|
|
});
|
|
|
|
Client.search(
|
|
terms,
|
|
function(data) {
|
|
AppDispatcher.handleServerAction({
|
|
type: ActionTypes.RECIEVED_SEARCH,
|
|
results: data,
|
|
is_mention_search: true
|
|
});
|
|
},
|
|
function(err) {
|
|
dispatchError(err, "search");
|
|
}
|
|
);
|
|
},
|
|
render: function() {
|
|
|
|
if (this.state.channel == null) {
|
|
return (
|
|
<div></div>
|
|
);
|
|
}
|
|
|
|
var description = utils.textToJsx(this.state.channel.description, {"singleline": true, "noMentionHighlight": true});
|
|
var popoverContent = React.renderToString(<MessageWrapper message={this.state.channel.description}/>);
|
|
var channelTitle = "";
|
|
var channelName = this.state.channel.name;
|
|
var currentId = UserStore.getCurrentId();
|
|
var isAdmin = this.state.memberChannel.roles.indexOf("admin") > -1 || this.state.memberTeam.roles.indexOf("admin") > -1;
|
|
var searchForm = <th className="search-bar__container"><NavbarSearchBox /></th>;
|
|
var isDirect = false;
|
|
|
|
if (this.state.channel.type === 'O') {
|
|
channelTitle = this.state.channel.display_name;
|
|
} else if (this.state.channel.type === 'P') {
|
|
channelTitle = this.state.channel.display_name;
|
|
} else if (this.state.channel.type === 'D') {
|
|
isDirect = true;
|
|
if (this.state.users.length > 1) {
|
|
if (this.state.users[0].id === UserStore.getCurrentId()) {
|
|
channelTitle = <UserProfile userId={this.state.users[1].id} overwriteName={this.state.users[1].full_name ? this.state.users[1].full_name : this.state.users[1].username} />;
|
|
} else {
|
|
channelTitle = <UserProfile userId={this.state.users[0].id} overwriteName={this.state.users[0].full_name ? this.state.users[0].full_name : this.state.users[0].username} />;
|
|
}
|
|
}
|
|
}
|
|
|
|
return (
|
|
<table className="channel-header alt">
|
|
<tr>
|
|
<th>
|
|
{ !isDirect ?
|
|
<div className="channel-header__info">
|
|
<div className="dropdown">
|
|
<a href="#" className="dropdown-toggle theme" type="button" id="channel_header_dropdown" data-toggle="dropdown" aria-expanded="true">
|
|
<strong className="heading">{channelTitle} </strong>
|
|
<span className="glyphicon glyphicon-chevron-down header-dropdown__icon"></span>
|
|
</a>
|
|
<ul className="dropdown-menu" role="menu" aria-labelledby="channel_header_dropdown">
|
|
<li role="presentation"><a role="menuitem" data-toggle="modal" data-target="#channel_info" data-channelid={this.state.channel.id} href="#">View Info</a></li>
|
|
<li role="presentation"><a role="menuitem" data-toggle="modal" data-target="#channel_invite" href="#">Add Members</a></li>
|
|
{ isAdmin ?
|
|
<li role="presentation"><a role="menuitem" data-toggle="modal" data-target="#channel_members" href="#">Manage Members</a></li>
|
|
: ""
|
|
}
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#edit_channel" data-desc={this.state.channel.description} data-title={this.state.channel.display_name} data-channelid={this.state.channel.id}>Set Channel Description...</a></li>
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#channel_notifications" data-title={this.state.channel.display_name} data-channelid={this.state.channel.id}>Notification Preferences</a></li>
|
|
{ isAdmin && channelName != Constants.DEFAULT_CHANNEL ?
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#rename_channel" data-display={this.state.channel.display_name} data-name={this.state.channel.name} data-channelid={this.state.channel.id}>Rename Channel...</a></li>
|
|
: ""
|
|
}
|
|
{ isAdmin && channelName != Constants.DEFAULT_CHANNEL ?
|
|
<li role="presentation"><a role="menuitem" href="#" data-toggle="modal" data-target="#delete_channel" data-title={this.state.channel.display_name} data-channelid={this.state.channel.id}>Delete Channel...</a></li>
|
|
: ""
|
|
}
|
|
{ channelName != Constants.DEFAULT_CHANNEL ?
|
|
<li role="presentation"><a role="menuitem" href="#" onClick={this.handleLeave}>Leave Channel</a></li>
|
|
: ""
|
|
}
|
|
</ul>
|
|
</div>
|
|
<div data-toggle="popover" data-content={popoverContent} className="description">{description}</div>
|
|
</div>
|
|
:
|
|
<a href="#"><strong className="heading">{channelTitle}</strong></a>
|
|
}
|
|
</th>
|
|
<th><ExtraMembers members={this.state.users} channelId={this.state.channel.id} /></th>
|
|
{ searchForm }
|
|
<th>
|
|
<div className="dropdown" style={{"marginLeft":"5px", "marginRight":"10px"}}>
|
|
<a href="#" className="dropdown-toggle theme" type="button" id="channel_header_right_dropdown" data-toggle="dropdown" aria-expanded="true">
|
|
<i className="fa fa-caret-down"></i>
|
|
</a>
|
|
<ul className="dropdown-menu" role="menu" aria-labelledby="channel_header_right_dropdown" style={{"left": "-150px"}}>
|
|
<li role="presentation"><a role="menuitem" href="#" onClick={this.searchMentions}>Recent Mentions</a></li>
|
|
</ul>
|
|
</div>
|
|
</th>
|
|
</tr>
|
|
</table>
|
|
);
|
|
}
|
|
});
|
|
|
|
|