Files
mattermost/web/react/components/user_settings_general.jsx
2015-09-02 11:14:50 -07:00

529 lines
18 KiB
JavaScript

// Copyright (c) 2015 Spinpunch, Inc. All Rights Reserved.
// See License.txt for license information.
var UserStore = require('../stores/user_store.jsx');
var ConfigStore = require('../stores/config_store.jsx');
var SettingItemMin = require('./setting_item_min.jsx');
var SettingItemMax = require('./setting_item_max.jsx');
var SettingPicture = require('./setting_picture.jsx');
var client = require('../utils/client.jsx');
var AsyncClient = require('../utils/async_client.jsx');
var utils = require('../utils/utils.jsx');
var assign = require('object-assign');
export default class UserSettingsGeneralTab extends React.Component {
constructor(props) {
super(props);
this.submitActive = false;
this.submitUsername = this.submitUsername.bind(this);
this.submitNickname = this.submitNickname.bind(this);
this.submitName = this.submitName.bind(this);
this.submitEmail = this.submitEmail.bind(this);
this.submitUser = this.submitUser.bind(this);
this.submitPicture = this.submitPicture.bind(this);
this.updateUsername = this.updateUsername.bind(this);
this.updateFirstName = this.updateFirstName.bind(this);
this.updateLastName = this.updateLastName.bind(this);
this.updateNickname = this.updateNickname.bind(this);
this.updateEmail = this.updateEmail.bind(this);
this.updatePicture = this.updatePicture.bind(this);
this.updateSection = this.updateSection.bind(this);
this.handleClose = this.handleClose.bind(this);
this.setupInitialState = this.setupInitialState.bind(this);
this.state = this.setupInitialState(props);
}
submitUsername(e) {
e.preventDefault();
var user = this.props.user;
var username = this.state.username.trim();
var usernameError = utils.isValidUsername(username);
if (usernameError === 'Cannot use a reserved word as a username.') {
this.setState({clientError: 'This username is reserved, please choose a new one.'});
return;
} else if (usernameError) {
this.setState({clientError: "Username must begin with a letter, and contain between 3 to 15 lowercase characters made up of numbers, letters, and the symbols '.', '-' and '_'."});
return;
}
if (user.username === username) {
this.setState({clientError: 'You must submit a new username'});
return;
}
user.username = username;
this.submitUser(user);
}
submitNickname(e) {
e.preventDefault();
var user = UserStore.getCurrentUser();
var nickname = this.state.nickname.trim();
if (user.nickname === nickname) {
this.setState({clientError: 'You must submit a new nickname'});
return;
}
user.nickname = nickname;
this.submitUser(user);
}
submitName(e) {
e.preventDefault();
var user = UserStore.getCurrentUser();
var firstName = this.state.firstName.trim();
var lastName = this.state.lastName.trim();
if (user.first_name === firstName && user.last_name === lastName) {
this.setState({clientError: 'You must submit a new first or last name'});
return;
}
user.first_name = firstName;
user.last_name = lastName;
this.submitUser(user);
}
submitEmail(e) {
e.preventDefault();
var user = UserStore.getCurrentUser();
var email = this.state.email.trim().toLowerCase();
if (user.email === email) {
return;
}
if (email === '' || !utils.isEmail(email)) {
this.setState({emailError: 'Please enter a valid email address'});
return;
}
user.email = email;
this.submitUser(user);
}
submitUser(user) {
client.updateUser(user,
function updateSuccess() {
this.updateSection('');
AsyncClient.getMe();
}.bind(this),
function updateFailure(err) {
var state = this.setupInitialState(this.props);
if (err.message) {
state.serverError = err.message;
} else {
state.serverError = err;
}
this.setState(state);
}.bind(this)
);
}
submitPicture(e) {
e.preventDefault();
if (!this.state.picture) {
return;
}
if (!this.submitActive) {
return;
}
var picture = this.state.picture;
if (picture.type !== 'image/jpeg' && picture.type !== 'image/png') {
this.setState({clientError: 'Only JPG or PNG images may be used for profile pictures'});
return;
}
var formData = new FormData();
formData.append('image', picture, picture.name);
this.setState({loadingPicture: true});
client.uploadProfileImage(formData,
function imageUploadSuccess() {
this.submitActive = false;
AsyncClient.getMe();
window.location.reload();
}.bind(this),
function imageUploadFailure(err) {
var state = this.setupInitialState(this.props);
state.serverError = err;
this.setState(state);
}.bind(this)
);
}
updateUsername(e) {
this.setState({username: e.target.value});
}
updateFirstName(e) {
this.setState({firstName: e.target.value});
}
updateLastName(e) {
this.setState({lastName: e.target.value});
}
updateNickname(e) {
this.setState({nickname: e.target.value});
}
updateEmail(e) {
this.setState({email: e.target.value});
}
updatePicture(e) {
if (e.target.files && e.target.files[0]) {
this.setState({picture: e.target.files[0]});
this.submitActive = true;
this.setState({clientError: null});
} else {
this.setState({picture: null});
}
}
updateSection(section) {
this.setState(assign({}, this.setupInitialState(this.props), {clientError: '', serverError: '', emailError: ''}));
this.submitActive = false;
this.props.updateSection(section);
}
handleClose() {
$(React.findDOMNode(this)).find('.form-control').each(function clearForms() {
this.value = '';
});
this.setState(assign({}, this.setupInitialState(this.props), {clientError: null, serverError: null, emailError: null}));
this.props.updateSection('');
}
componentDidMount() {
$('#user_settings').on('hidden.bs.modal', this.handleClose);
}
componentWillUnmount() {
$('#user_settings').off('hidden.bs.modal', this.handleClose);
}
setupInitialState(props) {
var user = props.user;
var emailEnabled = !ConfigStore.getSettingAsBoolean('ByPassEmail', false);
return {username: user.username, firstName: user.first_name, lastName: user.last_name, nickname: user.nickname,
email: user.email, picture: null, loadingPicture: false, emailEnabled: emailEnabled};
}
render() {
var user = this.props.user;
var clientError = null;
if (this.state.clientError) {
clientError = this.state.clientError;
}
var serverError = null;
if (this.state.serverError) {
serverError = this.state.serverError;
}
var emailError = null;
if (this.state.emailError) {
emailError = this.state.emailError;
}
var nameSection;
var inputs = [];
if (this.props.activeSection === 'name') {
inputs.push(
<div
key='firstNameSetting'
className='form-group'
>
<label className='col-sm-5 control-label'>First Name</label>
<div className='col-sm-7'>
<input
className='form-control'
type='text'
onChange={this.updateFirstName}
value={this.state.firstName}
/>
</div>
</div>
);
inputs.push(
<div
key='lastNameSetting'
className='form-group'
>
<label className='col-sm-5 control-label'>Last Name</label>
<div className='col-sm-7'>
<input
className='form-control'
type='text'
onChange={this.updateLastName}
value={this.state.lastName}
/>
</div>
</div>
);
nameSection = (
<SettingItemMax
title='Full Name'
inputs={inputs}
submit={this.submitName}
server_error={serverError}
client_error={clientError}
updateSection={function clearSection(e) {
this.updateSection('');
e.preventDefault();
}.bind(this)}
/>
);
} else {
var fullName = '';
if (user.first_name && user.last_name) {
fullName = user.first_name + ' ' + user.last_name;
} else if (user.first_name) {
fullName = user.first_name;
} else if (user.last_name) {
fullName = user.last_name;
}
nameSection = (
<SettingItemMin
title='Full Name'
describe={fullName}
updateSection={function updateNameSection() {
this.updateSection('name');
}.bind(this)}
/>
);
}
var nicknameSection;
if (this.props.activeSection === 'nickname') {
let nicknameLabel = 'Nickname';
if (utils.isMobile()) {
nicknameLabel = '';
}
inputs.push(
<div
key='nicknameSetting'
className='form-group'
>
<label className='col-sm-5 control-label'>{nicknameLabel}</label>
<div className='col-sm-7'>
<input
className='form-control'
type='text'
onChange={this.updateNickname}
value={this.state.nickname}
/>
</div>
</div>
);
nicknameSection = (
<SettingItemMax
title='Nickname'
inputs={inputs}
submit={this.submitNickname}
server_error={serverError}
client_error={clientError}
updateSection={function clearSection(e) {
this.updateSection('');
e.preventDefault();
}.bind(this)}
/>
);
} else {
nicknameSection = (
<SettingItemMin
title='Nickname'
describe={UserStore.getCurrentUser().nickname}
updateSection={function updateNicknameSection() {
this.updateSection('nickname');
}.bind(this)}
/>
);
}
var usernameSection;
if (this.props.activeSection === 'username') {
let usernameLabel = 'Username';
if (utils.isMobile()) {
usernameLabel = '';
}
inputs.push(
<div
key='usernameSetting'
className='form-group'
>
<label className='col-sm-5 control-label'>{usernameLabel}</label>
<div className='col-sm-7'>
<input
className='form-control'
type='text'
onChange={this.updateUsername}
value={this.state.username}
/>
</div>
</div>
);
usernameSection = (
<SettingItemMax
title='Username'
inputs={inputs}
submit={this.submitUsername}
server_error={serverError}
client_error={clientError}
updateSection={function clearSection(e) {
this.updateSection('');
e.preventDefault();
}.bind(this)}
/>
);
} else {
usernameSection = (
<SettingItemMin
title='Username'
describe={UserStore.getCurrentUser().username}
updateSection={function updateUsernameSection() {
this.updateSection('username');
}.bind(this)}
/>
);
}
var emailSection;
if (this.props.activeSection === 'email') {
let helpText = <div>Email is used for notifications, and requires verification if changed.</div>;
if (!this.state.emailEnabled) {
helpText = <div className='text-danger'><br />Email has been disabled by your system administrator. No notification emails will be sent until it is enabled.</div>;
}
inputs.push(
<div key='emailSetting'>
<div className='form-group'>
<label className='col-sm-5 control-label'>Primary Email</label>
<div className='col-sm-7'>
<input
className='form-control'
type='text'
onChange={this.updateEmail}
value={this.state.email}
/>
</div>
</div>
{helpText}
</div>
);
emailSection = (
<SettingItemMax
title='Email'
inputs={inputs}
submit={this.submitEmail}
server_error={serverError}
client_error={emailError}
updateSection={function clearSection(e) {
this.updateSection('');
e.preventDefault();
}.bind(this)}
/>
);
} else {
emailSection = (
<SettingItemMin
title='Email'
describe={UserStore.getCurrentUser().email}
updateSection={function updateEmailSection() {
this.updateSection('email');
}.bind(this)}
/>
);
}
var pictureSection;
if (this.props.activeSection === 'picture') {
pictureSection = (
<SettingPicture
title='Profile Picture'
submit={this.submitPicture}
src={'/api/v1/users/' + user.id + '/image?time=' + user.last_picture_update}
server_error={serverError}
client_error={clientError}
updateSection={function clearSection(e) {
this.updateSection('');
e.preventDefault();
}.bind(this)}
picture={this.state.picture}
pictureChange={this.updatePicture}
submitActive={this.submitActive}
loadingPicture={this.state.loadingPicture}
/>
);
} else {
var minMessage = 'Click \'Edit\' to upload an image.';
if (user.last_picture_update) {
minMessage = 'Image last updated ' + utils.displayDate(user.last_picture_update);
}
pictureSection = (
<SettingItemMin
title='Profile Picture'
describe={minMessage}
updateSection={function updatePictureSection() {
this.updateSection('picture');
}.bind(this)}
/>
);
}
return (
<div>
<div className='modal-header'>
<button
type='button'
className='close'
data-dismiss='modal'
aria-label='Close'
>
<span aria-hidden='true'>&times;</span>
</button>
<h4
className='modal-title'
ref='title'
>
<i className='modal-back'></i>
General Settings
</h4>
</div>
<div className='user-settings'>
<h3 className='tab-header'>General Settings</h3>
<div className='divider-dark first'/>
{nameSection}
<div className='divider-light'/>
{usernameSection}
<div className='divider-light'/>
{nicknameSection}
<div className='divider-light'/>
{emailSection}
<div className='divider-light'/>
{pictureSection}
<div className='divider-dark'/>
</div>
</div>
);
}
}
UserSettingsGeneralTab.propTypes = {
user: React.PropTypes.object,
updateSection: React.PropTypes.func,
activeSection: React.PropTypes.string
};