import React from 'react'; import PropTypes from 'prop-types'; import { BackHandler, FlatList, Keyboard, RefreshControl, Text, View } from 'react-native'; import { connect } from 'react-redux'; import { dequal } from 'dequal'; import Orientation from 'react-native-orientation-locker'; import { Q } from '@nozbe/watermelondb'; import { withSafeAreaInsets } from 'react-native-safe-area-context'; import database from '../../lib/database'; import RocketChat from '../../lib/rocketchat'; import RoomItem, { ROW_HEIGHT } from '../../presentation/RoomItem'; import log, { events, logEvent } from '../../utils/log'; import I18n from '../../i18n'; import { closeSearchHeader as closeSearchHeaderAction, closeServerDropdown as closeServerDropdownAction, openSearchHeader as openSearchHeaderAction, roomsRequest as roomsRequestAction, toggleSortDropdown as toggleSortDropdownAction } from '../../actions/rooms'; import debounce from '../../utils/debounce'; import { isIOS, isTablet } from '../../utils/deviceInfo'; import * as HeaderButton from '../../containers/HeaderButton'; import StatusBar from '../../containers/StatusBar'; import ActivityIndicator from '../../containers/ActivityIndicator'; import { selectServerRequest as selectServerRequestAction } from '../../actions/server'; import { animateNextTransition } from '../../utils/layoutAnimation'; import { withTheme } from '../../theme'; import { themes } from '../../constants/colors'; import EventEmitter from '../../utils/events'; import { KEY_COMMAND, handleCommandAddNewServer, handleCommandNextRoom, handleCommandPreviousRoom, handleCommandSearching, handleCommandSelectRoom, handleCommandShowNewMessage, handleCommandShowPreferences } from '../../commands'; import { MAX_SIDEBAR_WIDTH } from '../../constants/tablet'; import { getUserSelector } from '../../selectors/login'; import { goRoom } from '../../utils/goRoom'; import SafeAreaView from '../../containers/SafeAreaView'; import Header, { getHeaderTitlePosition } from '../../containers/Header'; import { withDimensions } from '../../dimensions'; import { showConfirmationAlert, showErrorAlert } from '../../utils/info'; import { E2E_BANNER_TYPE } from '../../lib/encryption/constants'; import { getInquiryQueueSelector } from '../../ee/omnichannel/selectors/inquiry'; import { changeLivechatStatus, isOmnichannelStatusAvailable } from '../../ee/omnichannel/lib'; import ListHeader from './ListHeader'; import RoomsListHeaderView from './Header'; import ServerDropdown from './ServerDropdown'; import SortDropdown from './SortDropdown'; import styles from './styles'; const INITIAL_NUM_TO_RENDER = isTablet ? 20 : 12; 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 = 'Open_Livechats'; const QUERY_SIZE = 20; const filterIsUnread = s => (s.unread > 0 || s.tunread?.length > 0 || s.alert) && !s.hideUnreadStatus; const filterIsFavorite = s => s.f; const filterIsOmnichannel = s => s.t === 'l'; const filterIsTeam = s => s.teamMain; const filterIsDiscussion = s => s.prid; const shouldUpdateProps = [ 'searchText', 'loadingServer', 'showServerDropdown', 'showSortDropdown', 'sortBy', 'groupByType', 'showFavorites', 'showUnread', 'useRealName', 'StoreLastMessage', 'theme', 'isMasterDetail', 'refreshing', 'queueSize', 'inquiryEnabled', 'encryptionBanner' ]; const getItemLayout = (data, index) => ({ length: ROW_HEIGHT, offset: ROW_HEIGHT * index, index }); const keyExtractor = item => item.rid; class RoomsListView extends React.Component { static propTypes = { navigation: PropTypes.object, user: PropTypes.shape({ id: PropTypes.string, username: PropTypes.string, token: PropTypes.string, statusLivechat: PropTypes.string, roles: PropTypes.object }), server: PropTypes.string, searchText: PropTypes.string, changingServer: PropTypes.bool, loadingServer: PropTypes.bool, showServerDropdown: PropTypes.bool, showSortDropdown: PropTypes.bool, sortBy: PropTypes.string, groupByType: PropTypes.bool, showFavorites: PropTypes.bool, showUnread: PropTypes.bool, refreshing: PropTypes.bool, StoreLastMessage: PropTypes.bool, theme: PropTypes.string, toggleSortDropdown: PropTypes.func, openSearchHeader: PropTypes.func, closeSearchHeader: PropTypes.func, appStart: PropTypes.func, roomsRequest: PropTypes.func, closeServerDropdown: PropTypes.func, useRealName: PropTypes.bool, isMasterDetail: PropTypes.bool, rooms: PropTypes.array, width: PropTypes.number, insets: PropTypes.object, queueSize: PropTypes.number, inquiryEnabled: PropTypes.bool, encryptionBanner: PropTypes.string }; constructor(props) { super(props); console.time(`${this.constructor.name} init`); console.time(`${this.constructor.name} mount`); this.animated = false; this.mounted = false; this.count = 0; this.state = { searching: false, search: [], loading: true, chatsUpdate: [], chats: [], item: {} }; this.setHeader(); this.getSubscriptions(); } componentDidMount() { const { navigation, closeServerDropdown } = this.props; this.mounted = true; if (isTablet) { EventEmitter.addEventListener(KEY_COMMAND, this.handleCommands); } this.unsubscribeFocus = navigation.addListener('focus', () => { Orientation.unlockAllOrientations(); this.animated = true; // Check if there were changes while not focused (it's set on sCU) if (this.shouldUpdate) { this.forceUpdate(); this.shouldUpdate = false; } this.backHandler = BackHandler.addEventListener('hardwareBackPress', this.handleBackPress); }); this.unsubscribeBlur = navigation.addListener('blur', () => { this.animated = false; closeServerDropdown(); this.cancelSearch(); if (this.backHandler && this.backHandler.remove) { this.backHandler.remove(); } }); console.timeEnd(`${this.constructor.name} mount`); } UNSAFE_componentWillReceiveProps(nextProps) { 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.search(nextProps.searchText); } } shouldComponentUpdate(nextProps, nextState) { const { chatsUpdate, searching, item } = this.state; // eslint-disable-next-line react/destructuring-assignment const propsUpdated = shouldUpdateProps.some(key => nextProps[key] !== this.props[key]); if (propsUpdated) { return true; } // Compare changes only once const chatsNotEqual = !dequal(nextState.chatsUpdate, chatsUpdate); // If they aren't equal, set to update if focused if (chatsNotEqual) { this.shouldUpdate = true; } if (nextState.searching !== searching) { return true; } if (nextState.item?.rid !== item?.rid) { return true; } // Abort if it's not focused if (!nextProps.navigation.isFocused()) { return false; } const { loading, search } = this.state; const { rooms, width, insets } = this.props; if (nextState.loading !== loading) { return true; } if (nextProps.width !== width) { return true; } if (!dequal(nextState.search, search)) { return true; } if (!dequal(nextProps.rooms, rooms)) { return true; } if (!dequal(nextProps.insets, insets)) { return true; } // If it's focused and there are changes, update if (chatsNotEqual) { this.shouldUpdate = false; return true; } return false; } componentDidUpdate(prevProps) { const { sortBy, groupByType, showFavorites, showUnread, rooms, isMasterDetail, insets } = this.props; const { item } = this.state; if ( !( prevProps.sortBy === sortBy && prevProps.groupByType === groupByType && prevProps.showFavorites === showFavorites && prevProps.showUnread === showUnread ) ) { this.getSubscriptions(); } // Update current item in case of another action triggers an update on rooms reducer if (isMasterDetail && item?.rid !== rooms[0] && !dequal(rooms, prevProps.rooms)) { // eslint-disable-next-line react/no-did-update-set-state this.setState({ item: { rid: rooms[0] } }); } if (insets.left !== prevProps.insets.left || insets.right !== prevProps.insets.right) { 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`); } getHeader = () => { const { searching } = this.state; const { navigation, isMasterDetail, insets } = this.props; const headerTitlePosition = getHeaderTitlePosition({ insets, numIconsRight: searching ? 0 : 3 }); return { headerTitleAlign: 'left', headerLeft: () => searching ? ( ) : ( navigation.navigate('ModalStackNavigator', { screen: 'SettingsView' }) : () => navigation.toggleDrawer() } /> ), headerTitle: () => , headerTitleContainerStyle: { left: headerTitlePosition.left, right: headerTitlePosition.right }, headerRight: () => searching ? null : ( ) }; }; setHeader = () => { const { navigation } = this.props; const options = this.getHeader(); navigation.setOptions(options); }; internalSetState = (...args) => { if (this.animated) { animateNextTransition(); } this.setState(...args); }; addRoomsGroup = (data, header, allData) => { if (data.length > 0) { if (header) { allData.push({ rid: header, separator: true }); } allData = allData.concat(data); } return allData; }; getSubscriptions = async () => { 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)]; if (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.collections .get('subscriptions') .query(...defaultWhereClause) .observeWithColumns(['alert']); // When we're NOT grouping } else { this.count += QUERY_SIZE; observable = await db.collections .get('subscriptions') .query(...defaultWhereClause, Q.experimentalSkip(0), Q.experimentalTake(this.count)) .observe(); } this.querySubscription = observable.subscribe(data => { let tempChats = []; let chats = data; let chatsUpdate = []; if (showUnread) { /** * If unread on top, we trigger re-render based on order changes and sub.alert * RoomItem handles its own re-render */ chatsUpdate = data.map(item => ({ rid: item.rid, alert: item.alert })); } else { /** * Otherwise, we trigger re-render only when chats order changes * RoomItem handles its own re-render */ chatsUpdate = data.map(item => item.rid); } const isOmnichannelAgent = user?.roles?.includes('livechat-agent'); if (isOmnichannelAgent) { const omnichannel = chats.filter(s => filterIsOmnichannel(s)); chats = chats.filter(s => !filterIsOmnichannel(s)); tempChats = this.addRoomsGroup(omnichannel, OMNICHANNEL_HEADER, 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; } if (this.mounted) { this.internalSetState({ chats: tempChats, chatsUpdate, loading: false }); } else { this.state.chats = tempChats; this.state.chatsUpdate = chatsUpdate; this.state.loading = false; } }); }; unsubscribeQuery = () => { if (this.querySubscription && this.querySubscription.unsubscribe) { this.querySubscription.unsubscribe(); } }; initSearching = () => { logEvent(events.RL_SEARCH); const { openSearchHeader } = this.props; this.internalSetState({ searching: true }, () => { openSearchHeader(); this.search(''); this.setHeader(); }); }; cancelSearch = () => { const { searching } = this.state; const { closeSearchHeader } = this.props; if (!searching) { return; } Keyboard.dismiss(); this.setState({ searching: false, search: [] }, () => { this.setHeader(); 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 search = debounce(async text => { const result = await RocketChat.search({ text }); // if the search was cancelled before the promise is resolved const { searching } = this.state; if (!searching) { return; } this.internalSetState({ search: result, searching: true }); this.scrollToTop(); }, 300); getRoomTitle = item => RocketChat.getRoomTitle(item); getRoomAvatar = item => RocketChat.getRoomAvatar(item); isGroupChat = item => RocketChat.isGroupChat(item); isRead = item => RocketChat.isRead(item); getUserPresence = uid => RocketChat.getUserPresence(uid); getUidDirectMessage = room => RocketChat.getUidDirectMessage(room); get isGrouping() { const { showUnread, showFavorites, groupByType } = this.props; return showUnread || showFavorites || groupByType; } onPressItem = (item = {}) => { 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 }); } }; toggleSort = () => { logEvent(events.RL_TOGGLE_SORT_DROPDOWN); const { toggleSortDropdown } = this.props; this.scrollToTop(); setTimeout(() => { toggleSortDropdown(); }, 100); }; toggleFav = async (rid, favorite) => { logEvent(favorite ? events.RL_UNFAVORITE_CHANNEL : events.RL_FAVORITE_CHANNEL); try { const db = database.active; const result = await RocketChat.toggleFavorite(rid, !favorite); if (result.success) { const subCollection = db.get('subscriptions'); await db.action(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_FAIL); log(e); } }; toggleRead = async (rid, isRead) => { logEvent(isRead ? events.RL_UNREAD_CHANNEL : events.RL_READ_CHANNEL); try { const db = database.active; const result = await RocketChat.toggleRead(isRead, rid); if (result.success) { const subCollection = db.get('subscriptions'); await db.action(async () => { try { const subRecord = await subCollection.find(rid); await subRecord.update(sub => { sub.alert = isRead; sub.unread = 0; }); } catch (e) { log(e); } }); } } catch (e) { logEvent(events.RL_TOGGLE_READ_F); log(e); } }; hideChannel = async (rid, type) => { logEvent(events.RL_HIDE_CHANNEL); try { const db = database.active; const result = await RocketChat.hideRoom(rid, type); if (result.success) { const subCollection = db.get('subscriptions'); await db.action(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, queueSize, inquiryEnabled, user } = this.props; // if not-available, prompt to change to available if (!isOmnichannelStatusAvailable(user)) { showConfirmationAlert({ message: I18n.t('Omnichannel_enable_alert'), confirmationText: I18n.t('Yes'), onPress: async () => { try { await changeLivechatStatus(); } catch { // Do nothing } } }); } if (!inquiryEnabled) { return; } // prevent navigation to empty list if (!queueSize) { return showErrorAlert(I18n.t('Queue_is_empty'), I18n.t('Oops')); } if (isMasterDetail) { navigation.navigate('ModalStackNavigator', { screen: 'QueueListView' }); } else { navigation.navigate('QueueListView'); } }; goRoom = ({ item, isMasterDetail }) => { logEvent(events.RL_GO_ROOM); const { item: currentItem } = this.state; const { rooms } = this.props; if (currentItem?.rid === item.rid || rooms?.includes(item.rid)) { return; } // Only mark room as focused when in master detail layout if (isMasterDetail) { this.setState({ item }); } goRoom({ item, isMasterDetail }); }; goRoomByIndex = index => { const { chats } = this.state; const { isMasterDetail } = this.props; const filteredChats = chats.filter(c => !c.separator); const room = filteredChats[index - 1]; if (room) { this.goRoom({ item: room, isMasterDetail }); } }; findOtherRoom = (index, sign) => { const { chats } = this.state; const otherIndex = index + sign; const otherRoom = chats[otherIndex]; if (!otherRoom) { return; } if (otherRoom.separator) { return this.findOtherRoom(otherIndex, sign); } else { return otherRoom; } }; // Go to previous or next room based on sign (-1 or 1) // It's used by iPad key commands goOtherRoom = sign => { const { item } = this.state; if (!item) { return; } // Don't run during search const { search } = this.state; if (search.length > 0) { return; } const { chats } = this.state; const { isMasterDetail } = this.props; 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 }) => { const { navigation, server, isMasterDetail } = 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)) { navigation.navigate('NewServerView', { previousServer: server }); } }; onRefresh = () => { const { searching } = this.state; const { roomsRequest } = this.props; if (searching) { return; } roomsRequest({ allData: true }); }; onEndReached = () => { // Run only when we're not grouping by anything if (!this.isGrouping) { this.getSubscriptions(); } }; getScrollRef = ref => (this.scroll = ref); renderListHeader = () => { const { searching } = this.state; const { sortBy, queueSize, inquiryEnabled, encryptionBanner, user } = this.props; return ( ); }; renderHeader = () => { const { isMasterDetail } = this.props; if (!isMasterDetail) { return null; } const options = this.getHeader(); return
; }; renderItem = ({ item }) => { if (item.separator) { return this.renderSectionHeader(item.rid); } const { item: currentItem } = this.state; const { user: { username }, StoreLastMessage, useRealName, theme, isMasterDetail, width } = this.props; const id = this.getUidDirectMessage(item); return ( ); }; renderSectionHeader = header => { const { theme } = this.props; return ( {I18n.t(header)} ); }; renderScroll = () => { const { loading, chats, search, searching } = this.state; const { theme, refreshing } = this.props; if (loading) { return ; } return ( } windowSize={9} onEndReached={this.onEndReached} onEndReachedThreshold={0.5} /> ); }; render = () => { console.count(`${this.constructor.name}.render calls`); const { sortBy, groupByType, showFavorites, showUnread, showServerDropdown, showSortDropdown, theme, navigation } = this.props; return ( {this.renderHeader()} {this.renderScroll()} {showSortDropdown ? ( ) : null} {showServerDropdown ? : null} ); }; } const mapStateToProps = state => ({ 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, showSortDropdown: state.rooms.showSortDropdown, 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, StoreLastMessage: state.settings.Store_Last_Message, rooms: state.room.rooms, queueSize: getInquiryQueueSelector(state).length, inquiryEnabled: state.inquiry.enabled, encryptionBanner: state.encryption.banner }); const mapDispatchToProps = dispatch => ({ toggleSortDropdown: () => dispatch(toggleSortDropdownAction()), openSearchHeader: () => dispatch(openSearchHeaderAction()), closeSearchHeader: () => dispatch(closeSearchHeaderAction()), roomsRequest: params => dispatch(roomsRequestAction(params)), selectServerRequest: server => dispatch(selectServerRequestAction(server)), closeServerDropdown: () => dispatch(closeServerDropdownAction()) }); export default connect(mapStateToProps, mapDispatchToProps)(withDimensions(withTheme(withSafeAreaInsets(RoomsListView))));