519 lines
14 KiB
JavaScript
519 lines
14 KiB
JavaScript
import React from 'react';
|
|
import PropTypes from 'prop-types';
|
|
import {
|
|
View, ScrollView, Keyboard, Dimensions, BackHandler
|
|
} from 'react-native';
|
|
import { connect } from 'react-redux';
|
|
import Dialog from 'react-native-dialog';
|
|
import SHA256 from 'js-sha256';
|
|
import Icon from 'react-native-vector-icons/MaterialIcons';
|
|
import ImagePicker from 'react-native-image-crop-picker';
|
|
import RNPickerSelect from 'react-native-picker-select';
|
|
import { Navigation } from 'react-native-navigation';
|
|
import SafeAreaView from 'react-native-safe-area-view';
|
|
|
|
import LoggedView from '../View';
|
|
import KeyboardView from '../../presentation/KeyboardView';
|
|
import sharedStyles from '../Styles';
|
|
import styles from './styles';
|
|
import scrollPersistTaps from '../../utils/scrollPersistTaps';
|
|
import { showErrorAlert, showToast } from '../../utils/info';
|
|
import RocketChat from '../../lib/rocketchat';
|
|
import RCTextInput from '../../containers/TextInput';
|
|
import log from '../../utils/log';
|
|
import I18n from '../../i18n';
|
|
import Button from '../../containers/Button';
|
|
import Avatar from '../../containers/Avatar';
|
|
import Touch from '../../utils/touch';
|
|
import Drawer from '../../Drawer';
|
|
import { DEFAULT_HEADER } from '../../constants/headerOptions';
|
|
import { appStart as appStartAction } from '../../actions';
|
|
import { setUser as setUserAction } from '../../actions/login';
|
|
|
|
@connect(state => ({
|
|
user: {
|
|
id: state.login.user && state.login.user.id,
|
|
name: state.login.user && state.login.user.name,
|
|
username: state.login.user && state.login.user.username,
|
|
customFields: state.login.user && state.login.user.customFields,
|
|
emails: state.login.user && state.login.user.emails
|
|
},
|
|
Accounts_CustomFields: state.settings.Accounts_CustomFields,
|
|
baseUrl: state.settings.Site_Url || state.server ? state.server.server : ''
|
|
}), dispatch => ({
|
|
appStart: () => dispatch(appStartAction()),
|
|
setUser: params => dispatch(setUserAction(params))
|
|
}))
|
|
/** @extends React.Component */
|
|
export default class ProfileView extends LoggedView {
|
|
static options() {
|
|
return {
|
|
...DEFAULT_HEADER,
|
|
topBar: {
|
|
...DEFAULT_HEADER.topBar,
|
|
leftButtons: [{
|
|
id: 'settings',
|
|
icon: { uri: 'settings', scale: Dimensions.get('window').scale },
|
|
testID: 'rooms-list-view-sidebar'
|
|
}],
|
|
title: {
|
|
...DEFAULT_HEADER.topBar.title,
|
|
text: I18n.t('Profile')
|
|
}
|
|
},
|
|
sideMenu: {
|
|
left: {
|
|
enabled: true
|
|
},
|
|
right: {
|
|
enabled: true
|
|
}
|
|
}
|
|
};
|
|
}
|
|
|
|
static propTypes = {
|
|
baseUrl: PropTypes.string,
|
|
componentId: PropTypes.string,
|
|
user: PropTypes.object,
|
|
Accounts_CustomFields: PropTypes.string,
|
|
appStart: PropTypes.func,
|
|
setUser: PropTypes.func
|
|
}
|
|
|
|
constructor(props) {
|
|
super('ProfileView', props);
|
|
this.state = {
|
|
showPasswordAlert: false,
|
|
saving: false,
|
|
name: null,
|
|
username: null,
|
|
email: null,
|
|
newPassword: null,
|
|
currentPassword: null,
|
|
avatarUrl: null,
|
|
avatar: {},
|
|
avatarSuggestions: {},
|
|
customFields: {}
|
|
};
|
|
Navigation.events().bindComponent(this);
|
|
BackHandler.addEventListener('hardwareBackPress', this.handleBackPress);
|
|
}
|
|
|
|
async componentDidMount() {
|
|
this.init();
|
|
|
|
try {
|
|
const result = await RocketChat.getAvatarSuggestion();
|
|
this.setState({ avatarSuggestions: result });
|
|
} catch (e) {
|
|
log('getAvatarSuggestion', e);
|
|
}
|
|
}
|
|
|
|
componentWillReceiveProps(nextProps) {
|
|
const { user } = this.props;
|
|
if (user !== nextProps.user) {
|
|
this.init(nextProps.user);
|
|
}
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
BackHandler.removeEventListener('hardwareBackPress', this.handleBackPress);
|
|
}
|
|
|
|
navigationButtonPressed = ({ buttonId }) => {
|
|
if (buttonId === 'settings') {
|
|
Drawer.toggle();
|
|
}
|
|
}
|
|
|
|
handleBackPress = () => {
|
|
const { appStart } = this.props;
|
|
appStart('background');
|
|
return false;
|
|
}
|
|
|
|
setAvatar = (avatar) => {
|
|
this.setState({ avatar });
|
|
}
|
|
|
|
init = (user) => {
|
|
const { user: userProps } = this.props;
|
|
const {
|
|
name, username, emails, customFields
|
|
} = user || userProps;
|
|
|
|
this.setState({
|
|
name,
|
|
username,
|
|
email: emails ? emails[0].address : null,
|
|
newPassword: null,
|
|
currentPassword: null,
|
|
avatarUrl: null,
|
|
avatar: {},
|
|
customFields: customFields || {}
|
|
});
|
|
}
|
|
|
|
formIsChanged = () => {
|
|
const {
|
|
name, username, email, newPassword, avatar, customFields
|
|
} = this.state;
|
|
const { user } = this.props;
|
|
let customFieldsChanged = false;
|
|
|
|
const customFieldsKeys = Object.keys(customFields);
|
|
if (customFieldsKeys.length) {
|
|
customFieldsKeys.forEach((key) => {
|
|
if (!user.customFields || user.customFields[key] !== customFields[key]) {
|
|
customFieldsChanged = true;
|
|
}
|
|
});
|
|
}
|
|
|
|
return !(user.name === name
|
|
&& user.username === username
|
|
&& !newPassword
|
|
&& (user.emails && user.emails[0].address === email)
|
|
&& !avatar.data
|
|
&& !customFieldsChanged
|
|
);
|
|
}
|
|
|
|
closePasswordAlert = () => {
|
|
this.setState({ showPasswordAlert: false });
|
|
}
|
|
|
|
handleError = (e, func, action) => {
|
|
if (e.data && e.data.errorType === 'error-too-many-requests') {
|
|
return showErrorAlert(e.data.error);
|
|
}
|
|
showErrorAlert(I18n.t('There_was_an_error_while_action', { action: I18n.t(action) }));
|
|
log(func, e);
|
|
}
|
|
|
|
submit = async() => {
|
|
Keyboard.dismiss();
|
|
|
|
if (!this.formIsChanged()) {
|
|
return;
|
|
}
|
|
|
|
this.setState({ saving: true, showPasswordAlert: false });
|
|
|
|
const {
|
|
name, username, email, newPassword, currentPassword, avatar, customFields
|
|
} = this.state;
|
|
const { user, setUser } = this.props;
|
|
const params = {};
|
|
|
|
// Name
|
|
if (user.name !== name) {
|
|
params.name = name;
|
|
}
|
|
|
|
// Username
|
|
if (user.username !== username) {
|
|
params.username = username;
|
|
}
|
|
|
|
// Email
|
|
if (user.emails && user.emails[0].address !== email) {
|
|
params.email = email;
|
|
}
|
|
|
|
// newPassword
|
|
if (newPassword) {
|
|
params.newPassword = newPassword;
|
|
}
|
|
|
|
// currentPassword
|
|
if (currentPassword) {
|
|
params.currentPassword = SHA256(currentPassword);
|
|
}
|
|
|
|
const requirePassword = !!params.email || newPassword;
|
|
if (requirePassword && !params.currentPassword) {
|
|
return this.setState({ showPasswordAlert: true, saving: false });
|
|
}
|
|
|
|
try {
|
|
if (avatar.url) {
|
|
try {
|
|
await RocketChat.setAvatarFromService(avatar);
|
|
} catch (e) {
|
|
this.setState({ saving: false, currentPassword: null });
|
|
return this.handleError(e, 'setAvatarFromService', 'changing_avatar');
|
|
}
|
|
}
|
|
|
|
params.customFields = customFields;
|
|
|
|
const result = await RocketChat.saveUserProfile(params);
|
|
if (result.success) {
|
|
if (params.customFields) {
|
|
setUser({ customFields });
|
|
}
|
|
this.setState({ saving: false });
|
|
showToast(I18n.t('Profile_saved_successfully'));
|
|
this.init();
|
|
}
|
|
} catch (e) {
|
|
this.setState({ saving: false, currentPassword: null });
|
|
this.handleError(e, 'saveUserProfile', 'saving_profile');
|
|
}
|
|
}
|
|
|
|
resetAvatar = async() => {
|
|
try {
|
|
const { user } = this.props;
|
|
await RocketChat.resetAvatar(user.id);
|
|
showToast(I18n.t('Avatar_changed_successfully'));
|
|
this.init();
|
|
} catch (e) {
|
|
this.handleError(e, 'resetAvatar', 'changing_avatar');
|
|
}
|
|
}
|
|
|
|
pickImage = async() => {
|
|
const options = {
|
|
cropping: true,
|
|
compressImageQuality: 0.8,
|
|
cropperAvoidEmptySpaceAroundImage: false,
|
|
cropperChooseText: I18n.t('Choose'),
|
|
cropperCancelText: I18n.t('Cancel'),
|
|
includeBase64: true
|
|
};
|
|
try {
|
|
const response = await ImagePicker.openPicker(options);
|
|
this.setAvatar({ url: response.path, data: `data:image/jpeg;base64,${ response.data }`, service: 'upload' });
|
|
} catch (error) {
|
|
console.warn(error);
|
|
}
|
|
}
|
|
|
|
renderAvatarButton = ({
|
|
key, child, onPress, disabled = false
|
|
}) => (
|
|
<Touch
|
|
key={key}
|
|
testID={key}
|
|
onPress={onPress}
|
|
underlayColor='rgba(255, 255, 255, 0.5)'
|
|
activeOpacity={0.3}
|
|
disabled={disabled}
|
|
>
|
|
<View
|
|
style={[styles.avatarButton, { opacity: disabled ? 0.5 : 1 }]}
|
|
>
|
|
{child}
|
|
</View>
|
|
</Touch>
|
|
)
|
|
|
|
renderAvatarButtons = () => {
|
|
const { avatarUrl, avatarSuggestions } = this.state;
|
|
const { user, baseUrl } = this.props;
|
|
|
|
return (
|
|
<View style={styles.avatarButtons}>
|
|
{this.renderAvatarButton({
|
|
child: <Avatar text={`@${ user.username }`} size={50} baseUrl={baseUrl} />,
|
|
onPress: () => this.resetAvatar(),
|
|
key: 'profile-view-reset-avatar'
|
|
})}
|
|
{this.renderAvatarButton({
|
|
child: <Icon name='file-upload' size={30} />,
|
|
onPress: () => this.pickImage(),
|
|
key: 'profile-view-upload-avatar'
|
|
})}
|
|
{this.renderAvatarButton({
|
|
child: <Icon name='link' size={30} />,
|
|
onPress: () => this.setAvatar({ url: avatarUrl, data: avatarUrl, service: 'url' }),
|
|
disabled: !avatarUrl,
|
|
key: 'profile-view-avatar-url-button'
|
|
})}
|
|
{Object.keys(avatarSuggestions).map((service) => {
|
|
const { url, blob, contentType } = avatarSuggestions[service];
|
|
return this.renderAvatarButton({
|
|
key: `profile-view-avatar-${ service }`,
|
|
child: <Avatar avatar={url} size={50} baseUrl={baseUrl} />,
|
|
onPress: () => this.setAvatar({
|
|
url, data: blob, service, contentType
|
|
})
|
|
});
|
|
})}
|
|
</View>
|
|
);
|
|
}
|
|
|
|
renderCustomFields = () => {
|
|
const { customFields } = this.state;
|
|
const { Accounts_CustomFields } = this.props;
|
|
|
|
if (!Accounts_CustomFields) {
|
|
return null;
|
|
}
|
|
try {
|
|
const parsedCustomFields = JSON.parse(Accounts_CustomFields);
|
|
return Object.keys(parsedCustomFields).map((key, index, array) => {
|
|
if (parsedCustomFields[key].type === 'select') {
|
|
const options = parsedCustomFields[key].options.map(option => ({ label: option, value: option }));
|
|
return (
|
|
<RNPickerSelect
|
|
key={key}
|
|
items={options}
|
|
onValueChange={(value) => {
|
|
const newValue = {};
|
|
newValue[key] = value;
|
|
this.setState({ customFields: { ...customFields, ...newValue } });
|
|
}}
|
|
value={customFields[key]}
|
|
>
|
|
<RCTextInput
|
|
inputRef={(e) => { this[key] = e; }}
|
|
label={key}
|
|
placeholder={key}
|
|
value={customFields[key]}
|
|
testID='settings-view-language'
|
|
/>
|
|
</RNPickerSelect>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<RCTextInput
|
|
inputRef={(e) => { this[key] = e; }}
|
|
key={key}
|
|
label={key}
|
|
placeholder={key}
|
|
value={customFields[key]}
|
|
onChangeText={(value) => {
|
|
const newValue = {};
|
|
newValue[key] = value;
|
|
this.setState({ customFields: { ...customFields, ...newValue } });
|
|
}}
|
|
onSubmitEditing={() => {
|
|
if (array.length - 1 > index) {
|
|
return this[array[index + 1]].focus();
|
|
}
|
|
this.avatarUrl.focus();
|
|
}}
|
|
/>
|
|
);
|
|
});
|
|
} catch (error) {
|
|
return null;
|
|
}
|
|
}
|
|
|
|
render() {
|
|
const {
|
|
name, username, email, newPassword, avatarUrl, customFields, avatar, saving, showPasswordAlert
|
|
} = this.state;
|
|
const { baseUrl } = this.props;
|
|
|
|
return (
|
|
<KeyboardView
|
|
contentContainerStyle={sharedStyles.container}
|
|
keyboardVerticalOffset={128}
|
|
>
|
|
<ScrollView
|
|
contentContainerStyle={sharedStyles.containerScrollView}
|
|
testID='profile-view-list'
|
|
{...scrollPersistTaps}
|
|
>
|
|
<SafeAreaView style={sharedStyles.container} testID='profile-view' forceInset={{ bottom: 'never' }}>
|
|
<View style={styles.avatarContainer} testID='profile-view-avatar'>
|
|
<Avatar
|
|
text={username}
|
|
avatar={avatar && avatar.url}
|
|
size={100}
|
|
baseUrl={baseUrl}
|
|
/>
|
|
</View>
|
|
<RCTextInput
|
|
inputRef={(e) => { this.name = e; }}
|
|
label={I18n.t('Name')}
|
|
placeholder={I18n.t('Name')}
|
|
value={name}
|
|
onChangeText={value => this.setState({ name: value })}
|
|
onSubmitEditing={() => { this.username.focus(); }}
|
|
testID='profile-view-name'
|
|
/>
|
|
<RCTextInput
|
|
inputRef={(e) => { this.username = e; }}
|
|
label={I18n.t('Username')}
|
|
placeholder={I18n.t('Username')}
|
|
value={username}
|
|
onChangeText={value => this.setState({ username: value })}
|
|
onSubmitEditing={() => { this.email.focus(); }}
|
|
testID='profile-view-username'
|
|
/>
|
|
<RCTextInput
|
|
inputRef={(e) => { this.email = e; }}
|
|
label={I18n.t('Email')}
|
|
placeholder={I18n.t('Email')}
|
|
value={email}
|
|
onChangeText={value => this.setState({ email: value })}
|
|
onSubmitEditing={() => { this.newPassword.focus(); }}
|
|
testID='profile-view-email'
|
|
/>
|
|
<RCTextInput
|
|
inputRef={(e) => { this.newPassword = e; }}
|
|
label={I18n.t('New_Password')}
|
|
placeholder={I18n.t('New_Password')}
|
|
value={newPassword}
|
|
onChangeText={value => this.setState({ newPassword: value })}
|
|
onSubmitEditing={() => {
|
|
if (Object.keys(customFields).length) {
|
|
return this[Object.keys(customFields)[0]].focus();
|
|
}
|
|
this.avatarUrl.focus();
|
|
}}
|
|
secureTextEntry
|
|
testID='profile-view-new-password'
|
|
/>
|
|
{this.renderCustomFields()}
|
|
<RCTextInput
|
|
inputRef={(e) => { this.avatarUrl = e; }}
|
|
label={I18n.t('Avatar_Url')}
|
|
placeholder={I18n.t('Avatar_Url')}
|
|
value={avatarUrl}
|
|
onChangeText={value => this.setState({ avatarUrl: value })}
|
|
onSubmitEditing={this.submit}
|
|
testID='profile-view-avatar-url'
|
|
/>
|
|
{this.renderAvatarButtons()}
|
|
<Button
|
|
title={I18n.t('Save_Changes')}
|
|
type='primary'
|
|
onPress={this.submit}
|
|
disabled={!this.formIsChanged()}
|
|
testID='profile-view-submit'
|
|
loading={saving}
|
|
/>
|
|
<Dialog.Container visible={showPasswordAlert}>
|
|
<Dialog.Title>
|
|
{I18n.t('Please_enter_your_password')}
|
|
</Dialog.Title>
|
|
<Dialog.Description>
|
|
{I18n.t('For_your_security_you_must_enter_your_current_password_to_continue')}
|
|
</Dialog.Description>
|
|
<Dialog.Input
|
|
onChangeText={value => this.setState({ currentPassword: value })}
|
|
secureTextEntry
|
|
testID='profile-view-typed-password'
|
|
style={styles.dialogInput}
|
|
/>
|
|
<Dialog.Button label={I18n.t('Cancel')} onPress={this.closePasswordAlert} />
|
|
<Dialog.Button label={I18n.t('Save')} onPress={this.submit} />
|
|
</Dialog.Container>
|
|
</SafeAreaView>
|
|
</ScrollView>
|
|
</KeyboardView>
|
|
);
|
|
}
|
|
}
|