2017-09-21 17:08:00 +00:00
|
|
|
import React, { Component } from 'react';
|
|
|
|
import PropTypes from 'prop-types';
|
2018-09-25 19:28:42 +00:00
|
|
|
import {
|
2019-02-27 14:26:40 +00:00
|
|
|
ScrollView, Text, View, StyleSheet, FlatList, LayoutAnimation, SafeAreaView, Image
|
2018-09-25 19:28:42 +00:00
|
|
|
} from 'react-native';
|
2017-09-21 17:08:00 +00:00
|
|
|
import { connect } from 'react-redux';
|
2018-06-05 01:17:02 +00:00
|
|
|
import Icon from 'react-native-vector-icons/MaterialIcons';
|
2018-12-21 10:55:35 +00:00
|
|
|
import equal from 'deep-equal';
|
2017-09-21 17:08:00 +00:00
|
|
|
|
2019-01-31 16:08:38 +00:00
|
|
|
import Navigation from '../lib/Navigation';
|
2018-09-25 19:28:42 +00:00
|
|
|
import { logout as logoutAction } from '../actions/login';
|
2019-01-31 16:08:38 +00:00
|
|
|
import Avatar from '../containers/Avatar';
|
|
|
|
import Status from '../containers/status';
|
2018-06-05 01:17:02 +00:00
|
|
|
import Touch from '../utils/touch';
|
|
|
|
import { STATUS_COLORS } from '../constants/colors';
|
|
|
|
import RocketChat from '../lib/rocketchat';
|
|
|
|
import log from '../utils/log';
|
2018-06-01 17:38:13 +00:00
|
|
|
import I18n from '../i18n';
|
2018-08-10 17:26:36 +00:00
|
|
|
import scrollPersistTaps from '../utils/scrollPersistTaps';
|
2019-02-27 14:26:40 +00:00
|
|
|
import { getReadableVersion, isIOS, isAndroid } from '../utils/deviceInfo';
|
|
|
|
import Icons from '../lib/Icons';
|
2017-09-21 17:08:00 +00:00
|
|
|
|
|
|
|
const styles = StyleSheet.create({
|
2018-08-01 19:35:06 +00:00
|
|
|
container: {
|
|
|
|
flex: 1,
|
|
|
|
backgroundColor: '#fff'
|
2017-09-21 17:08:00 +00:00
|
|
|
},
|
2018-06-05 01:17:02 +00:00
|
|
|
item: {
|
|
|
|
flexDirection: 'row',
|
2017-09-21 17:08:00 +00:00
|
|
|
alignItems: 'center'
|
|
|
|
},
|
2018-06-05 01:17:02 +00:00
|
|
|
itemLeft: {
|
|
|
|
marginHorizontal: 10,
|
|
|
|
width: 30,
|
|
|
|
alignItems: 'center'
|
|
|
|
},
|
2019-02-27 14:26:40 +00:00
|
|
|
itemCenter: {
|
|
|
|
flex: 1
|
|
|
|
},
|
2018-06-05 01:17:02 +00:00
|
|
|
itemText: {
|
|
|
|
marginVertical: 16,
|
|
|
|
fontWeight: 'bold',
|
|
|
|
color: '#292E35'
|
2017-09-21 17:08:00 +00:00
|
|
|
},
|
2018-06-05 01:17:02 +00:00
|
|
|
separator: {
|
|
|
|
borderBottomWidth: StyleSheet.hairlineWidth,
|
|
|
|
borderColor: '#ddd',
|
|
|
|
marginVertical: 4
|
2017-09-21 17:08:00 +00:00
|
|
|
},
|
2018-06-05 01:17:02 +00:00
|
|
|
header: {
|
|
|
|
paddingVertical: 16,
|
|
|
|
flexDirection: 'row',
|
|
|
|
alignItems: 'center'
|
|
|
|
},
|
|
|
|
headerTextContainer: {
|
|
|
|
flex: 1,
|
|
|
|
flexDirection: 'column',
|
|
|
|
alignItems: 'flex-start'
|
|
|
|
},
|
|
|
|
headerUsername: {
|
|
|
|
flexDirection: 'row',
|
|
|
|
alignItems: 'center'
|
|
|
|
},
|
|
|
|
avatar: {
|
|
|
|
marginHorizontal: 10
|
|
|
|
},
|
|
|
|
status: {
|
|
|
|
borderRadius: 12,
|
|
|
|
width: 12,
|
|
|
|
height: 12,
|
|
|
|
marginRight: 5
|
|
|
|
},
|
|
|
|
currentServerText: {
|
|
|
|
fontWeight: 'bold'
|
2018-09-21 17:24:15 +00:00
|
|
|
},
|
|
|
|
version: {
|
|
|
|
marginHorizontal: 5,
|
|
|
|
marginBottom: 5,
|
|
|
|
fontWeight: '600',
|
|
|
|
color: '#292E35',
|
|
|
|
fontSize: 13
|
2019-02-27 14:26:40 +00:00
|
|
|
},
|
|
|
|
disclosureContainer: {
|
|
|
|
marginLeft: 6,
|
|
|
|
marginRight: 9,
|
|
|
|
alignItems: 'center',
|
|
|
|
justifyContent: 'center'
|
|
|
|
},
|
|
|
|
disclosureIndicator: {
|
|
|
|
width: 20,
|
|
|
|
height: 20
|
2017-09-21 17:08:00 +00:00
|
|
|
}
|
|
|
|
});
|
2018-03-02 21:31:44 +00:00
|
|
|
const keyExtractor = item => item.id;
|
2018-07-10 13:40:32 +00:00
|
|
|
|
2017-09-21 17:08:00 +00:00
|
|
|
@connect(state => ({
|
2018-11-14 21:42:03 +00:00
|
|
|
Site_Name: state.settings.Site_Name,
|
2018-07-10 13:40:32 +00:00
|
|
|
user: {
|
|
|
|
id: state.login.user && state.login.user.id,
|
|
|
|
language: state.login.user && state.login.user.language,
|
|
|
|
status: state.login.user && state.login.user.status,
|
2019-02-07 19:58:20 +00:00
|
|
|
username: state.login.user && state.login.user.username,
|
|
|
|
token: state.login.user && state.login.user.token
|
2018-09-11 16:32:52 +00:00
|
|
|
},
|
|
|
|
baseUrl: state.settings.Site_Url || state.server ? state.server.server : ''
|
2017-09-21 17:08:00 +00:00
|
|
|
}), dispatch => ({
|
2019-02-27 14:26:40 +00:00
|
|
|
logout: () => dispatch(logoutAction())
|
2017-09-21 17:08:00 +00:00
|
|
|
}))
|
|
|
|
export default class Sidebar extends Component {
|
2019-02-27 14:26:40 +00:00
|
|
|
static options() {
|
|
|
|
return {
|
|
|
|
topBar: {
|
|
|
|
leftButtons: [{
|
|
|
|
id: 'cancel',
|
|
|
|
icon: isAndroid ? Icons.getSource('close', false) : undefined,
|
|
|
|
systemItem: 'cancel'
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-09-21 17:08:00 +00:00
|
|
|
static propTypes = {
|
2018-09-11 16:32:52 +00:00
|
|
|
baseUrl: PropTypes.string,
|
2018-10-23 21:39:48 +00:00
|
|
|
componentId: PropTypes.string,
|
2018-11-14 21:42:03 +00:00
|
|
|
Site_Name: PropTypes.string.isRequired,
|
2018-07-10 13:40:32 +00:00
|
|
|
user: PropTypes.object,
|
2019-02-27 14:26:40 +00:00
|
|
|
logout: PropTypes.func.isRequired
|
2017-09-21 17:08:00 +00:00
|
|
|
}
|
|
|
|
|
2018-03-06 17:40:44 +00:00
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
2018-06-05 01:17:02 +00:00
|
|
|
this.state = {
|
2018-12-21 10:55:35 +00:00
|
|
|
showStatus: false,
|
|
|
|
status: []
|
2018-06-05 01:17:02 +00:00
|
|
|
};
|
2018-10-23 21:39:48 +00:00
|
|
|
Navigation.events().bindComponent(this);
|
2018-03-06 17:40:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
2018-06-13 01:33:00 +00:00
|
|
|
this.setStatus();
|
|
|
|
}
|
|
|
|
|
|
|
|
componentWillReceiveProps(nextProps) {
|
2018-09-25 19:28:42 +00:00
|
|
|
const { user } = this.props;
|
|
|
|
if (nextProps.user && user && user.language !== nextProps.user.language) {
|
2018-06-13 01:33:00 +00:00
|
|
|
this.setStatus();
|
|
|
|
}
|
2018-03-06 17:40:44 +00:00
|
|
|
}
|
|
|
|
|
2018-12-21 10:55:35 +00:00
|
|
|
shouldComponentUpdate(nextProps, nextState) {
|
|
|
|
const { status, showStatus } = this.state;
|
2019-02-27 14:26:40 +00:00
|
|
|
const { Site_Name, user, baseUrl } = this.props;
|
2018-12-21 10:55:35 +00:00
|
|
|
if (nextState.showStatus !== showStatus) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (nextProps.Site_Name !== Site_Name) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (nextProps.Site_Name !== Site_Name) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (nextProps.baseUrl !== baseUrl) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (nextProps.user && user) {
|
|
|
|
if (nextProps.user.language !== user.language) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (nextProps.user.status !== user.status) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (nextProps.user.username !== user.username) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!equal(nextState.status, status)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-10-23 21:39:48 +00:00
|
|
|
navigationButtonPressed = ({ buttonId }) => {
|
|
|
|
if (buttonId === 'cancel') {
|
|
|
|
const { componentId } = this.props;
|
|
|
|
Navigation.dismissModal(componentId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-13 01:33:00 +00:00
|
|
|
setStatus = () => {
|
2018-12-21 10:55:35 +00:00
|
|
|
this.setState({
|
|
|
|
status: [{
|
|
|
|
id: 'online',
|
|
|
|
name: I18n.t('Online')
|
|
|
|
}, {
|
|
|
|
id: 'busy',
|
|
|
|
name: I18n.t('Busy')
|
|
|
|
}, {
|
|
|
|
id: 'away',
|
|
|
|
name: I18n.t('Away')
|
|
|
|
}, {
|
|
|
|
id: 'offline',
|
|
|
|
name: I18n.t('Invisible')
|
|
|
|
}]
|
2018-06-13 01:33:00 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-09-21 17:24:32 +00:00
|
|
|
toggleStatus = () => {
|
2018-06-05 01:17:02 +00:00
|
|
|
LayoutAnimation.easeInEaseOut();
|
2018-09-25 19:28:42 +00:00
|
|
|
this.setState(prevState => ({ showStatus: !prevState.showStatus }));
|
2018-06-05 01:17:02 +00:00
|
|
|
}
|
|
|
|
|
2019-02-27 14:26:40 +00:00
|
|
|
sidebarNavigate = (route) => {
|
|
|
|
const { componentId } = this.props;
|
|
|
|
Navigation.push(componentId, {
|
|
|
|
component: {
|
|
|
|
name: route
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
logout = () => {
|
|
|
|
const { componentId, logout } = this.props;
|
|
|
|
Navigation.dismissModal(componentId);
|
|
|
|
logout();
|
2018-06-05 01:17:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
renderSeparator = key => <View key={key} style={styles.separator} />;
|
2017-09-21 17:08:00 +00:00
|
|
|
|
2018-06-05 01:17:02 +00:00
|
|
|
renderItem = ({
|
2019-02-27 14:26:40 +00:00
|
|
|
text, left, onPress, testID, disclosure
|
2018-06-05 01:17:02 +00:00
|
|
|
}) => (
|
|
|
|
<Touch
|
|
|
|
key={text}
|
|
|
|
onPress={onPress}
|
|
|
|
underlayColor='rgba(255, 255, 255, 0.5)'
|
|
|
|
activeOpacity={0.3}
|
|
|
|
testID={testID}
|
2017-09-21 17:08:00 +00:00
|
|
|
>
|
2019-02-27 14:26:40 +00:00
|
|
|
<View style={styles.item}>
|
2018-07-10 13:40:32 +00:00
|
|
|
<View style={styles.itemLeft}>
|
2018-06-05 01:17:02 +00:00
|
|
|
{left}
|
|
|
|
</View>
|
2019-02-27 14:26:40 +00:00
|
|
|
<View style={styles.itemCenter}>
|
|
|
|
<Text style={styles.itemText}>
|
|
|
|
{text}
|
|
|
|
</Text>
|
|
|
|
</View>
|
|
|
|
{disclosure ? this.renderDisclosure() : null}
|
2017-09-21 17:08:00 +00:00
|
|
|
</View>
|
2018-06-05 01:17:02 +00:00
|
|
|
</Touch>
|
|
|
|
)
|
|
|
|
|
2018-09-25 19:28:42 +00:00
|
|
|
renderStatusItem = ({ item }) => {
|
|
|
|
const { user } = this.props;
|
|
|
|
return (
|
|
|
|
this.renderItem({
|
|
|
|
text: item.name,
|
|
|
|
left: <View style={[styles.status, { backgroundColor: STATUS_COLORS[item.id] }]} />,
|
2018-10-23 21:39:48 +00:00
|
|
|
current: user.status === item.id,
|
2018-09-25 19:28:42 +00:00
|
|
|
onPress: () => {
|
|
|
|
this.toggleStatus();
|
|
|
|
if (user.status !== item.id) {
|
|
|
|
try {
|
|
|
|
RocketChat.setUserPresenceDefaultStatus(item.id);
|
|
|
|
} catch (e) {
|
|
|
|
log('setUserPresenceDefaultStatus', e);
|
|
|
|
}
|
2018-06-05 01:17:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
2018-09-25 19:28:42 +00:00
|
|
|
);
|
|
|
|
}
|
2018-06-05 01:17:02 +00:00
|
|
|
|
2019-02-27 14:26:40 +00:00
|
|
|
// Remove it after https://github.com/RocketChat/Rocket.Chat.ReactNative/pull/643
|
|
|
|
renderDisclosure = () => {
|
|
|
|
if (isIOS) {
|
|
|
|
return (
|
|
|
|
<View style={styles.disclosureContainer}>
|
|
|
|
<Image source={{ uri: 'disclosure_indicator' }} style={styles.disclosureIndicator} />
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
2018-09-25 19:28:42 +00:00
|
|
|
}
|
|
|
|
|
2019-02-27 14:26:40 +00:00
|
|
|
renderNavigation = () => (
|
|
|
|
[
|
|
|
|
this.renderItem({
|
|
|
|
text: I18n.t('Profile'),
|
|
|
|
left: <Icon name='person' size={20} />,
|
|
|
|
onPress: () => this.sidebarNavigate('ProfileView'),
|
|
|
|
testID: 'sidebar-profile',
|
|
|
|
disclosure: true
|
|
|
|
}),
|
|
|
|
this.renderItem({
|
|
|
|
text: I18n.t('Settings'),
|
|
|
|
left: <Icon name='settings' size={20} />,
|
|
|
|
onPress: () => this.sidebarNavigate('SettingsView'),
|
|
|
|
testID: 'sidebar-settings',
|
|
|
|
disclosure: true
|
|
|
|
}),
|
|
|
|
this.renderSeparator('separator-logout'),
|
|
|
|
this.renderItem({
|
|
|
|
text: I18n.t('Logout'),
|
|
|
|
left: <Icon name='exit-to-app' size={20} />,
|
|
|
|
onPress: () => this.logout(),
|
|
|
|
testID: 'sidebar-logout'
|
|
|
|
})
|
|
|
|
]
|
|
|
|
)
|
|
|
|
|
2018-09-25 19:28:42 +00:00
|
|
|
renderStatus = () => {
|
|
|
|
const { status } = this.state;
|
|
|
|
const { user } = this.props;
|
|
|
|
return (
|
|
|
|
<FlatList
|
|
|
|
key='status-list'
|
|
|
|
data={status}
|
|
|
|
extraData={user}
|
|
|
|
renderItem={this.renderStatusItem}
|
|
|
|
keyExtractor={keyExtractor}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
2017-09-21 17:08:00 +00:00
|
|
|
|
|
|
|
render() {
|
2018-09-25 19:28:42 +00:00
|
|
|
const { showStatus } = this.state;
|
2018-11-14 21:42:03 +00:00
|
|
|
const { user, Site_Name, baseUrl } = this.props;
|
2018-09-25 19:28:42 +00:00
|
|
|
|
2018-07-10 13:40:32 +00:00
|
|
|
if (!user) {
|
|
|
|
return null;
|
|
|
|
}
|
2017-09-21 17:08:00 +00:00
|
|
|
return (
|
2019-01-31 16:08:38 +00:00
|
|
|
<SafeAreaView testID='sidebar-view' style={styles.container}>
|
2018-08-10 17:26:36 +00:00
|
|
|
<ScrollView style={styles.container} {...scrollPersistTaps}>
|
2018-06-05 01:17:02 +00:00
|
|
|
<Touch
|
2018-09-21 17:24:32 +00:00
|
|
|
onPress={() => this.toggleStatus()}
|
2018-06-05 01:17:02 +00:00
|
|
|
underlayColor='rgba(255, 255, 255, 0.5)'
|
|
|
|
activeOpacity={0.3}
|
2018-09-21 17:24:32 +00:00
|
|
|
testID='sidebar-toggle-status'
|
2017-11-13 13:53:45 +00:00
|
|
|
>
|
2018-06-05 01:17:02 +00:00
|
|
|
<View style={styles.header}>
|
|
|
|
<Avatar
|
|
|
|
text={user.username}
|
|
|
|
size={30}
|
|
|
|
style={styles.avatar}
|
2018-09-11 16:32:52 +00:00
|
|
|
baseUrl={baseUrl}
|
2019-02-07 19:58:20 +00:00
|
|
|
user={user}
|
2018-06-05 01:17:02 +00:00
|
|
|
/>
|
|
|
|
<View style={styles.headerTextContainer}>
|
|
|
|
<View style={styles.headerUsername}>
|
|
|
|
<Status style={styles.status} id={user.id} />
|
2018-08-01 19:35:06 +00:00
|
|
|
<Text numberOfLines={1}>{user.username}</Text>
|
2018-06-05 01:17:02 +00:00
|
|
|
</View>
|
2018-11-14 21:42:03 +00:00
|
|
|
<Text style={styles.currentServerText} numberOfLines={1}>{Site_Name}</Text>
|
2018-06-05 01:17:02 +00:00
|
|
|
</View>
|
|
|
|
<Icon
|
2018-09-25 19:28:42 +00:00
|
|
|
name={showStatus ? 'keyboard-arrow-up' : 'keyboard-arrow-down'}
|
2018-06-05 01:17:02 +00:00
|
|
|
size={30}
|
2018-07-10 13:40:32 +00:00
|
|
|
style={{ paddingHorizontal: 10 }}
|
2018-06-05 01:17:02 +00:00
|
|
|
/>
|
2017-11-13 13:53:45 +00:00
|
|
|
</View>
|
2018-06-05 01:17:02 +00:00
|
|
|
</Touch>
|
|
|
|
|
|
|
|
{this.renderSeparator('separator-header')}
|
|
|
|
|
2018-09-25 19:28:42 +00:00
|
|
|
{!showStatus ? this.renderNavigation() : null}
|
|
|
|
{showStatus ? this.renderStatus() : null}
|
2018-08-10 17:26:36 +00:00
|
|
|
</ScrollView>
|
2018-09-21 17:24:15 +00:00
|
|
|
<Text style={styles.version}>
|
2019-01-29 19:52:56 +00:00
|
|
|
{getReadableVersion}
|
2018-09-21 17:24:15 +00:00
|
|
|
</Text>
|
2018-08-10 17:26:36 +00:00
|
|
|
</SafeAreaView>
|
2017-09-21 17:08:00 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|