summaryrefslogtreecommitdiffstats
path: root/web/react/components/channel_header.jsx
diff options
context:
space:
mode:
Diffstat (limited to 'web/react/components/channel_header.jsx')
-rw-r--r--web/react/components/channel_header.jsx249
1 files changed, 249 insertions, 0 deletions
diff --git a/web/react/components/channel_header.jsx b/web/react/components/channel_header.jsx
new file mode 100644
index 000000000..006c168ba
--- /dev/null
+++ b/web/react/components/channel_header.jsx
@@ -0,0 +1,249 @@
+// 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;
+
+function getExtraInfoStateFromStores() {
+ return {
+ extra_info: ChannelStore.getCurrentExtraInfo()
+ };
+}
+
+var ExtraMembers = React.createClass({
+ componentDidMount: function() {
+ ChannelStore.addExtraInfoChangeListener(this._onChange);
+ ChannelStore.addChangeListener(this._onChange);
+
+ 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');
+ }
+ });
+
+ },
+ componentWillUnmount: function() {
+ ChannelStore.removeExtraInfoChangeListener(this._onChange);
+ ChannelStore.removeChangeListener(this._onChange);
+ },
+ _onChange: function() {
+ var newState = getExtraInfoStateFromStores();
+ if (!utils.areStatesEqual(newState, this.state)) {
+ this.setState(newState);
+ }
+ },
+ getInitialState: function() {
+ return getExtraInfoStateFromStores();
+ },
+ render: function() {
+ var count = this.state.extra_info.members.length == 0 ? "-" : this.state.extra_info.members.length;
+ count = this.state.extra_info.members.length > 19 ? "20+" : count;
+ var data_content = "";
+
+ this.state.extra_info.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="#">Invite 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 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>
+ );
+ }
+});
+
+