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 equal from 'deep-equal'; 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); } } shouldComponentUpdate(nextProps, nextState) { if (!equal(nextState, this.state)) { return true; } if (!equal(nextProps, this.props)) { return true; } return false; } 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 }) => ( {child} ) renderAvatarButtons = () => { const { avatarUrl, avatarSuggestions } = this.state; const { user, baseUrl } = this.props; return ( {this.renderAvatarButton({ child: , onPress: () => this.resetAvatar(), key: 'profile-view-reset-avatar' })} {this.renderAvatarButton({ child: , onPress: () => this.pickImage(), key: 'profile-view-upload-avatar' })} {this.renderAvatarButton({ child: , 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: , onPress: () => this.setAvatar({ url, data: blob, service, contentType }) }); })} ); } 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 ( { const newValue = {}; newValue[key] = value; this.setState({ customFields: { ...customFields, ...newValue } }); }} value={customFields[key]} > { this[key] = e; }} label={key} placeholder={key} value={customFields[key]} testID='settings-view-language' /> ); } return ( { 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 ( { 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' /> { 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' /> { 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' /> { 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()} { 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()}