import React from 'react'; import { BackHandler, Keyboard, NativeEventSubscription, RefreshControl, Text, View } from 'react-native'; import { batch, connect } from 'react-redux'; import { dequal } from 'dequal'; import Orientation from 'react-native-orientation-locker'; import { Q } from '@nozbe/watermelondb'; import { Subscription } from 'rxjs'; import { StackNavigationOptions, StackNavigationProp } from '@react-navigation/stack'; import { Header } from '@react-navigation/elements'; import { FlashList } from '@shopify/flash-list'; import { CompositeNavigationProp, RouteProp } from '@react-navigation/native'; import { Dispatch } from 'redux'; import database from '../../lib/database'; import RoomItem, { ROW_HEIGHT, ROW_HEIGHT_CONDENSED } from '../../containers/RoomItem'; import log, { logEvent, events } from '../../lib/methods/helpers/log'; import I18n from '../../i18n'; import { closeSearchHeader, closeServerDropdown, openSearchHeader, roomsRequest } from '../../actions/rooms'; import { appStart } from '../../actions/app'; import * as HeaderButton from '../../containers/HeaderButton'; import StatusBar from '../../containers/StatusBar'; import ActivityIndicator from '../../containers/ActivityIndicator'; import { serverInitAdd } from '../../actions/server'; import { animateNextTransition } from '../../lib/methods/helpers/layoutAnimation'; import { TSupportedThemes, withTheme } from '../../theme'; import EventEmitter from '../../lib/methods/helpers/events'; import { themedHeader } from '../../lib/methods/helpers/navigation'; import { KEY_COMMAND, handleCommandAddNewServer, handleCommandNextRoom, handleCommandPreviousRoom, handleCommandSearching, handleCommandSelectRoom, handleCommandShowNewMessage, handleCommandShowPreferences, IKeyCommandEvent } from '../../commands'; import { getUserSelector } from '../../selectors/login'; import { goRoom } from '../../lib/methods/helpers/goRoom'; import SafeAreaView from '../../containers/SafeAreaView'; import { getInquiryQueueSelector } from '../../ee/omnichannel/selectors/inquiry'; import { IApplicationState, ISubscription, IUser, RootEnum, SubscriptionType, TSubscriptionModel } from '../../definitions'; import styles from './styles'; import ServerDropdown from './ServerDropdown'; import ListHeader, { TEncryptionBanner } from './ListHeader'; import RoomsListHeaderView from './Header'; import { ChatsStackParamList, DrawerParamList } from '../../stacks/types'; import { RoomTypes, search } from '../../lib/methods'; import { getRoomAvatar, getRoomTitle, getUidDirectMessage, hasPermission, isRead, debounce, isTablet } from '../../lib/methods/helpers'; import { E2E_BANNER_TYPE, DisplayMode, SortBy, themes } from '../../lib/constants'; import { Services } from '../../lib/services'; type TNavigation = CompositeNavigationProp< StackNavigationProp, CompositeNavigationProp, StackNavigationProp> >; interface IRoomsListViewProps { navigation: TNavigation; route: RouteProp; theme: TSupportedThemes; dispatch: Dispatch; user: IUser; server: string; searchText: string; changingServer: boolean; loadingServer: boolean; showServerDropdown: boolean; sortBy: string; groupByType: boolean; showFavorites: boolean; showUnread: boolean; refreshing: boolean; StoreLastMessage: boolean; useRealName: boolean; isMasterDetail: boolean; subscribedRoom: string; queueSize: number; inquiryEnabled: boolean; encryptionBanner: string; showAvatar: boolean; displayMode: string; createTeamPermission?: string[]; createDirectMessagePermission?: string[]; createPublicChannelPermission?: string[]; createPrivateChannelPermission?: string[]; createDiscussionPermission?: string[]; } interface IRoomsListViewState { searching?: boolean; search?: IRoomItem[]; loading?: boolean; chats?: IRoomItem[]; item?: ISubscription; canCreateRoom?: boolean; } interface IRoomItem extends ISubscription { search?: boolean; outside?: boolean; } const CHATS_HEADER = 'Chats'; const UNREAD_HEADER = 'Unread'; const FAVORITES_HEADER = 'Favorites'; const DISCUSSIONS_HEADER = 'Discussions'; const TEAMS_HEADER = 'Teams'; const CHANNELS_HEADER = 'Channels'; const DM_HEADER = 'Direct_Messages'; const OMNICHANNEL_HEADER_IN_PROGRESS = 'Open_Livechats'; const OMNICHANNEL_HEADER_ON_HOLD = 'On_hold_Livechats'; const QUERY_SIZE = 20; const filterIsUnread = (s: any) => (s.unread > 0 || s.tunread?.length > 0 || s.alert) && !s.hideUnreadStatus; const filterIsFavorite = (s: any) => s.f; const filterIsOmnichannel = (s: any) => s.t === 'l'; const filterIsTeam = (s: any) => s.teamMain; const filterIsDiscussion = (s: any) => s.prid; const keyExtractor = (item: ISubscription) => item.rid; class RoomsListView extends React.Component { private animated: boolean; private count: number; private unsubscribeFocus?: () => void; private unsubscribeBlur?: () => void; private backHandler?: NativeEventSubscription; private querySubscription?: Subscription; private scroll?: any; private useRealName?: boolean; constructor(props: IRoomsListViewProps) { super(props); console.time(`${this.constructor.name} init`); console.time(`${this.constructor.name} mount`); this.animated = false; this.count = 0; this.state = { searching: false, search: [], loading: true, chats: [], item: {} as ISubscription, canCreateRoom: false }; this.setHeader(); this.getSubscriptions(); } componentDidMount() { const { navigation, dispatch } = this.props; this.handleHasPermission(); if (isTablet) { EventEmitter.addEventListener(KEY_COMMAND, this.handleCommands); } this.unsubscribeFocus = navigation.addListener('focus', () => { Orientation.unlockAllOrientations(); this.animated = true; this.backHandler = BackHandler.addEventListener('hardwareBackPress', this.handleBackPress); }); this.unsubscribeBlur = navigation.addListener('blur', () => { this.animated = false; dispatch(closeServerDropdown()); this.cancelSearch(); if (this.backHandler && this.backHandler.remove) { this.backHandler.remove(); } }); console.timeEnd(`${this.constructor.name} mount`); } UNSAFE_componentWillReceiveProps(nextProps: IRoomsListViewProps) { const { loadingServer, searchText, server, changingServer } = this.props; // when the server is changed if (server !== nextProps.server && loadingServer !== nextProps.loadingServer && nextProps.loadingServer) { this.setState({ loading: true }); } // when the server is changing and stopped loading if (changingServer && loadingServer !== nextProps.loadingServer && !nextProps.loadingServer) { this.getSubscriptions(); } if (searchText !== nextProps.searchText) { this.handleSearch(nextProps.searchText); } } shouldComponentUpdate(nextProps: Readonly, nextState: Readonly): boolean { const { createTeamPermission, createPublicChannelPermission, createPrivateChannelPermission, createDirectMessagePermission, createDiscussionPermission } = this.props; if ( !dequal(createTeamPermission, nextProps.createTeamPermission) || !dequal(createPublicChannelPermission, nextProps.createPublicChannelPermission) || !dequal(createPrivateChannelPermission, nextProps.createPrivateChannelPermission) || !dequal(createDirectMessagePermission, nextProps.createDirectMessagePermission) || !dequal(createDiscussionPermission, nextProps.createDiscussionPermission) ) { return true; } const { chats, search } = this.state; if (!dequal(chats, nextState.chats) || !dequal(search, nextState.search)) { return true; } const { sortBy, groupByType, showFavorites, showUnread, subscribedRoom, isMasterDetail, showAvatar, displayMode, encryptionBanner, showServerDropdown } = this.props; if ( sortBy !== nextProps.sortBy || groupByType !== nextProps.groupByType || showFavorites !== nextProps.showFavorites || showUnread !== nextProps.showUnread || showAvatar !== nextProps.showAvatar || displayMode !== nextProps.displayMode || subscribedRoom !== nextProps.subscribedRoom || isMasterDetail !== nextProps.isMasterDetail || encryptionBanner !== nextProps.encryptionBanner || showServerDropdown !== nextProps.showServerDropdown ) { return true; } const { searching, loading, canCreateRoom } = this.state; if (searching !== nextState.searching || loading !== nextState.loading || canCreateRoom !== nextState.canCreateRoom) { return true; } return false; } componentDidUpdate(prevProps: IRoomsListViewProps) { const { sortBy, groupByType, showFavorites, showUnread, subscribedRoom, isMasterDetail, createTeamPermission, createPublicChannelPermission, createPrivateChannelPermission, createDirectMessagePermission, createDiscussionPermission, showAvatar, displayMode } = this.props; const { item } = this.state; if ( !( prevProps.sortBy === sortBy && prevProps.groupByType === groupByType && prevProps.showFavorites === showFavorites && prevProps.showUnread === showUnread && prevProps.showAvatar === showAvatar && prevProps.displayMode === displayMode ) ) { this.getSubscriptions(); } // Update current item in case of another action triggers an update on room subscribed reducer if (isMasterDetail && item?.rid !== subscribedRoom && subscribedRoom !== prevProps.subscribedRoom) { this.setState({ item: { rid: subscribedRoom } as ISubscription }); } if ( !dequal(createTeamPermission, prevProps.createTeamPermission) || !dequal(createPublicChannelPermission, prevProps.createPublicChannelPermission) || !dequal(createPrivateChannelPermission, prevProps.createPrivateChannelPermission) || !dequal(createDirectMessagePermission, prevProps.createDirectMessagePermission) || !dequal(createDiscussionPermission, prevProps.createDiscussionPermission) ) { this.handleHasPermission(); this.setHeader(); } } componentWillUnmount() { this.unsubscribeQuery(); if (this.unsubscribeFocus) { this.unsubscribeFocus(); } if (this.unsubscribeBlur) { this.unsubscribeBlur(); } if (this.backHandler && this.backHandler.remove) { this.backHandler.remove(); } if (isTablet) { EventEmitter.removeListener(KEY_COMMAND, this.handleCommands); } console.countReset(`${this.constructor.name}.render calls`); } handleHasPermission = async () => { const { createTeamPermission, createDirectMessagePermission, createPublicChannelPermission, createPrivateChannelPermission, createDiscussionPermission } = this.props; const permissions = [ createPublicChannelPermission, createPrivateChannelPermission, createTeamPermission, createDirectMessagePermission, createDiscussionPermission ]; const permissionsToCreate = await hasPermission(permissions); const canCreateRoom = permissionsToCreate.filter((r: boolean) => r === true).length > 0; this.setState({ canCreateRoom }, () => this.setHeader()); }; getHeader = (): StackNavigationOptions => { const { searching, canCreateRoom } = this.state; const { navigation, isMasterDetail } = this.props; if (searching) { return { headerTitleAlign: 'left', headerTitleContainerStyle: { flex: 1, marginHorizontal: 0, marginRight: 15, maxWidth: undefined }, headerRightContainerStyle: { flexGrow: 0 }, headerLeft: () => ( ), headerTitle: () => , headerRight: () => null }; } return { headerTitleAlign: 'left', headerTitleContainerStyle: { flex: 1, marginHorizontal: 4, maxWidth: undefined }, headerRightContainerStyle: { flexGrow: undefined, flexBasis: undefined }, headerLeft: () => ( navigation.navigate('ModalStackNavigator', { screen: 'SettingsView' }) : // @ts-ignore () => navigation.toggleDrawer() } /> ), headerTitle: () => , headerRight: () => ( {canCreateRoom ? ( ) : null} ) }; }; setHeader = () => { const { navigation } = this.props; const options = this.getHeader(); navigation.setOptions(options); }; internalSetState = ( state: | (( prevState: Readonly, props: Readonly ) => Pick | IRoomsListViewState | null) | (Pick | IRoomsListViewState | null), callback?: () => void ) => { this.setState(state, callback); if (this.animated) { this.scroll?.prepareForLayoutAnimationRender(); animateNextTransition(); } }; addRoomsGroup = (data: ISubscription[], header: string, allData: ISubscription[]) => { if (data.length > 0) { if (header) { allData.push({ rid: header, separator: true } as TSubscriptionModel); } allData = allData.concat(data); } return allData; }; getSubscriptions = async () => { console.log('🚀 ~ file: index.tsx ~ line 408 ~ RoomsListView ~ getSubscriptions'); this.unsubscribeQuery(); const { sortBy, showUnread, showFavorites, groupByType, user } = this.props; const db = database.active; let observable; const defaultWhereClause = [Q.where('archived', false), Q.where('open', true)] as (Q.WhereDescription | Q.SortBy)[]; const columnsToObserve = ['alert', 'f', 'on_hold', 'room_updated_at']; if (sortBy === SortBy.Alphabetical) { defaultWhereClause.push(Q.experimentalSortBy(`${this.useRealName ? 'fname' : 'name'}`, Q.asc)); } else { defaultWhereClause.push(Q.experimentalSortBy('room_updated_at', Q.desc)); } // When we're grouping by something if (this.isGrouping) { observable = await db .get('subscriptions') .query(...defaultWhereClause) .observeWithColumns(columnsToObserve); // When we're NOT grouping } else { this.count += QUERY_SIZE; observable = await db .get('subscriptions') .query(...defaultWhereClause, Q.experimentalSkip(0), Q.experimentalTake(this.count)) .observeWithColumns(columnsToObserve); } this.querySubscription = observable.subscribe(data => { let tempChats: ISubscription[] = []; let chats = data.map(item => item.asPlain()); const isOmnichannelAgent = user?.roles?.includes('livechat-agent'); if (isOmnichannelAgent) { const omnichannel = chats.filter(s => filterIsOmnichannel(s)); const omnichannelInProgress = omnichannel.filter(s => !s.onHold); const omnichannelOnHold = omnichannel.filter(s => s.onHold); chats = chats.filter(s => !filterIsOmnichannel(s)); tempChats = this.addRoomsGroup(omnichannelInProgress, OMNICHANNEL_HEADER_IN_PROGRESS, tempChats); tempChats = this.addRoomsGroup(omnichannelOnHold, OMNICHANNEL_HEADER_ON_HOLD, tempChats); } // unread if (showUnread) { const unread = chats.filter(s => filterIsUnread(s)); chats = chats.filter(s => !filterIsUnread(s)); tempChats = this.addRoomsGroup(unread, UNREAD_HEADER, tempChats); } // favorites if (showFavorites) { const favorites = chats.filter(s => filterIsFavorite(s)); chats = chats.filter(s => !filterIsFavorite(s)); tempChats = this.addRoomsGroup(favorites, FAVORITES_HEADER, tempChats); } // type if (groupByType) { const teams = chats.filter(s => filterIsTeam(s)); const discussions = chats.filter(s => filterIsDiscussion(s)); const channels = chats.filter(s => (s.t === 'c' || s.t === 'p') && !filterIsDiscussion(s) && !filterIsTeam(s)); const direct = chats.filter(s => s.t === 'd' && !filterIsDiscussion(s) && !filterIsTeam(s)); tempChats = this.addRoomsGroup(teams, TEAMS_HEADER, tempChats); tempChats = this.addRoomsGroup(discussions, DISCUSSIONS_HEADER, tempChats); tempChats = this.addRoomsGroup(channels, CHANNELS_HEADER, tempChats); tempChats = this.addRoomsGroup(direct, DM_HEADER, tempChats); } else if (showUnread || showFavorites || isOmnichannelAgent) { tempChats = this.addRoomsGroup(chats, CHATS_HEADER, tempChats); } else { tempChats = chats; } this.internalSetState({ chats: tempChats, loading: false }); }); }; unsubscribeQuery = () => { if (this.querySubscription && this.querySubscription.unsubscribe) { this.querySubscription.unsubscribe(); } }; initSearching = () => { logEvent(events.RL_SEARCH); const { dispatch } = this.props; this.internalSetState({ searching: true }, () => { dispatch(openSearchHeader()); this.handleSearch(''); this.setHeader(); }); }; cancelSearch = () => { const { searching } = this.state; const { dispatch } = this.props; if (!searching) { return; } Keyboard.dismiss(); this.setState({ searching: false, search: [] }, () => { this.setHeader(); dispatch(closeSearchHeader()); setTimeout(() => { this.scrollToTop(); }, 200); }); }; handleBackPress = () => { const { searching } = this.state; if (searching) { this.cancelSearch(); return true; } return false; }; // eslint-disable-next-line react/sort-comp handleSearch = debounce(async (text: string) => { const result = await search({ text }); // if the search was cancelled before the promise is resolved const { searching } = this.state; if (!searching) { return; } this.internalSetState({ search: result as IRoomItem[], searching: true }); this.scrollToTop(); }, 300); isSwipeEnabled = (item: IRoomItem) => !(item?.search || item?.joinCodeRequired || item?.outside); get isGrouping() { const { showUnread, showFavorites, groupByType } = this.props; return showUnread || showFavorites || groupByType; } onPressItem = (item = {} as ISubscription) => { const { navigation, isMasterDetail } = this.props; if (!navigation.isFocused()) { return; } this.cancelSearch(); this.goRoom({ item, isMasterDetail }); }; scrollToTop = () => { if (this.scroll?.scrollToOffset) { this.scroll.scrollToOffset({ offset: 0 }); } }; toggleFav = async (rid: string, favorite: boolean): Promise => { logEvent(favorite ? events.RL_UNFAVORITE_CHANNEL : events.RL_FAVORITE_CHANNEL); try { const db = database.active; const result = await Services.toggleFavorite(rid, !favorite); if (result.success) { const subCollection = db.get('subscriptions'); await db.write(async () => { try { const subRecord = await subCollection.find(rid); await subRecord.update(sub => { sub.f = !favorite; }); } catch (e) { log(e); } }); } } catch (e) { logEvent(events.RL_TOGGLE_FAVORITE_F); log(e); } }; toggleRead = async (rid: string, tIsRead: boolean) => { logEvent(tIsRead ? events.RL_UNREAD_CHANNEL : events.RL_READ_CHANNEL); try { const db = database.active; const result = await Services.toggleRead(tIsRead, rid); if (result.success) { const subCollection = db.get('subscriptions'); await db.write(async () => { try { const subRecord = await subCollection.find(rid); await subRecord.update(sub => { sub.alert = tIsRead; sub.unread = 0; }); } catch (e) { log(e); } }); } } catch (e) { logEvent(events.RL_TOGGLE_READ_F); log(e); } }; hideChannel = async (rid: string, type: SubscriptionType) => { logEvent(events.RL_HIDE_CHANNEL); try { const db = database.active; const result = await Services.hideRoom(rid, type as RoomTypes); if (result.success) { const subCollection = db.get('subscriptions'); await db.write(async () => { try { const subRecord = await subCollection.find(rid); await subRecord.destroyPermanently(); } catch (e) { log(e); } }); } } catch (e) { logEvent(events.RL_HIDE_CHANNEL_F); log(e); } }; goDirectory = () => { logEvent(events.RL_GO_DIRECTORY); const { navigation, isMasterDetail } = this.props; if (isMasterDetail) { navigation.navigate('ModalStackNavigator', { screen: 'DirectoryView' }); } else { navigation.navigate('DirectoryView'); } }; goQueue = () => { logEvent(events.RL_GO_QUEUE); const { navigation, isMasterDetail, inquiryEnabled } = this.props; if (!inquiryEnabled) { return; } if (isMasterDetail) { navigation.navigate('ModalStackNavigator', { screen: 'QueueListView' }); } else { navigation.navigate('QueueListView'); } }; goRoom = ({ item, isMasterDetail }: { item: ISubscription; isMasterDetail: boolean }) => { logEvent(events.RL_GO_ROOM); const { item: currentItem } = this.state; const { subscribedRoom } = this.props; if (currentItem?.rid === item.rid || subscribedRoom === item.rid) { return; } // Only mark room as focused when in master detail layout if (isMasterDetail) { this.setState({ item }); } goRoom({ item, isMasterDetail }); }; goRoomByIndex = (index: number) => { const { chats } = this.state; const { isMasterDetail } = this.props; const filteredChats = chats ? chats.filter(c => !c.separator) : []; const room = filteredChats[index - 1]; if (room) { this.goRoom({ item: room, isMasterDetail }); } }; findOtherRoom = (index: number, sign: number): ISubscription | void => { const { chats } = this.state; const otherIndex = index + sign; const otherRoom = chats?.length ? chats[otherIndex] : ({} as IRoomItem); if (!otherRoom) { return; } if (otherRoom.separator) { return this.findOtherRoom(otherIndex, sign); } return otherRoom; }; // Go to previous or next room based on sign (-1 or 1) // It's used by iPad key commands goOtherRoom = (sign: number) => { const { item } = this.state; if (!item) { return; } // Don't run during search const { search } = this.state; if (search && search?.length > 0) { return; } const { chats } = this.state; const { isMasterDetail } = this.props; if (!chats?.length) { return; } const index = chats.findIndex(c => c.rid === item.rid); const otherRoom = this.findOtherRoom(index, sign); if (otherRoom) { this.goRoom({ item: otherRoom, isMasterDetail }); } }; goToNewMessage = () => { logEvent(events.RL_GO_NEW_MSG); const { navigation, isMasterDetail } = this.props; if (isMasterDetail) { navigation.navigate('ModalStackNavigator', { screen: 'NewMessageView' }); } else { navigation.navigate('NewMessageStackNavigator'); } }; goEncryption = () => { logEvent(events.RL_GO_E2E_SAVE_PASSWORD); const { navigation, isMasterDetail, encryptionBanner } = this.props; const isSavePassword = encryptionBanner === E2E_BANNER_TYPE.SAVE_PASSWORD; if (isMasterDetail) { const screen = isSavePassword ? 'E2ESaveYourPasswordView' : 'E2EEnterYourPasswordView'; navigation.navigate('ModalStackNavigator', { screen }); } else { const screen = isSavePassword ? 'E2ESaveYourPasswordStackNavigator' : 'E2EEnterYourPasswordStackNavigator'; navigation.navigate(screen); } }; handleCommands = ({ event }: { event: IKeyCommandEvent }) => { const { navigation, server, isMasterDetail, dispatch } = this.props; const { input } = event; if (handleCommandShowPreferences(event)) { navigation.navigate('SettingsView'); } else if (handleCommandSearching(event)) { this.initSearching(); } else if (handleCommandSelectRoom(event)) { this.goRoomByIndex(input); } else if (handleCommandPreviousRoom(event)) { this.goOtherRoom(-1); } else if (handleCommandNextRoom(event)) { this.goOtherRoom(1); } else if (handleCommandShowNewMessage(event)) { if (isMasterDetail) { navigation.navigate('ModalStackNavigator', { screen: 'NewMessageView' }); } else { navigation.navigate('NewMessageStack'); } } else if (handleCommandAddNewServer(event)) { batch(() => { dispatch(appStart({ root: RootEnum.ROOT_OUTSIDE })); dispatch(serverInitAdd(server)); }); } }; onRefresh = () => { const { searching } = this.state; const { dispatch } = this.props; if (searching) { return; } dispatch(roomsRequest({ allData: true })); }; onEndReached = () => { const { searching } = this.state; if (searching) { return; } this.getSubscriptions(); }; getScrollRef = (ref: any) => (this.scroll = ref); renderListHeader = () => { const { searching } = this.state; const { queueSize, inquiryEnabled, encryptionBanner, user } = this.props; return ( ); }; renderHeader = () => { const { isMasterDetail, theme } = this.props; if (!isMasterDetail) { return null; } const options = this.getHeader(); return
; }; renderItem = ({ item }: { item: IRoomItem }) => { if (item.separator) { return this.renderSectionHeader(item.rid); } const { item: currentItem } = this.state; const { user: { username }, StoreLastMessage, useRealName, showAvatar, displayMode } = this.props; const id = getUidDirectMessage(item); const swipeEnabled = this.isSwipeEnabled(item); return ( ); }; renderSectionHeader = (header: string) => { const { theme } = this.props; return ( {I18n.t(header)} ); }; render = () => { console.count(`${this.constructor.name}.render calls`); const { showServerDropdown, theme, navigation, refreshing, displayMode, queueSize, inquiryEnabled, encryptionBanner } = this.props; const { loading, chats, search, searching } = this.state; const height = displayMode === DisplayMode.Condensed ? ROW_HEIGHT_CONDENSED : ROW_HEIGHT; return ( {this.renderHeader()} {loading ? ( ) : ( (item.separator ? 'section' : 'item')} ListHeaderComponent={this.renderListHeader} estimatedItemSize={height} keyboardShouldPersistTaps='always' refreshControl={ searching ? undefined : ( ) } onEndReached={this.isGrouping ? undefined : this.onEndReached} onEndReachedThreshold={this.isGrouping ? undefined : 0.5} /> )} {/* TODO - this ts-ignore is here because the route props, on IBaseScreen*/} {/* @ts-ignore*/} {showServerDropdown ? : null} ); }; } const mapStateToProps = (state: IApplicationState) => ({ user: getUserSelector(state), isMasterDetail: state.app.isMasterDetail, server: state.server.server, changingServer: state.server.changingServer, searchText: state.rooms.searchText, loadingServer: state.server.loading, showServerDropdown: state.rooms.showServerDropdown, refreshing: state.rooms.refreshing, sortBy: state.sortPreferences.sortBy, groupByType: state.sortPreferences.groupByType, showFavorites: state.sortPreferences.showFavorites, showUnread: state.sortPreferences.showUnread, useRealName: state.settings.UI_Use_Real_Name as boolean, StoreLastMessage: state.settings.Store_Last_Message as boolean, subscribedRoom: state.room.subscribedRoom, queueSize: getInquiryQueueSelector(state).length, inquiryEnabled: state.inquiry.enabled, encryptionBanner: state.encryption.banner, showAvatar: state.sortPreferences.showAvatar, displayMode: state.sortPreferences.displayMode, createTeamPermission: state.permissions['create-team'], createDirectMessagePermission: state.permissions['create-d'], createPublicChannelPermission: state.permissions['create-c'], createPrivateChannelPermission: state.permissions['create-p'], createDiscussionPermission: state.permissions['start-discussion'] }); export default connect(mapStateToProps)(withTheme(RoomsListView));