/* eslint-disable complexity */ import { Q } from '@nozbe/watermelondb'; import { StackNavigationOptions, StackNavigationProp } from '@react-navigation/stack'; import isEmpty from 'lodash/isEmpty'; import React from 'react'; import { Share, Switch, Text, View } from 'react-native'; import { connect } from 'react-redux'; import { Observable, Subscription } from 'rxjs'; import { CompositeNavigationProp } from '@react-navigation/native'; import { leaveRoom } from '../../actions/room'; import { setLoading } from '../../actions/selectedUsers'; import Avatar from '../../containers/Avatar'; import * as HeaderButton from '../../containers/HeaderButton'; import * as List from '../../containers/List'; import { MarkdownPreview } from '../../containers/markdown'; import RoomTypeIcon from '../../containers/RoomTypeIcon'; import SafeAreaView from '../../containers/SafeAreaView'; import Status from '../../containers/Status'; import StatusBar from '../../containers/StatusBar'; import { IApplicationState, IBaseScreen, ISubscription, IUser, SubscriptionType, TSubscriptionModel } from '../../definitions'; import { withDimensions } from '../../dimensions'; import I18n from '../../i18n'; import database from '../../lib/database'; import protectedFunction from '../../lib/methods/helpers/protectedFunction'; import { getUserSelector } from '../../selectors/login'; import { ChatsStackParamList } from '../../stacks/types'; import { withTheme } from '../../theme'; import { showConfirmationAlert, showErrorAlert } from '../../lib/methods/helpers/info'; import log, { events, logEvent } from '../../lib/methods/helpers/log'; import Touch from '../../containers/Touch'; import sharedStyles from '../Styles'; import styles from './styles'; import { ERoomType } from '../../definitions/ERoomType'; import { E2E_ROOM_TYPES, SWITCH_TRACK_COLOR, themes } from '../../lib/constants'; import { callJitsi, getPermalinkChannel } from '../../lib/methods'; import { canAutoTranslate as canAutoTranslateMethod, getRoomAvatar, getRoomTitle, getUidDirectMessage, hasPermission, isGroupChat, compareServerVersion } from '../../lib/methods/helpers'; import { Services } from '../../lib/services'; import { getSubscriptionByRoomId } from '../../lib/database/services/Subscription'; import { IActionSheetProvider, withActionSheet } from '../../containers/ActionSheet'; import { MasterDetailInsideStackParamList } from '../../stacks/MasterDetailStack/types'; import { closeLivechat } from '../../lib/methods/helpers/closeLivechat'; import { videoConfStartAndJoin } from '../../lib/methods/videoConf'; import { ILivechatDepartment } from '../../definitions/ILivechatDepartment'; import { ILivechatTag } from '../../definitions/ILivechatTag'; interface IOnPressTouch { (item: { route?: T; params?: ChatsStackParamList[T]; event?: Function }): void; } interface IRoomActionsViewProps extends IActionSheetProvider, IBaseScreen { userId: string; jitsiEnabled: boolean; jitsiEnableTeams: boolean; jitsiEnableChannels: boolean; encryptionEnabled: boolean; fontScale: number; serverVersion: string | null; editRoomPermission?: string[]; toggleRoomE2EEncryptionPermission?: string[]; viewBroadcastMemberListPermission?: string[]; createTeamPermission?: string[]; addTeamChannelPermission?: string[]; convertTeamPermission?: string[]; viewCannedResponsesPermission?: string[]; livechatAllowManualOnHold?: boolean; livechatRequestComment?: boolean; navigation: CompositeNavigationProp< StackNavigationProp, StackNavigationProp >; videoConf_Enable_DMs: boolean; videoConf_Enable_Channels: boolean; videoConf_Enable_Groups: boolean; videoConf_Enable_Teams: boolean; } interface IRoomActionsViewState { room: TSubscriptionModel; membersCount: number; member: Partial; joined: boolean; canViewMembers: boolean; canAutoTranslate: boolean; canEdit: boolean; canToggleEncryption: boolean; canCreateTeam: boolean; canAddChannelToTeam: boolean; canConvertTeam: boolean; } class RoomActionsView extends React.Component { private mounted: boolean; private rid: string; private t: string; private joined: boolean; private omnichannelPermissions?: { canForwardGuest: boolean; canReturnQueue: boolean; canViewCannedResponse: boolean; canPlaceLivechatOnHold: boolean; }; private roomObservable?: Observable; private subscription?: Subscription; static navigationOptions = ({ navigation, isMasterDetail }: Pick): StackNavigationOptions => { const options: StackNavigationOptions = { title: I18n.t('Actions') }; if (isMasterDetail) { options.headerLeft = () => ; } return options; }; constructor(props: IRoomActionsViewProps) { super(props); this.mounted = false; const room = props.route.params?.room; const member = props.route.params?.member; this.rid = props.route.params?.rid; this.t = props.route.params?.t; this.joined = props.route.params?.joined; this.omnichannelPermissions = props.route.params?.omnichannelPermissions; this.state = { room: room || { rid: this.rid, t: this.t }, membersCount: 0, member: member || {}, joined: !!room, canViewMembers: false, canAutoTranslate: false, canEdit: false, canToggleEncryption: false, canCreateTeam: false, canAddChannelToTeam: false, canConvertTeam: false }; if (room && room.observe && room.rid) { this.roomObservable = room.observe(); this.subscription = this.roomObservable.subscribe(changes => { if (this.mounted) { this.setState({ room: changes }); } else { // @ts-ignore this.state.room = changes; } }); } } async componentDidMount() { this.mounted = true; const { room, member } = this.state; if (room.rid) { if (!room.id) { if (room.t === SubscriptionType.OMNICHANNEL) { if (!this.isOmnichannelPreview) { const result = await getSubscriptionByRoomId(room.rid); if (result) { this.setState({ room: result }); } } } else { try { const result = await Services.getChannelInfo(room.rid); if (result.success) { // @ts-ignore this.setState({ room: { ...result.channel, rid: result.channel._id } }); } } catch (e) { log(e); } } } if (room && room.t !== 'd' && (await this.canViewMembers())) { try { const counters = await Services.getRoomCounters(room.rid, room.t as any); if (counters.success) { this.setState({ membersCount: counters.members, joined: counters.joined }); } } catch (e) { log(e); } } else if (room.t === 'd' && isEmpty(member)) { this.updateRoomMember(); } const canAutoTranslate = canAutoTranslateMethod(); const canEdit = await this.canEdit(); const canToggleEncryption = await this.canToggleEncryption(); const canViewMembers = await this.canViewMembers(); const canCreateTeam = await this.canCreateTeam(); const canAddChannelToTeam = await this.canAddChannelToTeam(); const canConvertTeam = await this.canConvertTeam(); this.setState({ canAutoTranslate, canEdit, canToggleEncryption, canViewMembers, canCreateTeam, canAddChannelToTeam, canConvertTeam }); } } componentWillUnmount() { if (this.subscription && this.subscription.unsubscribe) { this.subscription.unsubscribe(); } } get isOmnichannelPreview() { const { room } = this.state; return room.t === 'l' && room.status === 'queued' && !this.joined; } onPressTouchable: IOnPressTouch = (item: { route?: keyof ChatsStackParamList; params?: ChatsStackParamList[keyof ChatsStackParamList]; event?: Function; }) => { const { route, event, params } = item; if (route) { /** * TODO: params can vary too much and ts is going to be happy * Instead of playing with this, we should think on a better `logEvent` function */ // @ts-ignore logEvent(events[`RA_GO_${route.replace('View', '').toUpperCase()}${params.name ? params.name.toUpperCase() : ''}`]); const { navigation } = this.props; navigation.navigate(route, params); } if (event) { return event(); } }; canEdit = async () => { const { room } = this.state; const { editRoomPermission } = this.props; const { rid } = room; const permissions = await hasPermission([editRoomPermission], rid); const canEdit = permissions[0]; return canEdit; }; canCreateTeam = async () => { const { room } = this.state; const { createTeamPermission } = this.props; const { rid } = room; const permissions = await hasPermission([createTeamPermission], rid); const canCreateTeam = permissions[0]; return canCreateTeam; }; canAddChannelToTeam = async () => { const { room } = this.state; const { addTeamChannelPermission } = this.props; const { rid } = room; const permissions = await hasPermission([addTeamChannelPermission], rid); const canAddChannelToTeam = permissions[0]; return canAddChannelToTeam; }; canConvertTeam = async () => { const { room } = this.state; const { convertTeamPermission } = this.props; const { rid } = room; const permissions = await hasPermission([convertTeamPermission], rid); const canConvertTeam = permissions[0]; return canConvertTeam; }; canToggleEncryption = async () => { const { room } = this.state; const { toggleRoomE2EEncryptionPermission } = this.props; const { rid } = room; const permissions = await hasPermission([toggleRoomE2EEncryptionPermission], rid); const canToggleEncryption = permissions[0]; return canToggleEncryption; }; canViewMembers = async () => { const { room } = this.state; const { viewBroadcastMemberListPermission } = this.props; const { rid, t, broadcast } = room; if (broadcast) { const permissions = await hasPermission([viewBroadcastMemberListPermission], rid); if (!permissions[0]) { return false; } } // This method is executed only in componentDidMount and returns a value // We save the state to read in render const result = t === 'c' || t === 'p'; return result; }; renderEncryptedSwitch = () => { const { room, canToggleEncryption, canEdit } = this.state; const { encrypted } = room; const { serverVersion } = this.props; let hasPermission = false; if (compareServerVersion(serverVersion, 'lowerThan', '3.11.0')) { hasPermission = canEdit; } else { hasPermission = canToggleEncryption; } return ( ); }; closeLivechat = async () => { const { room: { rid, departmentId } } = this.state; const { livechatRequestComment, isMasterDetail, navigation } = this.props; let departmentInfo: ILivechatDepartment | undefined; let tagsList: ILivechatTag[] | undefined; if (departmentId) { const result = await Services.getDepartmentInfo(departmentId); if (result.success) { departmentInfo = result.department as ILivechatDepartment; } } if (departmentInfo?.requestTagBeforeClosingChat) { tagsList = await Services.getTagsList(); } if (!livechatRequestComment && !departmentInfo?.requestTagBeforeClosingChat) { const comment = I18n.t('Chat_closed_by_agent'); return closeLivechat({ rid, isMasterDetail, comment }); } navigation.navigate('CloseLivechatView', { rid, departmentId, departmentInfo, tagsList }); }; placeOnHoldLivechat = () => { const { navigation } = this.props; const { room } = this.state; showConfirmationAlert({ title: I18n.t('Are_you_sure_question_mark'), message: I18n.t('Would_like_to_place_on_hold'), confirmationText: I18n.t('Yes'), onPress: async () => { try { await Services.onHoldLivechat(room.rid); navigation.navigate('RoomsListView'); } catch (e: any) { showErrorAlert(e.data?.error, I18n.t('Oops')); } } }); }; returnLivechat = () => { const { room: { rid } } = this.state; showConfirmationAlert({ message: I18n.t('Would_you_like_to_return_the_inquiry'), confirmationText: I18n.t('Yes'), onPress: async () => { try { await Services.returnLivechat(rid); } catch (e: any) { showErrorAlert(e.reason, I18n.t('Oops')); } } }); }; updateRoomMember = async () => { const { room } = this.state; try { if (!isGroupChat(room)) { const roomUserId = getUidDirectMessage(room); const result = await Services.getUserInfo(roomUserId); if (result.success) { this.setState({ member: result.user as any }); } } } catch (e) { log(e); this.setState({ member: {} }); } }; addUser = async () => { const { room } = this.state; const { dispatch, navigation } = this.props; const { rid } = room; try { dispatch(setLoading(true)); await Services.addUsersToRoom(rid); navigation.pop(); } catch (e) { log(e); } finally { dispatch(setLoading(false)); } }; toggleBlockUser = async () => { logEvent(events.RA_TOGGLE_BLOCK_USER); const { room } = this.state; const { rid, blocker } = room; const { member } = this.state; try { await Services.toggleBlockUser(rid, member._id as string, !blocker); } catch (e) { logEvent(events.RA_TOGGLE_BLOCK_USER_F); log(e); } }; toggleEncrypted = async () => { logEvent(events.RA_TOGGLE_ENCRYPTED); const { room } = this.state; const { rid } = room; const db = database.active; // Toggle encrypted value const encrypted = !room.encrypted; try { // Instantly feedback to the user await db.write(async () => { await room.update( protectedFunction((r: TSubscriptionModel) => { r.encrypted = encrypted; }) ); }); try { // Send new room setting value to server const { result } = await Services.saveRoomSettings(rid, { encrypted }); // If it was saved successfully if (result) { return; } } catch { // do nothing } // If something goes wrong we go back to the previous value await db.write(async () => { await room.update( protectedFunction((r: TSubscriptionModel) => { r.encrypted = room.encrypted; }) ); }); } catch (e) { logEvent(events.RA_TOGGLE_ENCRYPTED_F); log(e); } }; handleShare = () => { logEvent(events.RA_SHARE); const { room } = this.state; const permalink = getPermalinkChannel(room); if (!permalink) { return; } Share.share({ message: permalink }); }; leaveChannel = () => { const { room } = this.state; const { dispatch } = this.props; showConfirmationAlert({ message: I18n.t('Are_you_sure_you_want_to_leave_the_room', { room: getRoomTitle(room) }), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('leave') }), onPress: () => dispatch(leaveRoom(ERoomType.c, room)) }); }; convertTeamToChannel = async () => { const { room } = this.state; const { navigation, userId } = this.props; try { if (!room.teamId) { return; } const result = await Services.teamListRoomsOfUser({ teamId: room.teamId, userId }); if (result.success) { if (result.rooms?.length) { const teamChannels = result.rooms.map(r => ({ rid: r._id, name: r.name, teamId: r.teamId })); navigation.navigate('SelectListView', { title: 'Converting_Team_To_Channel', data: teamChannels, infoText: 'Select_Team_Channels_To_Delete', nextAction: (data: string[]) => this.convertTeamToChannelConfirmation(data) }); } else { this.convertTeamToChannelConfirmation(); } } } catch (e) { this.convertTeamToChannelConfirmation(); } }; handleConvertTeamToChannel = async (selected: string[]) => { logEvent(events.RA_CONVERT_TEAM_TO_CHANNEL); try { const { room } = this.state; const { navigation } = this.props; if (!room.teamId) { return; } const result = await Services.convertTeamToChannel({ teamId: room.teamId, selected }); if (result.success) { navigation.navigate('RoomView'); } } catch (e) { logEvent(events.RA_CONVERT_TEAM_TO_CHANNEL_F); log(e); } }; convertTeamToChannelConfirmation = (selected: string[] = []) => { showConfirmationAlert({ title: I18n.t('Confirmation'), message: I18n.t('You_are_converting_the_team'), confirmationText: I18n.t('Convert'), onPress: () => this.handleConvertTeamToChannel(selected) }); }; leaveTeam = async () => { const { room } = this.state; const { navigation, dispatch, userId } = this.props; try { if (!room.teamId) { return; } const result = await Services.teamListRoomsOfUser({ teamId: room.teamId, userId }); if (result.success) { if (result.rooms?.length) { const teamChannels = result.rooms.map(r => ({ rid: r._id, name: r.name, teamId: r.teamId, alert: r.isLastOwner })); navigation.navigate('SelectListView', { title: 'Leave_Team', data: teamChannels as any, infoText: 'Select_Team_Channels', nextAction: data => dispatch(leaveRoom(ERoomType.t, room, data)), showAlert: () => showErrorAlert(I18n.t('Last_owner_team_room'), I18n.t('Cannot_leave')) }); } else { showConfirmationAlert({ message: I18n.t('You_are_leaving_the_team', { team: getRoomTitle(room) }), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('leave') }), onPress: () => dispatch(leaveRoom(ERoomType.t, room)) }); } } } catch (e) { showConfirmationAlert({ message: I18n.t('You_are_leaving_the_team', { team: getRoomTitle(room) }), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('leave') }), onPress: () => dispatch(leaveRoom(ERoomType.t, room)) }); } }; handleConvertToTeam = async () => { logEvent(events.RA_CONVERT_TO_TEAM); try { const { room } = this.state; const { navigation } = this.props; const result = await Services.convertChannelToTeam({ rid: room.rid, name: room.name, type: room.t as any }); if (result.success) { navigation.navigate('RoomView'); } } catch (e) { logEvent(events.RA_CONVERT_TO_TEAM_F); log(e); } }; convertToTeam = () => { showConfirmationAlert({ title: I18n.t('Confirmation'), message: I18n.t('Convert_to_Team_Warning'), confirmationText: I18n.t('Convert'), onPress: () => this.handleConvertToTeam() }); }; handleMoveToTeam = async (selected: string[]) => { logEvent(events.RA_MOVE_TO_TEAM); try { const { room } = this.state; const { navigation } = this.props; const result = await Services.addRoomsToTeam({ teamId: selected?.[0], rooms: [room.rid] }); if (result.success) { navigation.navigate('RoomView'); } } catch (e) { logEvent(events.RA_MOVE_TO_TEAM_F); log(e); showErrorAlert(I18n.t('There_was_an_error_while_action', { action: I18n.t('moving_channel_to_team') })); } }; moveToTeam = async () => { try { const { navigation } = this.props; const db = database.active; const subCollection = db.get('subscriptions'); const teamRooms = await subCollection.query(Q.where('team_main', true)).fetch(); if (teamRooms.length) { const data = teamRooms.map(team => ({ rid: team.teamId as string, t: team.t, name: team.name })); navigation.navigate('SelectListView', { title: 'Move_to_Team', infoText: 'Move_Channel_Paragraph', nextAction: () => { navigation.push('SelectListView', { title: 'Select_Team', data, isRadio: true, isSearch: true, onSearch: onChangeText => this.searchTeam(onChangeText), nextAction: selected => showConfirmationAlert({ title: I18n.t('Confirmation'), message: I18n.t('Move_to_Team_Warning'), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('move') }), onPress: () => this.handleMoveToTeam(selected) }) }); } }); } } catch (e) { log(e); } }; searchTeam = async (onChangeText: string) => { logEvent(events.RA_SEARCH_TEAM); try { const { addTeamChannelPermission, createTeamPermission } = this.props; const QUERY_SIZE = 50; const db = database.active; const teams = await db .get('subscriptions') .query( Q.where('team_main', true), Q.where('name', Q.like(`%${onChangeText}%`)), Q.experimentalTake(QUERY_SIZE), Q.experimentalSortBy('room_updated_at', Q.desc) ) .fetch(); const asyncFilter = async (teamArray: TSubscriptionModel[]) => { const results = await Promise.all( teamArray.map(async team => { const permissions = await hasPermission([addTeamChannelPermission, createTeamPermission], team.rid); if (!permissions[0]) { return false; } return true; }) ); return teamArray.filter((_v, index) => results[index]); }; const teamsFiltered = await asyncFilter(teams); return teamsFiltered; } catch (e) { log(e); } }; startVideoConf = ({ video }: { video: boolean }): void => { const { room } = this.state; const { serverVersion } = this.props; if (compareServerVersion(serverVersion, 'greaterThanOrEqualTo', '5.0.0')) { videoConfStartAndJoin(room.rid, video); } else { callJitsi(room, !video); } }; renderRoomInfo = () => { const { room, member } = this.state; const { rid, name, t, topic, source } = room; const { theme, fontScale } = this.props; const avatar = getRoomAvatar(room); const isGroupChatHandler = isGroupChat(room); return ( this.onPressTouchable({ route: 'RoomInfoView', // forward room only if room isn't joined params: { rid, t, room, member } }) } style={{ backgroundColor: themes[theme].backgroundColor }} accessibilityLabel={I18n.t('Room_Info')} enabled={!isGroupChatHandler} testID='room-actions-info' > {t === 'd' && member._id ? ( ) : undefined} {room.t === 'd' ? ( {room.fname} ) : ( {getRoomTitle(room)} )} {room.t === 'd' && ( )} {isGroupChatHandler ? null : } ); }; renderJitsi = () => { const { room } = this.state; const { jitsiEnabled, jitsiEnableTeams, jitsiEnableChannels, serverVersion, videoConf_Enable_DMs, videoConf_Enable_Channels, videoConf_Enable_Groups, videoConf_Enable_Teams } = this.props; const isJitsiDisabledForTeams = room.teamMain && !jitsiEnableTeams; const isJitsiDisabledForChannels = !room.teamMain && (room.t === 'p' || room.t === 'c') && !jitsiEnableChannels; const isVideoConfDisabledForTeams = !!room.teamMain && !videoConf_Enable_Teams; const isVideoConfDisabledForChannels = !room.teamMain && room.t === 'c' && !videoConf_Enable_Channels; const isVideoConfDisabledForGroups = !room.teamMain && room.t === 'p' && !videoConf_Enable_Groups; const isVideoConfDisabledForDirect = !room.teamMain && room.t === 'd' && !videoConf_Enable_DMs; if (compareServerVersion(serverVersion, 'greaterThanOrEqualTo', '5.0.0')) { if ( isVideoConfDisabledForTeams || isVideoConfDisabledForChannels || isVideoConfDisabledForGroups || isVideoConfDisabledForDirect ) { return null; } } else if (!jitsiEnabled || isJitsiDisabledForTeams || isJitsiDisabledForChannels) { return null; } return ( this.startVideoConf({ video: false })} testID='room-actions-voice' left={() => } showActionIndicator /> this.startVideoConf({ video: true })} testID='room-actions-video' left={() => } showActionIndicator /> ); }; renderE2EEncryption = () => { const { room } = this.state; const { encryptionEnabled } = this.props; // If this room type can be encrypted // If e2e is enabled if (E2E_ROOM_TYPES[room.t] && encryptionEnabled) { return ( } right={this.renderEncryptedSwitch} /> ); } return null; }; renderLastSection = () => { const { room, joined } = this.state; const { theme } = this.props; const { t, blocker } = room; if (!joined || t === 'l') { return null; } if (t === 'd' && !isGroupChat(room)) { return ( this.onPressTouchable({ event: this.toggleBlockUser }) } testID='room-actions-block-user' left={() => } showActionIndicator color={themes[theme].dangerColor} /> ); } if (t === 'p' || t === 'c') { return ( this.onPressTouchable({ event: room.teamMain ? this.leaveTeam : this.leaveChannel }) } testID='room-actions-leave-channel' left={() => } showActionIndicator color={themes[theme].dangerColor} /> ); } return null; }; teamChannelActions = (t: string, room: ISubscription) => { const { canEdit, canCreateTeam, canAddChannelToTeam } = this.state; const canConvertToTeam = canEdit && canCreateTeam && !room.teamMain; const canMoveToTeam = canEdit && canAddChannelToTeam && !room.teamId; return ( <> {['c', 'p'].includes(t) && canConvertToTeam ? ( <> this.onPressTouchable({ event: this.convertToTeam }) } testID='room-actions-convert-to-team' left={() => } showActionIndicator /> ) : null} {['c', 'p'].includes(t) && canMoveToTeam ? ( <> this.onPressTouchable({ event: this.moveToTeam }) } testID='room-actions-move-to-team' left={() => } showActionIndicator /> ) : null} ); }; teamToChannelActions = (t: string, room: ISubscription) => { const { canEdit, canConvertTeam } = this.state; const canConvertTeamToChannel = canEdit && canConvertTeam && !!room?.teamMain; return ( <> {['c', 'p'].includes(t) && canConvertTeamToChannel ? ( <> this.onPressTouchable({ event: this.convertTeamToChannel }) } left={() => } showActionIndicator /> ) : null} ); }; renderOmnichannelSection = () => { const { room } = this.state; const { rid, t } = room; const { theme } = this.props; if (t !== 'l' || this.isOmnichannelPreview) { return null; } return ( {this.omnichannelPermissions?.canForwardGuest ? ( <> this.onPressTouchable({ route: 'ForwardLivechatView', params: { rid } }) } left={() => } showActionIndicator /> ) : null} {this.omnichannelPermissions?.canPlaceLivechatOnHold ? ( <> this.onPressTouchable({ event: this.placeOnHoldLivechat }) } left={() => } showActionIndicator /> ) : null} {this.omnichannelPermissions?.canReturnQueue ? ( <> this.onPressTouchable({ event: this.returnLivechat }) } left={() => } showActionIndicator /> ) : null} <> this.onPressTouchable({ event: this.closeLivechat }) } left={() => } showActionIndicator /> ); }; render() { const { room, membersCount, canViewMembers, joined, canAutoTranslate } = this.state; const { rid, t, prid } = room; const isGroupChatHandler = isGroupChat(room); return ( {this.renderRoomInfo()} {this.renderJitsi()} {this.renderE2EEncryption()} {(['c', 'p'].includes(t) && canViewMembers) || isGroupChatHandler ? ( <> 0 ? `${membersCount} ${I18n.t('members')}` : undefined} onPress={() => this.onPressTouchable({ route: 'RoomMembersView', params: { rid, room, joined: this.joined } })} testID='room-actions-members' left={() => } showActionIndicator translateSubtitle={false} /> ) : null} {['c', 'p', 'd'].includes(t) && !prid ? ( <> this.onPressTouchable({ route: 'DiscussionsView', params: { rid, t } }) } testID='room-actions-discussions' left={() => } showActionIndicator /> ) : null} {['l'].includes(t) && !this.isOmnichannelPreview && this.omnichannelPermissions?.canViewCannedResponse ? ( <> this.onPressTouchable({ route: 'CannedResponsesListView', params: { rid } })} left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) ? ( <> this.onPressTouchable({ route: 'MessagesView', params: { rid, t, name: 'Files' } }) } testID='room-actions-files' left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) ? ( <> this.onPressTouchable({ route: 'MessagesView', params: { rid, t, name: 'Mentions' } }) } testID='room-actions-mentioned' left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) ? ( <> this.onPressTouchable({ route: 'MessagesView', params: { rid, t, name: 'Starred' } }) } testID='room-actions-starred' left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) ? ( <> this.onPressTouchable({ event: this.handleShare }) } testID='room-actions-share' left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) ? ( <> this.onPressTouchable({ route: 'MessagesView', params: { rid, t, name: 'Pinned' } }) } testID='room-actions-pinned' left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) && canAutoTranslate ? ( <> this.onPressTouchable({ route: 'AutoTranslateView', params: { rid, room } }) } testID='room-actions-auto-translate' left={() => } showActionIndicator /> ) : null} {['c', 'p', 'd'].includes(t) && joined ? ( <> this.onPressTouchable({ route: 'NotificationPrefView', params: { rid, room } }) } testID='room-actions-notifications' left={() => } showActionIndicator /> ) : null} {this.teamChannelActions(t, room)} {this.teamToChannelActions(t, room)} {this.renderOmnichannelSection()} {this.renderLastSection()} ); } } const mapStateToProps = (state: IApplicationState) => ({ userId: getUserSelector(state).id, jitsiEnabled: (state.settings.Jitsi_Enabled || false) as boolean, jitsiEnableTeams: (state.settings.Jitsi_Enable_Teams || false) as boolean, jitsiEnableChannels: (state.settings.Jitsi_Enable_Channels || false) as boolean, videoConf_Enable_DMs: (state.settings.VideoConf_Enable_DMs ?? true) as boolean, videoConf_Enable_Channels: (state.settings.VideoConf_Enable_Channels ?? true) as boolean, videoConf_Enable_Groups: (state.settings.VideoConf_Enable_Groups ?? true) as boolean, videoConf_Enable_Teams: (state.settings.VideoConf_Enable_Teams ?? true) as boolean, encryptionEnabled: state.encryption.enabled, serverVersion: state.server.version, isMasterDetail: state.app.isMasterDetail, editRoomPermission: state.permissions['edit-room'], toggleRoomE2EEncryptionPermission: state.permissions['toggle-room-e2e-encryption'], viewBroadcastMemberListPermission: state.permissions['view-broadcast-member-list'], createTeamPermission: state.permissions['create-team'], addTeamChannelPermission: state.permissions['add-team-channel'], convertTeamPermission: state.permissions['convert-team'], viewCannedResponsesPermission: state.permissions['view-canned-responses'], livechatAllowManualOnHold: state.settings.Livechat_allow_manual_on_hold as boolean, livechatRequestComment: state.settings.Livechat_request_comment_when_closing_conversation as boolean }); export default connect(mapStateToProps)(withTheme(withActionSheet(withDimensions(RoomActionsView))));