mirror of
https://github.com/mattermost/mattermost.git
synced 2025-02-25 18:55:24 -06:00
Split up user settings files
This commit is contained in:
File diff suppressed because it is too large
Load Diff
118
web/react/components/user_settings_appearance.jsx
Normal file
118
web/react/components/user_settings_appearance.jsx
Normal file
@@ -0,0 +1,118 @@
|
||||
var UserStore = require('../stores/user_store.jsx');
|
||||
var SettingItemMin = require('./setting_item_min.jsx');
|
||||
var SettingItemMax = require('./setting_item_max.jsx');
|
||||
var client = require('../utils/client.jsx');
|
||||
var utils = require('../utils/utils.jsx');
|
||||
|
||||
module.exports = React.createClass({
|
||||
submitTheme: function(e) {
|
||||
e.preventDefault();
|
||||
var user = UserStore.getCurrentUser();
|
||||
if (!user.props) user.props = {};
|
||||
user.props.theme = this.state.theme;
|
||||
|
||||
client.updateUser(user,
|
||||
function(data) {
|
||||
this.props.updateSection("");
|
||||
window.location.reload();
|
||||
}.bind(this),
|
||||
function(err) {
|
||||
state = this.getInitialState();
|
||||
state.server_error = err;
|
||||
this.setState(state);
|
||||
}.bind(this)
|
||||
);
|
||||
},
|
||||
updateTheme: function(e) {
|
||||
var hex = utils.rgb2hex(e.target.style.backgroundColor);
|
||||
this.setState({ theme: hex.toLowerCase() });
|
||||
},
|
||||
handleClose: function() {
|
||||
this.setState({server_error: null});
|
||||
this.props.updateTab('general');
|
||||
},
|
||||
componentDidMount: function() {
|
||||
if (this.props.activeSection === "theme") {
|
||||
$(this.refs[this.state.theme].getDOMNode()).addClass('active-border');
|
||||
}
|
||||
$('#user_settings').on('hidden.bs.modal', this.handleClose);
|
||||
},
|
||||
componentDidUpdate: function() {
|
||||
if (this.props.activeSection === "theme") {
|
||||
$('.color-btn').removeClass('active-border');
|
||||
$(this.refs[this.state.theme].getDOMNode()).addClass('active-border');
|
||||
}
|
||||
},
|
||||
componentWillUnmount: function() {
|
||||
$('#user_settings').off('hidden.bs.modal', this.handleClose);
|
||||
this.props.updateSection('');
|
||||
},
|
||||
getInitialState: function() {
|
||||
var user = UserStore.getCurrentUser();
|
||||
var theme = config.ThemeColors != null ? config.ThemeColors[0] : "#2389d7";
|
||||
if (user.props && user.props.theme) {
|
||||
theme = user.props.theme;
|
||||
}
|
||||
return { theme: theme.toLowerCase() };
|
||||
},
|
||||
render: function() {
|
||||
var server_error = this.state.server_error ? this.state.server_error : null;
|
||||
|
||||
|
||||
var themeSection;
|
||||
var self = this;
|
||||
|
||||
if (config.ThemeColors != null) {
|
||||
if (this.props.activeSection === 'theme') {
|
||||
var theme_buttons = [];
|
||||
|
||||
for (var i = 0; i < config.ThemeColors.length; i++) {
|
||||
theme_buttons.push(<button ref={config.ThemeColors[i]} type="button" className="btn btn-lg color-btn" style={{backgroundColor: config.ThemeColors[i]}} onClick={this.updateTheme} />);
|
||||
}
|
||||
|
||||
var inputs = [];
|
||||
|
||||
inputs.push(
|
||||
<li className="setting-list-item">
|
||||
<div className="btn-group" data-toggle="buttons-radio">
|
||||
{ theme_buttons }
|
||||
</div>
|
||||
</li>
|
||||
);
|
||||
|
||||
themeSection = (
|
||||
<SettingItemMax
|
||||
title="Theme Color"
|
||||
inputs={inputs}
|
||||
submit={this.submitTheme}
|
||||
server_error={server_error}
|
||||
updateSection={function(e){self.props.updateSection("");e.preventDefault;}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
themeSection = (
|
||||
<SettingItemMin
|
||||
title="Theme Color"
|
||||
describe={this.state.theme}
|
||||
updateSection={function(){self.props.updateSection("theme");}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<div>
|
||||
<div className="modal-header">
|
||||
<button type="button" className="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">×</span></button>
|
||||
<h4 className="modal-title" ref="title"><i className="modal-back"></i>Appearance Settings</h4>
|
||||
</div>
|
||||
<div className="user-settings">
|
||||
<h3 className="tab-header">Appearance Settings</h3>
|
||||
<div className="divider-dark first"/>
|
||||
{themeSection}
|
||||
<div className="divider-dark"/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
});
|
||||
428
web/react/components/user_settings_general.jsx
Normal file
428
web/react/components/user_settings_general.jsx
Normal file
@@ -0,0 +1,428 @@
|
||||
var UserStore = require('../stores/user_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');
|
||||
|
||||
module.exports = React.createClass({
|
||||
displayName: 'GeneralTab',
|
||||
submitActive: false,
|
||||
submitUsername: function(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: function(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: function(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: function(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: function(user) {
|
||||
client.updateUser(user,
|
||||
function() {
|
||||
this.updateSection('');
|
||||
AsyncClient.getMe();
|
||||
}.bind(this),
|
||||
function(err) {
|
||||
var state = this.getInitialState();
|
||||
if (err.message) {
|
||||
state.serverError = err.message;
|
||||
} else {
|
||||
state.serverError = err;
|
||||
}
|
||||
this.setState(state);
|
||||
}.bind(this)
|
||||
);
|
||||
},
|
||||
submitPicture: function(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() {
|
||||
this.submitActive = false;
|
||||
AsyncClient.getMe();
|
||||
window.location.reload();
|
||||
}.bind(this),
|
||||
function(err) {
|
||||
var state = this.getInitialState();
|
||||
state.serverError = err;
|
||||
this.setState(state);
|
||||
}.bind(this)
|
||||
);
|
||||
},
|
||||
updateUsername: function(e) {
|
||||
this.setState({username: e.target.value});
|
||||
},
|
||||
updateFirstName: function(e) {
|
||||
this.setState({firstName: e.target.value});
|
||||
},
|
||||
updateLastName: function(e) {
|
||||
this.setState({lastName: e.target.value});
|
||||
},
|
||||
updateNickname: function(e) {
|
||||
this.setState({nickname: e.target.value});
|
||||
},
|
||||
updateEmail: function(e) {
|
||||
this.setState({email: e.target.value});
|
||||
},
|
||||
updatePicture: function(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: function(section) {
|
||||
this.setState({clientError: ''});
|
||||
this.submitActive = false;
|
||||
this.props.updateSection(section);
|
||||
},
|
||||
handleClose: function() {
|
||||
$(this.getDOMNode()).find('.form-control').each(function() {
|
||||
this.value = '';
|
||||
});
|
||||
|
||||
this.setState(assign({}, this.getInitialState(), {clientError: null, serverError: null, emailError: null}));
|
||||
this.props.updateSection('');
|
||||
},
|
||||
componentDidMount: function() {
|
||||
$('#user_settings').on('hidden.bs.modal', this.handleClose);
|
||||
},
|
||||
componentWillUnmount: function() {
|
||||
$('#user_settings').off('hidden.bs.modal', this.handleClose);
|
||||
},
|
||||
getInitialState: function() {
|
||||
var user = this.props.user;
|
||||
|
||||
return {username: user.username, firstName: user.first_name, lastName: user.last_name, nickname: user.nickname,
|
||||
email: user.email, picture: null, loadingPicture: false};
|
||||
},
|
||||
render: function() {
|
||||
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 self = this;
|
||||
var inputs = [];
|
||||
|
||||
if (this.props.activeSection === 'name') {
|
||||
inputs.push(
|
||||
<div 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 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(e) {
|
||||
self.updateSection('');
|
||||
e.preventDefault();
|
||||
}}
|
||||
/>
|
||||
);
|
||||
} 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() {
|
||||
self.updateSection('name');
|
||||
}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
var nicknameSection;
|
||||
if (this.props.activeSection === 'nickname') {
|
||||
inputs.push(
|
||||
<div className='form-group'>
|
||||
<label className='col-sm-5 control-label'>{utils.isMobile() ? '' : 'Nickname'}</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(e) {
|
||||
self.updateSection('');
|
||||
e.preventDefault();
|
||||
}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
nicknameSection = (
|
||||
<SettingItemMin
|
||||
title='Nickname'
|
||||
describe={UserStore.getCurrentUser().nickname}
|
||||
updateSection={function() {
|
||||
self.updateSection('nickname');
|
||||
}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
var usernameSection;
|
||||
if (this.props.activeSection === 'username') {
|
||||
inputs.push(
|
||||
<div className='form-group'>
|
||||
<label className='col-sm-5 control-label'>{utils.isMobile() ? '' : 'Username'}</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(e) {
|
||||
self.updateSection('');
|
||||
e.preventDefault();
|
||||
}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
usernameSection = (
|
||||
<SettingItemMin
|
||||
title='Username'
|
||||
describe={UserStore.getCurrentUser().username}
|
||||
updateSection={function() {
|
||||
self.updateSection('username');
|
||||
}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
var emailSection;
|
||||
if (this.props.activeSection === 'email') {
|
||||
inputs.push(
|
||||
<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>
|
||||
);
|
||||
|
||||
emailSection = (
|
||||
<SettingItemMax
|
||||
title='Email'
|
||||
inputs={inputs}
|
||||
submit={this.submitEmail}
|
||||
server_error={serverError}
|
||||
client_error={emailError}
|
||||
updateSection={function(e) {
|
||||
self.updateSection('');
|
||||
e.preventDefault();
|
||||
}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
emailSection = (
|
||||
<SettingItemMin
|
||||
title='Email'
|
||||
describe={UserStore.getCurrentUser().email}
|
||||
updateSection={function() {
|
||||
self.updateSection('email');
|
||||
}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
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(e) {
|
||||
self.updateSection('');
|
||||
e.preventDefault();
|
||||
}}
|
||||
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() {
|
||||
self.updateSection('picture');
|
||||
}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
return (
|
||||
<div>
|
||||
<div className='modal-header'>
|
||||
<button type='button' className='close' data-dismiss='modal' aria-label='Close'><span aria-hidden='true'>×</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>
|
||||
);
|
||||
}
|
||||
});
|
||||
457
web/react/components/user_settings_notifications.jsx
Normal file
457
web/react/components/user_settings_notifications.jsx
Normal file
@@ -0,0 +1,457 @@
|
||||
var UserStore = require('../stores/user_store.jsx');
|
||||
var SettingItemMin = require('./setting_item_min.jsx');
|
||||
var SettingItemMax = require('./setting_item_max.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');
|
||||
|
||||
function getNotificationsStateFromStores() {
|
||||
var user = UserStore.getCurrentUser();
|
||||
var soundNeeded = !utils.isBrowserFirefox();
|
||||
var sound = (!user.notify_props || user.notify_props.desktop_sound == undefined) ? "true" : user.notify_props.desktop_sound;
|
||||
var desktop = (!user.notify_props || user.notify_props.desktop == undefined) ? "all" : user.notify_props.desktop;
|
||||
var email = (!user.notify_props || user.notify_props.email == undefined) ? "true" : user.notify_props.email;
|
||||
|
||||
var username_key = false;
|
||||
var mention_key = false;
|
||||
var custom_keys = "";
|
||||
var first_name_key = false;
|
||||
var all_key = false;
|
||||
var channel_key = false;
|
||||
|
||||
if (user.notify_props) {
|
||||
if (user.notify_props.mention_keys !== undefined) {
|
||||
var keys = user.notify_props.mention_keys.split(',');
|
||||
|
||||
if (keys.indexOf(user.username) !== -1) {
|
||||
username_key = true;
|
||||
keys.splice(keys.indexOf(user.username), 1);
|
||||
} else {
|
||||
username_key = false;
|
||||
}
|
||||
|
||||
if (keys.indexOf('@'+user.username) !== -1) {
|
||||
mention_key = true;
|
||||
keys.splice(keys.indexOf('@'+user.username), 1);
|
||||
} else {
|
||||
mention_key = false;
|
||||
}
|
||||
|
||||
custom_keys = keys.join(',');
|
||||
}
|
||||
|
||||
if (user.notify_props.first_name !== undefined) {
|
||||
first_name_key = user.notify_props.first_name === "true";
|
||||
}
|
||||
|
||||
if (user.notify_props.all !== undefined) {
|
||||
all_key = user.notify_props.all === "true";
|
||||
}
|
||||
|
||||
if (user.notify_props.channel !== undefined) {
|
||||
channel_key = user.notify_props.channel === "true";
|
||||
}
|
||||
}
|
||||
|
||||
return { notify_level: desktop, enable_email: email, soundNeeded: soundNeeded, enable_sound: sound, username_key: username_key, mention_key: mention_key, custom_keys: custom_keys, custom_keys_checked: custom_keys.length > 0, first_name_key: first_name_key, all_key: all_key, channel_key: channel_key };
|
||||
}
|
||||
|
||||
|
||||
module.exports = React.createClass({
|
||||
displayName: 'NotificationsTab',
|
||||
handleSubmit: function() {
|
||||
data = {}
|
||||
data["user_id"] = this.props.user.id;
|
||||
data["email"] = this.state.enable_email;
|
||||
data["desktop_sound"] = this.state.enable_sound;
|
||||
data["desktop"] = this.state.notify_level;
|
||||
|
||||
var mention_keys = [];
|
||||
if (this.state.username_key) mention_keys.push(this.props.user.username);
|
||||
if (this.state.mention_key) mention_keys.push('@'+this.props.user.username);
|
||||
|
||||
var string_keys = mention_keys.join(',');
|
||||
if (this.state.custom_keys.length > 0 && this.state.custom_keys_checked) {
|
||||
string_keys += ',' + this.state.custom_keys;
|
||||
}
|
||||
|
||||
data["mention_keys"] = string_keys;
|
||||
data["first_name"] = this.state.first_name_key ? "true" : "false";
|
||||
data["all"] = this.state.all_key ? "true" : "false";
|
||||
data["channel"] = this.state.channel_key ? "true" : "false";
|
||||
|
||||
client.updateUserNotifyProps(data,
|
||||
function(data) {
|
||||
this.props.updateSection("");
|
||||
AsyncClient.getMe();
|
||||
}.bind(this),
|
||||
function(err) {
|
||||
this.setState({ server_error: err.message });
|
||||
}.bind(this)
|
||||
);
|
||||
},
|
||||
handleClose: function() {
|
||||
$(this.getDOMNode()).find(".form-control").each(function() {
|
||||
this.value = "";
|
||||
});
|
||||
|
||||
this.setState(assign({},getNotificationsStateFromStores(),{server_error: null}));
|
||||
|
||||
this.props.updateTab('general');
|
||||
},
|
||||
componentDidMount: function() {
|
||||
UserStore.addChangeListener(this._onChange);
|
||||
$('#user_settings').on('hidden.bs.modal', this.handleClose);
|
||||
},
|
||||
componentWillUnmount: function() {
|
||||
UserStore.removeChangeListener(this._onChange);
|
||||
$('#user_settings').off('hidden.bs.modal', this.handleClose);
|
||||
this.props.updateSection('');
|
||||
},
|
||||
_onChange: function() {
|
||||
var newState = getNotificationsStateFromStores();
|
||||
if (!utils.areStatesEqual(newState, this.state)) {
|
||||
this.setState(newState);
|
||||
}
|
||||
},
|
||||
getInitialState: function() {
|
||||
return getNotificationsStateFromStores();
|
||||
},
|
||||
handleNotifyRadio: function(notifyLevel) {
|
||||
this.setState({ notify_level: notifyLevel });
|
||||
this.refs.wrapper.getDOMNode().focus();
|
||||
},
|
||||
handleEmailRadio: function(enableEmail) {
|
||||
this.setState({ enable_email: enableEmail });
|
||||
this.refs.wrapper.getDOMNode().focus();
|
||||
},
|
||||
handleSoundRadio: function(enableSound) {
|
||||
this.setState({ enable_sound: enableSound });
|
||||
this.refs.wrapper.getDOMNode().focus();
|
||||
},
|
||||
updateUsernameKey: function(val) {
|
||||
this.setState({ username_key: val });
|
||||
},
|
||||
updateMentionKey: function(val) {
|
||||
this.setState({ mention_key: val });
|
||||
},
|
||||
updateFirstNameKey: function(val) {
|
||||
this.setState({ first_name_key: val });
|
||||
},
|
||||
updateAllKey: function(val) {
|
||||
this.setState({ all_key: val });
|
||||
},
|
||||
updateChannelKey: function(val) {
|
||||
this.setState({ channel_key: val });
|
||||
},
|
||||
updateCustomMentionKeys: function() {
|
||||
var checked = this.refs.customcheck.getDOMNode().checked;
|
||||
|
||||
if (checked) {
|
||||
var text = this.refs.custommentions.getDOMNode().value;
|
||||
|
||||
// remove all spaces and split string into individual keys
|
||||
this.setState({ custom_keys: text.replace(/ /g, ''), custom_keys_checked: true });
|
||||
} else {
|
||||
this.setState({ custom_keys: "", custom_keys_checked: false });
|
||||
}
|
||||
},
|
||||
onCustomChange: function() {
|
||||
this.refs.customcheck.getDOMNode().checked = true;
|
||||
this.updateCustomMentionKeys();
|
||||
},
|
||||
render: function() {
|
||||
var server_error = this.state.server_error ? this.state.server_error : null;
|
||||
|
||||
var self = this;
|
||||
|
||||
var user = this.props.user;
|
||||
|
||||
var desktopSection;
|
||||
if (this.props.activeSection === 'desktop') {
|
||||
var notifyActive = [false, false, false];
|
||||
if (this.state.notify_level === "mention") {
|
||||
notifyActive[1] = true;
|
||||
} else if (this.state.notify_level === "none") {
|
||||
notifyActive[2] = true;
|
||||
} else {
|
||||
notifyActive[0] = true;
|
||||
}
|
||||
|
||||
var inputs = [];
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="radio">
|
||||
<label>
|
||||
<input type="radio" checked={notifyActive[0]} onClick={function(){self.handleNotifyRadio("all")}}>For all activity</input>
|
||||
</label>
|
||||
<br/>
|
||||
</div>
|
||||
<div className="radio">
|
||||
<label>
|
||||
<input type="radio" checked={notifyActive[1]} onClick={function(){self.handleNotifyRadio("mention")}}>Only for mentions and private messages</input>
|
||||
</label>
|
||||
<br/>
|
||||
</div>
|
||||
<div className="radio">
|
||||
<label>
|
||||
<input type="radio" checked={notifyActive[2]} onClick={function(){self.handleNotifyRadio("none")}}>Never</input>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
desktopSection = (
|
||||
<SettingItemMax
|
||||
title="Send desktop notifications"
|
||||
inputs={inputs}
|
||||
submit={this.handleSubmit}
|
||||
server_error={server_error}
|
||||
updateSection={function(e){self.props.updateSection("");e.preventDefault();}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
var describe = "";
|
||||
if (this.state.notify_level === "mention") {
|
||||
describe = "Only for mentions and private messages";
|
||||
} else if (this.state.notify_level === "none") {
|
||||
describe = "Never";
|
||||
} else {
|
||||
describe = "For all activity";
|
||||
}
|
||||
|
||||
desktopSection = (
|
||||
<SettingItemMin
|
||||
title="Send desktop notifications"
|
||||
describe={describe}
|
||||
updateSection={function(){self.props.updateSection("desktop");}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
var soundSection;
|
||||
if (this.props.activeSection === 'sound' && this.state.soundNeeded) {
|
||||
var soundActive = ["",""];
|
||||
if (this.state.enable_sound === "false") {
|
||||
soundActive[1] = "active";
|
||||
} else {
|
||||
soundActive[0] = "active";
|
||||
}
|
||||
|
||||
var inputs = [];
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="btn-group" data-toggle="buttons-radio">
|
||||
<button className={"btn btn-default "+soundActive[0]} onClick={function(){self.handleSoundRadio("true")}}>On</button>
|
||||
<button className={"btn btn-default "+soundActive[1]} onClick={function(){self.handleSoundRadio("false")}}>Off</button>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
soundSection = (
|
||||
<SettingItemMax
|
||||
title="Desktop notification sounds"
|
||||
inputs={inputs}
|
||||
submit={this.handleSubmit}
|
||||
server_error={server_error}
|
||||
updateSection={function(e){self.props.updateSection("");e.preventDefault();}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
var describe = "";
|
||||
if (!this.state.soundNeeded) {
|
||||
describe = "Please configure notification sounds in your browser settings"
|
||||
} else if (this.state.enable_sound === "false") {
|
||||
describe = "Off";
|
||||
} else {
|
||||
describe = "On";
|
||||
}
|
||||
|
||||
soundSection = (
|
||||
<SettingItemMin
|
||||
title="Desktop notification sounds"
|
||||
describe={describe}
|
||||
updateSection={function(){self.props.updateSection("sound");}}
|
||||
disableOpen = {!this.state.soundNeeded}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
var emailSection;
|
||||
if (this.props.activeSection === 'email') {
|
||||
var emailActive = ["",""];
|
||||
if (this.state.enable_email === "false") {
|
||||
emailActive[1] = "active";
|
||||
} else {
|
||||
emailActive[0] = "active";
|
||||
}
|
||||
|
||||
var inputs = [];
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="btn-group" data-toggle="buttons-radio">
|
||||
<button className={"btn btn-default "+emailActive[0]} onClick={function(){self.handleEmailRadio("true")}}>On</button>
|
||||
<button className={"btn btn-default "+emailActive[1]} onClick={function(){self.handleEmailRadio("false")}}>Off</button>
|
||||
</div>
|
||||
<div><br/>{"Email notifications are sent for mentions and private messages after you have been away from " + config.SiteName + " for 5 minutes."}</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
emailSection = (
|
||||
<SettingItemMax
|
||||
title="Email notifications"
|
||||
inputs={inputs}
|
||||
submit={this.handleSubmit}
|
||||
server_error={server_error}
|
||||
updateSection={function(e){self.props.updateSection("");e.preventDefault();}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
var describe = "";
|
||||
if (this.state.enable_email === "false") {
|
||||
describe = "Off";
|
||||
} else {
|
||||
describe = "On";
|
||||
}
|
||||
|
||||
emailSection = (
|
||||
<SettingItemMin
|
||||
title="Email notifications"
|
||||
describe={describe}
|
||||
updateSection={function(){self.props.updateSection("email");}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
var keysSection;
|
||||
if (this.props.activeSection === 'keys') {
|
||||
var inputs = [];
|
||||
|
||||
if (user.first_name) {
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="checkbox">
|
||||
<label>
|
||||
<input type="checkbox" checked={this.state.first_name_key} onChange={function(e){self.updateFirstNameKey(e.target.checked);}}>{'Your case sensitive first name "' + user.first_name + '"'}</input>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="checkbox">
|
||||
<label>
|
||||
<input type="checkbox" checked={this.state.username_key} onChange={function(e){self.updateUsernameKey(e.target.checked);}}>{'Your non-case sensitive username "' + user.username + '"'}</input>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="checkbox">
|
||||
<label>
|
||||
<input type="checkbox" checked={this.state.mention_key} onChange={function(e){self.updateMentionKey(e.target.checked);}}>{'Your username mentioned "@' + user.username + '"'}</input>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="checkbox">
|
||||
<label>
|
||||
<input type="checkbox" checked={this.state.all_key} onChange={function(e){self.updateAllKey(e.target.checked);}}>{'Team-wide mentions "@all"'}</input>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="checkbox">
|
||||
<label>
|
||||
<input type="checkbox" checked={this.state.channel_key} onChange={function(e){self.updateChannelKey(e.target.checked);}}>{'Channel-wide mentions "@channel"'}</input>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
inputs.push(
|
||||
<div>
|
||||
<div className="checkbox">
|
||||
<label>
|
||||
<input ref="customcheck" type="checkbox" checked={this.state.custom_keys_checked} onChange={this.updateCustomMentionKeys}>{'Other non-case sensitive words, separated by commas:'}</input>
|
||||
</label>
|
||||
</div>
|
||||
<input ref="custommentions" className="form-control mentions-input" type="text" defaultValue={this.state.custom_keys} onChange={this.onCustomChange} />
|
||||
</div>
|
||||
);
|
||||
|
||||
keysSection = (
|
||||
<SettingItemMax
|
||||
title="Words that trigger mentions"
|
||||
inputs={inputs}
|
||||
submit={this.handleSubmit}
|
||||
server_error={server_error}
|
||||
updateSection={function(e){self.props.updateSection("");e.preventDefault();}}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
var keys = [];
|
||||
if (this.state.first_name_key) keys.push(user.first_name);
|
||||
if (this.state.username_key) keys.push(user.username);
|
||||
if (this.state.mention_key) keys.push('@'+user.username);
|
||||
if (this.state.all_key) keys.push('@all');
|
||||
if (this.state.channel_key) keys.push('@channel');
|
||||
if (this.state.custom_keys.length > 0) keys = keys.concat(this.state.custom_keys.split(','));
|
||||
|
||||
var describe = "";
|
||||
for (var i = 0; i < keys.length; i++) {
|
||||
describe += '"' + keys[i] + '", ';
|
||||
}
|
||||
|
||||
if (describe.length > 0) {
|
||||
describe = describe.substring(0, describe.length - 2);
|
||||
} else {
|
||||
describe = "No words configured";
|
||||
}
|
||||
|
||||
keysSection = (
|
||||
<SettingItemMin
|
||||
title="Words that trigger mentions"
|
||||
describe={describe}
|
||||
updateSection={function(){self.props.updateSection("keys");}}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div>
|
||||
<div className="modal-header">
|
||||
<button type="button" className="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">×</span></button>
|
||||
<h4 className="modal-title" ref="title"><i className="modal-back"></i>Notifications</h4>
|
||||
</div>
|
||||
<div ref="wrapper" className="user-settings">
|
||||
<h3 className="tab-header">Notifications</h3>
|
||||
<div className="divider-dark first"/>
|
||||
{desktopSection}
|
||||
<div className="divider-light"/>
|
||||
{soundSection}
|
||||
<div className="divider-light"/>
|
||||
{emailSection}
|
||||
<div className="divider-light"/>
|
||||
{keysSection}
|
||||
<div className="divider-dark"/>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
);
|
||||
}
|
||||
});
|
||||
200
web/react/components/user_settings_security.jsx
Normal file
200
web/react/components/user_settings_security.jsx
Normal file
@@ -0,0 +1,200 @@
|
||||
var SettingItemMin = require('./setting_item_min.jsx');
|
||||
var SettingItemMax = require('./setting_item_max.jsx');
|
||||
var client = require('../utils/client.jsx');
|
||||
var AsyncClient = require('../utils/async_client.jsx');
|
||||
var Constants = require('../utils/constants.jsx');
|
||||
|
||||
module.exports = React.createClass({
|
||||
displayName: 'SecurityTab',
|
||||
submitPassword: function(e) {
|
||||
e.preventDefault();
|
||||
|
||||
var user = this.props.user;
|
||||
var currentPassword = this.state.currentPassword;
|
||||
var newPassword = this.state.newPassword;
|
||||
var confirmPassword = this.state.confirmPassword;
|
||||
|
||||
if (currentPassword === '') {
|
||||
this.setState({passwordError: 'Please enter your current password', serverError: ''});
|
||||
return;
|
||||
}
|
||||
|
||||
if (newPassword.length < 5) {
|
||||
this.setState({passwordError: 'New passwords must be at least 5 characters', serverError: ''});
|
||||
return;
|
||||
}
|
||||
|
||||
if (newPassword !== confirmPassword) {
|
||||
this.setState({passwordError: 'The new passwords you entered do not match', serverError: ''});
|
||||
return;
|
||||
}
|
||||
|
||||
var data = {};
|
||||
data.user_id = user.id;
|
||||
data.current_password = currentPassword;
|
||||
data.new_password = newPassword;
|
||||
|
||||
client.updatePassword(data,
|
||||
function() {
|
||||
this.props.updateSection('');
|
||||
AsyncClient.getMe();
|
||||
this.setState({currentPassword: '', newPassword: '', confirmPassword: ''});
|
||||
}.bind(this),
|
||||
function(err) {
|
||||
var state = this.getInitialState();
|
||||
if (err.message) {
|
||||
state.serverError = err.message;
|
||||
} else {
|
||||
state.serverError = err;
|
||||
}
|
||||
state.passwordError = '';
|
||||
this.setState(state);
|
||||
}.bind(this)
|
||||
);
|
||||
},
|
||||
updateCurrentPassword: function(e) {
|
||||
this.setState({currentPassword: e.target.value});
|
||||
},
|
||||
updateNewPassword: function(e) {
|
||||
this.setState({newPassword: e.target.value});
|
||||
},
|
||||
updateConfirmPassword: function(e) {
|
||||
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() {
|
||||
$(this.getDOMNode()).find('.form-control').each(function() {
|
||||
this.value = '';
|
||||
});
|
||||
this.setState({currentPassword: '', newPassword: '', confirmPassword: '', serverError: null, passwordError: null});
|
||||
|
||||
if (!this.state.willReturn) {
|
||||
this.props.updateTab('general');
|
||||
} else {
|
||||
this.setState({willReturn: false});
|
||||
}
|
||||
},
|
||||
componentDidMount: function() {
|
||||
$('#user_settings').on('hidden.bs.modal', this.handleClose);
|
||||
},
|
||||
componentWillUnmount: function() {
|
||||
$('#user_settings').off('hidden.bs.modal', this.handleClose);
|
||||
this.props.updateSection('');
|
||||
},
|
||||
getInitialState: function() {
|
||||
return {currentPassword: '', newPassword: '', confirmPassword: '', willReturn: false};
|
||||
},
|
||||
render: function() {
|
||||
var serverError = this.state.serverError ? this.state.serverError : null;
|
||||
var passwordError = this.state.passwordError ? this.state.passwordError : null;
|
||||
|
||||
var updateSectionStatus;
|
||||
var passwordSection;
|
||||
var self = this;
|
||||
if (this.props.activeSection === 'password') {
|
||||
var inputs = [];
|
||||
var submit = null;
|
||||
|
||||
if (this.props.user.auth_service === '') {
|
||||
inputs.push(
|
||||
<div className='form-group'>
|
||||
<label className='col-sm-5 control-label'>Current Password</label>
|
||||
<div className='col-sm-7'>
|
||||
<input className='form-control' type='password' onChange={this.updateCurrentPassword} value={this.state.currentPassword}/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
inputs.push(
|
||||
<div className='form-group'>
|
||||
<label className='col-sm-5 control-label'>New Password</label>
|
||||
<div className='col-sm-7'>
|
||||
<input className='form-control' type='password' onChange={this.updateNewPassword} value={this.state.newPassword}/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
inputs.push(
|
||||
<div className='form-group'>
|
||||
<label className='col-sm-5 control-label'>Retype New Password</label>
|
||||
<div className='col-sm-7'>
|
||||
<input className='form-control' type='password' onChange={this.updateConfirmPassword} value={this.state.confirmPassword}/>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
|
||||
submit = this.submitPassword;
|
||||
} else {
|
||||
inputs.push(
|
||||
<div className='form-group'>
|
||||
<label className='col-sm-12'>Log in occurs through GitLab. Please see your GitLab account settings page to update your password.</label>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
updateSectionStatus = function(e) {
|
||||
self.props.updateSection('');
|
||||
self.setState({currentPassword: '', newPassword: '', confirmPassword: '', serverError: null, passwordError: null});
|
||||
e.preventDefault();
|
||||
};
|
||||
|
||||
passwordSection = (
|
||||
<SettingItemMax
|
||||
title='Password'
|
||||
inputs={inputs}
|
||||
submit={submit}
|
||||
server_error={serverError}
|
||||
client_error={passwordError}
|
||||
updateSection={updateSectionStatus}
|
||||
/>
|
||||
);
|
||||
} else {
|
||||
var describe;
|
||||
if (this.props.user.auth_service === '') {
|
||||
var d = new Date(this.props.user.last_password_update);
|
||||
var hour = d.getHours() % 12 ? String(d.getHours() % 12) : '12';
|
||||
var min = d.getMinutes() < 10 ? '0' + d.getMinutes() : String(d.getMinutes());
|
||||
var timeOfDay = d.getHours() >= 12 ? ' pm' : ' am';
|
||||
describe = 'Last updated ' + Constants.MONTHS[d.getMonth()] + ' ' + d.getDate() + ', ' + d.getFullYear() + ' at ' + hour + ':' + min + timeOfDay;
|
||||
} else {
|
||||
describe = 'Log in done through GitLab';
|
||||
}
|
||||
|
||||
updateSectionStatus = function() {
|
||||
self.props.updateSection('password');
|
||||
};
|
||||
|
||||
passwordSection = (
|
||||
<SettingItemMin
|
||||
title='Password'
|
||||
describe={describe}
|
||||
updateSection={updateSectionStatus}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div>
|
||||
<div className='modal-header'>
|
||||
<button type='button' className='close' data-dismiss='modal' aria-label='Close'><span aria-hidden='true'>×</span></button>
|
||||
<h4 className='modal-title' ref='title'><i className='modal-back'></i>Security Settings</h4>
|
||||
</div>
|
||||
<div className='user-settings'>
|
||||
<h3 className='tab-header'>Security Settings</h3>
|
||||
<div className='divider-dark first'/>
|
||||
{passwordSection}
|
||||
<div className='divider-dark'/>
|
||||
<br></br>
|
||||
<a data-toggle='modal' className='security-links theme' data-target='#access-history' href='#' onClick={this.handleHistoryOpen}><i className='fa fa-clock-o'></i>View Access History</a>
|
||||
<b> </b>
|
||||
<a data-toggle='modal' className='security-links theme' data-target='#activity-log' href='#' onClick={this.handleDevicesOpen}><i className='fa fa-globe'></i>View and Logout of Active Sessions</a>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
});
|
||||
Reference in New Issue
Block a user