mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
Merge branch 'master' of https://github.com/mattermost/platform into mm-1696
Conflicts: web/react/components/setting_upload.jsx web/react/components/team_import_tab.jsx
This commit is contained in:
@@ -5,6 +5,7 @@ package store
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"strings"
|
||||
"github.com/mattermost/platform/model"
|
||||
"github.com/mattermost/platform/utils"
|
||||
)
|
||||
@@ -163,6 +164,17 @@ func (us SqlUserStore) Update(user *model.User, allowActiveUpdate bool) StoreCha
|
||||
user.EmailVerified = false
|
||||
}
|
||||
|
||||
if user.Username != oldUser.Username {
|
||||
nonUsernameKeys := []string{}
|
||||
splitKeys := strings.Split(user.NotifyProps["mention_keys"], ",")
|
||||
for _, key := range splitKeys {
|
||||
if key != oldUser.Username && key != "@" + oldUser.Username {
|
||||
nonUsernameKeys = append(nonUsernameKeys, key)
|
||||
}
|
||||
}
|
||||
user.NotifyProps["mention_keys"] = strings.Join(nonUsernameKeys, ",") + user.Username + ",@" + user.Username
|
||||
}
|
||||
|
||||
if count, err := us.GetMaster().Update(user); err != nil {
|
||||
if IsUniqueConstraintError(err.Error(), "Email", "users_email_teamid_key") {
|
||||
result.Err = model.NewAppError("SqlUserStore.Update", "This email is already taken. Please choose another", "user_id="+user.Id+", "+err.Error())
|
||||
|
||||
@@ -47,7 +47,7 @@ module.exports = React.createClass({
|
||||
return (
|
||||
<div>
|
||||
<h4>{"Find Your " + utils.toTitleCase(strings.Team)}</h4>
|
||||
<p>{"An email was sent with links to any " + strings.TeamPlural}</p>
|
||||
<p>{"An email was sent with links to any " + strings.TeamPlural + " to which you are a member."}</p>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@@ -56,7 +56,7 @@ module.exports = React.createClass({
|
||||
<div>
|
||||
<h4>Find Your Team</h4>
|
||||
<form onSubmit={this.handleSubmit}>
|
||||
<p>{"We'll send you an email with links to your " + strings.TeamPlural + "."}</p>
|
||||
<p>{"Get an email with links to any " + strings.TeamPlural + " to which you are a member."}</p>
|
||||
<div className="form-group">
|
||||
<label className='control-label'>Email</label>
|
||||
<div className={ email_error ? "form-group has-error" : "form-group" }>
|
||||
|
||||
@@ -14,11 +14,21 @@ function getStateFromStores() {
|
||||
};
|
||||
}
|
||||
|
||||
module.exports = React.createClass({
|
||||
displayName: 'MoreChannelsModal',
|
||||
export default class MoreChannels extends React.Component {
|
||||
constructor(props) {
|
||||
super(props);
|
||||
|
||||
componentDidMount: function() {
|
||||
ChannelStore.addMoreChangeListener(this._onChange);
|
||||
this.onListenerChange = this.onListenerChange.bind(this);
|
||||
this.handleJoin = this.handleJoin.bind(this);
|
||||
this.handleNewChannel = this.handleNewChannel.bind(this);
|
||||
|
||||
var initState = getStateFromStores();
|
||||
initState.channelType = '';
|
||||
initState.joiningChannel = -1;
|
||||
this.state = initState;
|
||||
}
|
||||
componentDidMount() {
|
||||
ChannelStore.addMoreChangeListener(this.onListenerChange);
|
||||
$(this.refs.modal.getDOMNode()).on('shown.bs.modal', function shown() {
|
||||
asyncClient.getMoreChannels(true);
|
||||
});
|
||||
@@ -28,43 +38,42 @@ module.exports = React.createClass({
|
||||
var button = e.relatedTarget;
|
||||
self.setState({channelType: $(button).attr('data-channeltype')});
|
||||
});
|
||||
},
|
||||
componentWillUnmount: function() {
|
||||
ChannelStore.removeMoreChangeListener(this._onChange);
|
||||
},
|
||||
_onChange: function() {
|
||||
}
|
||||
componentWillUnmount() {
|
||||
ChannelStore.removeMoreChangeListener(this.onListenerChange);
|
||||
}
|
||||
onListenerChange() {
|
||||
var newState = getStateFromStores();
|
||||
if (!utils.areStatesEqual(newState.channels, this.state.channels)) {
|
||||
this.setState(newState);
|
||||
}
|
||||
},
|
||||
getInitialState: function() {
|
||||
var initState = getStateFromStores();
|
||||
initState.channelType = '';
|
||||
return initState;
|
||||
},
|
||||
handleJoin: function(id) {
|
||||
client.joinChannel(id,
|
||||
function() {
|
||||
}
|
||||
handleJoin(channel, channelIndex) {
|
||||
this.setState({joiningChannel: channelIndex});
|
||||
client.joinChannel(channel.id,
|
||||
function joinSuccess() {
|
||||
$(this.refs.modal.getDOMNode()).modal('hide');
|
||||
asyncClient.getChannel(id);
|
||||
asyncClient.getChannel(channel.id);
|
||||
utils.switchChannel(channel);
|
||||
this.setState({joiningChannel: -1});
|
||||
}.bind(this),
|
||||
function(err) {
|
||||
function joinFail(err) {
|
||||
this.setState({joiningChannel: -1});
|
||||
this.state.serverError = err.message;
|
||||
this.setState(this.state);
|
||||
}.bind(this)
|
||||
);
|
||||
},
|
||||
handleNewChannel: function() {
|
||||
}
|
||||
handleNewChannel() {
|
||||
$(this.refs.modal.getDOMNode()).modal('hide');
|
||||
},
|
||||
render: function() {
|
||||
}
|
||||
render() {
|
||||
var serverError;
|
||||
if (this.state.serverError) {
|
||||
serverError = <div className='form-group has-error'><label className='control-label'>{this.state.serverError}</label></div>;
|
||||
}
|
||||
|
||||
var outter = this;
|
||||
var self = this;
|
||||
var moreChannels;
|
||||
|
||||
if (this.state.channels != null) {
|
||||
@@ -74,14 +83,29 @@ module.exports = React.createClass({
|
||||
moreChannels = (
|
||||
<table className='more-channel-table table'>
|
||||
<tbody>
|
||||
{channels.map(function cMap(channel) {
|
||||
{channels.map(function cMap(channel, index) {
|
||||
var joinButton;
|
||||
if (self.state.joiningChannel === index) {
|
||||
joinButton = (<img
|
||||
className='join-channel-loading-gif'
|
||||
src='/static/images/load.gif'
|
||||
/>);
|
||||
} else {
|
||||
joinButton = (<button
|
||||
onClick={self.handleJoin.bind(self, channel, index)}
|
||||
className='btn btn-primary'>Join
|
||||
</button>);
|
||||
}
|
||||
|
||||
return (
|
||||
<tr key={channel.id}>
|
||||
<td>
|
||||
<p className='more-channel-name'>{channel.display_name}</p>
|
||||
<p className='more-channel-description'>{channel.description}</p>
|
||||
</td>
|
||||
<td className='td--action'><button onClick={outter.handleJoin.bind(outter, channel.id)} className='btn btn-primary'>Join</button></td>
|
||||
<td className='td--action'>
|
||||
{joinButton}
|
||||
</td>
|
||||
</tr>
|
||||
);
|
||||
})}
|
||||
@@ -102,23 +126,47 @@ module.exports = React.createClass({
|
||||
}
|
||||
|
||||
return (
|
||||
<div className='modal fade' id='more_channels' ref='modal' tabIndex='-1' role='dialog' aria-hidden='true'>
|
||||
<div
|
||||
className='modal fade'
|
||||
id='more_channels'
|
||||
ref='modal'
|
||||
tabIndex='-1'
|
||||
role='dialog'
|
||||
aria-hidden='true'
|
||||
>
|
||||
<div className='modal-dialog'>
|
||||
<div className='modal-content'>
|
||||
<div className='modal-header'>
|
||||
<button type='button' className='close' data-dismiss='modal'>
|
||||
<button
|
||||
type='button'
|
||||
className='close'
|
||||
data-dismiss='modal'
|
||||
>
|
||||
<span aria-hidden='true'>×</span>
|
||||
<span className='sr-only'>Close</span>
|
||||
</button>
|
||||
<h4 className='modal-title'>More Channels</h4>
|
||||
<button data-toggle='modal' data-target='#new_channel' data-channeltype={this.state.channelType} type='button' className='btn btn-primary channel-create-btn' onClick={this.handleNewChannel}>Create New Channel</button>
|
||||
<button
|
||||
data-toggle='modal'
|
||||
data-target='#new_channel'
|
||||
data-channeltype={this.state.channelType}
|
||||
type='button'
|
||||
className='btn btn-primary channel-create-btn'
|
||||
onClick={this.handleNewChannel}>Create New Channel
|
||||
</button>
|
||||
</div>
|
||||
<div className='modal-body'>
|
||||
{moreChannels}
|
||||
{serverError}
|
||||
</div>
|
||||
<div className='modal-footer'>
|
||||
<button type='button' className='btn btn-default' data-dismiss='modal'>Close</button>
|
||||
<button
|
||||
type='button'
|
||||
className='btn btn-default'
|
||||
data-dismiss='modal'
|
||||
>
|
||||
Close
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@@ -126,4 +174,4 @@ module.exports = React.createClass({
|
||||
|
||||
);
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
@@ -152,6 +152,8 @@ module.exports = React.createClass({
|
||||
|
||||
var channelMenuDropdown = null;
|
||||
if (channel) {
|
||||
var viewInfoOption = <li role='presentation'><a role='menuitem' data-toggle='modal' data-target='#channel_info' data-channelid={channel.id} href='#'>View Info</a></li>
|
||||
|
||||
var addMembersOption = null;
|
||||
if (!isDirect && !ChannelStore.isDefault(channel)) {
|
||||
addMembersOption = <li role='presentation'><a role='menuitem' data-toggle='modal' data-target='#channel_invite' href='#'>Add Members</a></li>;
|
||||
@@ -192,6 +194,7 @@ module.exports = React.createClass({
|
||||
<span className='glyphicon glyphicon-chevron-down header-dropdown__icon'></span>
|
||||
</a>
|
||||
<ul className='dropdown-menu' role='menu' aria-labelledby='channel_header_dropdown'>
|
||||
{viewInfoOption}
|
||||
{addMembersOption}
|
||||
{manageMembersOption}
|
||||
{setChannelDescriptionOption}
|
||||
|
||||
@@ -88,6 +88,18 @@ module.exports = React.createClass({
|
||||
<div className='file-status file-name hide'></div>
|
||||
{serverError}
|
||||
{clientError}
|
||||
<span className='btn btn-sm btn-primary btn-file sel-btn'>Select File<input ref='uploadinput' accept={this.props.fileTypesAccepted} type='file' onChange={this.onFileSelect}/></span>
|
||||
<a
|
||||
className={'btn btn-sm btn-primary'}
|
||||
onClick={this.doSubmit}>
|
||||
Import
|
||||
</a>
|
||||
<a
|
||||
className='btn btn-sm theme'
|
||||
href='#'
|
||||
onClick={this.doCancel}>
|
||||
Cancel
|
||||
</a>
|
||||
</li>
|
||||
</ul>
|
||||
</li>
|
||||
|
||||
@@ -62,11 +62,9 @@ module.exports = React.createClass({
|
||||
this.setState({confirmPassword: e.target.value});
|
||||
},
|
||||
handleHistoryOpen: function() {
|
||||
this.setState({willReturn: true});
|
||||
$("#user_settings").modal('hide');
|
||||
},
|
||||
handleDevicesOpen: function() {
|
||||
this.setState({willReturn: true});
|
||||
$("#user_settings").modal('hide');
|
||||
},
|
||||
handleClose: function() {
|
||||
@@ -75,11 +73,7 @@ module.exports = React.createClass({
|
||||
});
|
||||
this.setState({currentPassword: '', newPassword: '', confirmPassword: '', serverError: null, passwordError: null});
|
||||
|
||||
if (!this.state.willReturn) {
|
||||
this.props.updateTab('general');
|
||||
} else {
|
||||
this.setState({willReturn: false});
|
||||
}
|
||||
this.props.updateTab('general');
|
||||
},
|
||||
componentDidMount: function() {
|
||||
$('#user_settings').on('hidden.bs.modal', this.handleClose);
|
||||
@@ -89,7 +83,7 @@ module.exports = React.createClass({
|
||||
this.props.updateSection('');
|
||||
},
|
||||
getInitialState: function() {
|
||||
return {currentPassword: '', newPassword: '', confirmPassword: '', willReturn: false};
|
||||
return {currentPassword: '', newPassword: '', confirmPassword: ''};
|
||||
},
|
||||
render: function() {
|
||||
var serverError = this.state.serverError ? this.state.serverError : null;
|
||||
|
||||
@@ -124,7 +124,15 @@
|
||||
}
|
||||
|
||||
.channel-loading-gif {
|
||||
height:15px;
|
||||
width:15px;
|
||||
margin-top:2px;
|
||||
height: 15px;
|
||||
width: 15px;
|
||||
margin-top: 2px;
|
||||
}
|
||||
|
||||
.join-channel-loading-gif {
|
||||
margin-top: 5px;
|
||||
margin-left: 10px;
|
||||
height: 25px;
|
||||
width: 25px;
|
||||
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user