import React, { Component } from 'react';
import PropTypes from 'prop-types';
import {
ScrollView, Text, View, TouchableWithoutFeedback
} from 'react-native';
import { connect } from 'react-redux';
import { dequal } from 'dequal';
import Avatar from '../../containers/Avatar';
import Status from '../../containers/Status/Status';
import { logEvent, events } from '../../utils/log';
import I18n from '../../i18n';
import scrollPersistTaps from '../../utils/scrollPersistTaps';
import { CustomIcon } from '../../lib/Icons';
import styles from './styles';
import SidebarItem from './SidebarItem';
import { themes } from '../../constants/colors';
import { withTheme } from '../../theme';
import { getUserSelector } from '../../selectors/login';
import SafeAreaView from '../../containers/SafeAreaView';
import Navigation from '../../lib/Navigation';
const Separator = React.memo(({ theme }) => <View style={[styles.separator, { borderColor: themes[theme].separatorColor }]} />);
Separator.propTypes = {
theme: PropTypes.string
};
class Sidebar extends Component {
static propTypes = {
baseUrl: PropTypes.string,
navigation: PropTypes.object,
Site_Name: PropTypes.string.isRequired,
user: PropTypes.object,
state: PropTypes.string,
theme: PropTypes.string,
loadingServer: PropTypes.bool,
useRealName: PropTypes.bool,
allowStatusMessage: PropTypes.bool,
isMasterDetail: PropTypes.bool,
viewStatisticsPermission: PropTypes.object,
viewRoomAdministrationPermission: PropTypes.object,
viewUserAdministrationPermission: PropTypes.object,
viewPrivilegedSettingPermission: PropTypes.object
}
constructor(props) {
super(props);
this.state = {
showStatus: false
shouldComponentUpdate(nextProps, nextState) {
const { showStatus, isAdmin } = this.state;
const {
Site_Name, user, baseUrl, state, isMasterDetail, useRealName, theme, viewStatisticsPermission, viewRoomAdministrationPermission, viewUserAdministrationPermission, viewPrivilegedSettingPermission
} = this.props;
// Drawer navigation state
if (state?.index !== nextProps.state?.index) {
return true;
if (nextState.showStatus !== showStatus) {
if (nextProps.Site_Name !== Site_Name) {
if (nextProps.baseUrl !== baseUrl) {
if (nextProps.theme !== theme) {
if (!dequal(nextProps.user, user)) {
if (nextProps.isMasterDetail !== isMasterDetail) {
if (nextProps.useRealName !== useRealName) {
if (nextState.isAdmin !== isAdmin) {
if (!dequal(nextProps.viewStatisticsPermission, viewStatisticsPermission)) {
if (!dequal(nextProps.viewRoomAdministrationPermission, viewRoomAdministrationPermission)) {
if (!dequal(nextProps.viewUserAdministrationPermission, viewUserAdministrationPermission)) {
if (!dequal(nextProps.viewPrivilegedSettingPermission, viewPrivilegedSettingPermission)) {
return false;
getIsAdmin() {
user, viewStatisticsPermission, viewRoomAdministrationPermission, viewUserAdministrationPermission, viewPrivilegedSettingPermission
const { roles } = user;
const allPermissions = [viewStatisticsPermission, viewRoomAdministrationPermission, viewUserAdministrationPermission, viewPrivilegedSettingPermission];
let isAdmin = false;
if (roles) {
isAdmin = allPermissions.reduce((result, permission) => {
if (permission) {
return (
result || permission.some(r => roles.indexOf(r) !== -1)
);
return result;
},
false);
return isAdmin;
sidebarNavigate = (route) => {
logEvent(events[`SIDEBAR_GO_${ route.replace('StackNavigator', '').replace('View', '').toUpperCase() }`]);
Navigation.navigate(route);
get currentItemKey() {
const { state } = this.props;
return state?.routeNames[state?.index];
onPressUser = () => {
const { navigation, isMasterDetail } = this.props;
if (isMasterDetail) {
return;
navigation.closeDrawer();
renderAdmin = () => {
const { theme, isMasterDetail } = this.props;
if (!this.getIsAdmin()) {
return null;
const routeName = isMasterDetail ? 'AdminPanelView' : 'AdminPanelStackNavigator';
<>
<Separator theme={theme} />
<SidebarItem
text={I18n.t('Admin_Panel')}
left={<CustomIcon name='settings' size={20} color={themes[theme].titleText} />}
onPress={() => this.sidebarNavigate(routeName)}
testID='sidebar-settings'
current={this.currentItemKey === routeName}
/>
</>
renderNavigation = () => {
const { theme } = this.props;
text={I18n.t('Chats')}
left={<CustomIcon name='message' size={20} color={themes[theme].titleText} />}
onPress={() => this.sidebarNavigate('ChatsStackNavigator')}
testID='sidebar-chats'
current={this.currentItemKey === 'ChatsStackNavigator'}
text={I18n.t('Profile')}
left={<CustomIcon name='user' size={20} color={themes[theme].titleText} />}
onPress={() => this.sidebarNavigate('ProfileStackNavigator')}
testID='sidebar-profile'
current={this.currentItemKey === 'ProfileStackNavigator'}
text={I18n.t('Settings')}
left={<CustomIcon name='administration' size={20} color={themes[theme].titleText} />}
onPress={() => this.sidebarNavigate('SettingsStackNavigator')}
current={this.currentItemKey === 'SettingsStackNavigator'}
{this.renderAdmin()}
renderCustomStatus = () => {
const { user, theme } = this.props;
text={user.statusText || I18n.t('Edit_Status')}
left={<Status size={24} status={user?.status} />}
right={<CustomIcon name='edit' size={20} color={themes[theme].titleText} />}
onPress={() => this.sidebarNavigate('StatusView')}
testID='sidebar-custom-status'
render() {
user, Site_Name, baseUrl, useRealName, allowStatusMessage, isMasterDetail, theme
if (!user) {
<SafeAreaView testID='sidebar-view' style={{ backgroundColor: themes[theme].focusedBackground }} vertical={isMasterDetail}>
<ScrollView
style={[
styles.container,
{
backgroundColor: isMasterDetail
? themes[theme].backgroundColor
: themes[theme].focusedBackground
]}
{...scrollPersistTaps}
>
<TouchableWithoutFeedback onPress={this.onPressUser} testID='sidebar-close-drawer'>
<View style={styles.header} theme={theme}>
<Avatar
text={user.username}
style={styles.avatar}
size={30}
<View style={styles.headerTextContainer}>
<View style={styles.headerUsername}>
<Text numberOfLines={1} style={[styles.username, { color: themes[theme].titleText }]}>{useRealName ? user.name : user.username}</Text>
</View>
<Text
style={[styles.currentServerText, { color: themes[theme].titleText }]}
numberOfLines={1}
accessibilityLabel={`Connected to ${ baseUrl }`}
>{Site_Name}
</Text>
</TouchableWithoutFeedback>
{allowStatusMessage ? this.renderCustomStatus() : null}
{!isMasterDetail ? (
{this.renderNavigation()}
) : (
)}
</ScrollView>
</SafeAreaView>
const mapStateToProps = state => ({
Site_Name: state.settings.Site_Name,
user: getUserSelector(state),
baseUrl: state.server.server,
loadingServer: state.server.loading,
useRealName: state.settings.UI_Use_Real_Name,
allowStatusMessage: state.settings.Accounts_AllowUserStatusMessageChange,
isMasterDetail: state.app.isMasterDetail,
viewStatisticsPermission: state.permissions['view-statistics'],
viewRoomAdministrationPermission: state.permissions['view-room-administration'],
viewUserAdministrationPermission: state.permissions['view-user-administration'],
viewPrivilegedSettingPermission: state.permissions['view-privileged-setting']
});
export default connect(mapStateToProps)(withTheme(Sidebar));