import React from 'react'; import PropTypes from 'prop-types'; import { View, Text, Share, Switch } from 'react-native'; import { connect } from 'react-redux'; import isEmpty from 'lodash/isEmpty'; import { Q } from '@nozbe/watermelondb'; import { compareServerVersion, methods } from '../../lib/utils'; import Touch from '../../utils/touch'; import { setLoading as setLoadingAction } from '../../actions/selectedUsers'; import { leaveRoom as leaveRoomAction, closeRoom as closeRoomAction } from '../../actions/room'; import styles from './styles'; import sharedStyles from '../Styles'; import Avatar from '../../containers/Avatar'; import Status from '../../containers/Status'; import * as List from '../../containers/List'; import RocketChat from '../../lib/rocketchat'; import log, { logEvent, events } from '../../utils/log'; import RoomTypeIcon from '../../containers/RoomTypeIcon'; import I18n from '../../i18n'; import StatusBar from '../../containers/StatusBar'; import { themes, SWITCH_TRACK_COLOR } from '../../constants/colors'; import { withTheme } from '../../theme'; import * as HeaderButton from '../../containers/HeaderButton'; import Markdown from '../../containers/markdown'; import { showConfirmationAlert, showErrorAlert } from '../../utils/info'; import SafeAreaView from '../../containers/SafeAreaView'; import { E2E_ROOM_TYPES } from '../../lib/encryption/constants'; import protectedFunction from '../../lib/methods/helpers/protectedFunction'; import database from '../../lib/database'; import { withDimensions } from '../../dimensions'; class RoomActionsView extends React.Component { static navigationOptions = ({ navigation, isMasterDetail }) => { const options = { title: I18n.t('Actions') }; if (isMasterDetail) { options.headerLeft = () => ; } return options; } static propTypes = { navigation: PropTypes.object, route: PropTypes.object, leaveRoom: PropTypes.func, jitsiEnabled: PropTypes.bool, encryptionEnabled: PropTypes.bool, setLoadingInvite: PropTypes.func, closeRoom: PropTypes.func, theme: PropTypes.string, fontScale: PropTypes.number, serverVersion: PropTypes.string, isMasterDetail: PropTypes.bool, addUserToJoinedRoomPermission: PropTypes.array, addUserToAnyCRoomPermission: PropTypes.array, addUserToAnyPRoomPermission: PropTypes.array, createInviteLinksPermission: PropTypes.array, editRoomPermission: PropTypes.array, toggleRoomE2EEncryptionPermission: PropTypes.array, viewBroadcastMemberListPermission: PropTypes.array, transferLivechatGuestPermission: PropTypes.array, createTeamPermission: PropTypes.array, addTeamChannelPermission: PropTypes.array } constructor(props) { 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.state = { room: room || { rid: this.rid, t: this.t }, membersCount: 0, member: member || {}, joined: !!room, canViewMembers: false, canAutoTranslate: false, canAddUser: false, canInviteUser: false, canForwardGuest: false, canReturnQueue: false, canEdit: false, canToggleEncryption: false, canCreateTeam: false, canAddChannelToTeam: 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 { this.state.room = changes; } }); } } async componentDidMount() { this.mounted = true; const { room, member } = this.state; if (room.rid) { if (!room.id && !this.isOmnichannelPreview) { try { const result = await RocketChat.getChannelInfo(room.rid); if (result.success) { this.setState({ room: { ...result.channel, rid: result.channel._id } }); } } catch (e) { log(e); } } if (room && room.t !== 'd' && this.canViewMembers()) { try { const counters = await RocketChat.getRoomCounters(room.rid, room.t); 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 = await RocketChat.canAutoTranslate(); const canAddUser = await this.canAddUser(); const canInviteUser = await this.canInviteUser(); 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(); this.setState({ canAutoTranslate, canAddUser, canInviteUser, canEdit, canToggleEncryption, canViewMembers, canCreateTeam, canAddChannelToTeam }); // livechat permissions if (room.t === 'l') { const canForwardGuest = await this.canForwardGuest(); const canReturnQueue = await this.canReturnQueue(); this.setState({ canForwardGuest, canReturnQueue }); } } } componentWillUnmount() { if (this.subscription && this.subscription.unsubscribe) { this.subscription.unsubscribe(); } } get isOmnichannelPreview() { const { room } = this.state; return room.t === 'l' && room.status === 'queued'; } onPressTouchable = (item) => { const { route, event, params } = item; if (route) { 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(); } } canAddUser = async() => { const { room, joined } = this.state; const { addUserToJoinedRoomPermission, addUserToAnyCRoomPermission, addUserToAnyPRoomPermission } = this.props; const { rid, t } = room; let canAddUser = false; const userInRoom = joined; const permissions = await RocketChat.hasPermission([addUserToJoinedRoomPermission, addUserToAnyCRoomPermission, addUserToAnyPRoomPermission], rid); if (userInRoom && permissions[0]) { canAddUser = true; } if (t === 'c' && permissions[1]) { canAddUser = true; } if (t === 'p' && permissions[2]) { canAddUser = true; } return canAddUser; } canInviteUser = async() => { const { room } = this.state; const { createInviteLinksPermission } = this.props; const { rid } = room; const permissions = await RocketChat.hasPermission([createInviteLinksPermission], rid); const canInviteUser = permissions[0]; return canInviteUser; } canEdit = async() => { const { room } = this.state; const { editRoomPermission } = this.props; const { rid } = room; const permissions = await RocketChat.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 RocketChat.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 RocketChat.hasPermission([addTeamChannelPermission], rid); const canAddChannelToTeam = permissions[0]; return canAddChannelToTeam; } canToggleEncryption = async() => { const { room } = this.state; const { toggleRoomE2EEncryptionPermission } = this.props; const { rid } = room; const permissions = await RocketChat.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 RocketChat.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; } canForwardGuest = async() => { const { room } = this.state; const { transferLivechatGuestPermission } = this.props; const { rid } = room; const permissions = await RocketChat.hasPermission([transferLivechatGuestPermission], rid); return permissions[0]; } canReturnQueue = async() => { try { const { returnQueue } = await RocketChat.getRoutingConfig(); return returnQueue; } catch { // do nothing } } renderEncryptedSwitch = () => { const { room, canToggleEncryption, canEdit } = this.state; const { encrypted } = room; const { serverVersion } = this.props; let hasPermission = false; if (compareServerVersion(serverVersion, '3.11.0', methods.lowerThan)) { hasPermission = canEdit; } else { hasPermission = canToggleEncryption; } return ( ); } closeLivechat = () => { const { room: { rid } } = this.state; const { closeRoom } = this.props; closeRoom(rid); } 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 RocketChat.returnLivechat(rid); } catch (e) { showErrorAlert(e.reason, I18n.t('Oops')); } } }); } updateRoomMember = async() => { const { room } = this.state; try { if (!RocketChat.isGroupChat(room)) { const roomUserId = RocketChat.getUidDirectMessage(room); const result = await RocketChat.getUserInfo(roomUserId); if (result.success) { this.setState({ member: result.user }); } } } catch (e) { log(e); this.setState({ member: {} }); } } addUser = async() => { const { room } = this.state; const { setLoadingInvite, navigation } = this.props; const { rid } = room; try { setLoadingInvite(true); await RocketChat.addUsersToRoom(rid); navigation.pop(); } catch (e) { log(e); } finally { setLoadingInvite(false); } } toggleBlockUser = async() => { logEvent(events.RA_TOGGLE_BLOCK_USER); const { room } = this.state; const { rid, blocker } = room; const { member } = this.state; try { await RocketChat.toggleBlockUser(rid, member._id, !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.action(async() => { await room.update(protectedFunction((r) => { r.encrypted = encrypted; })); }); try { // Send new room setting value to server const { result } = await RocketChat.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.action(async() => { await room.update(protectedFunction((r) => { 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 = RocketChat.getPermalinkChannel(room); if (!permalink) { return; } Share.share({ message: permalink }); }; leaveChannel = () => { const { room } = this.state; const { leaveRoom } = this.props; showConfirmationAlert({ message: I18n.t('Are_you_sure_you_want_to_leave_the_room', { room: RocketChat.getRoomTitle(room) }), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('leave') }), onPress: () => leaveRoom(room.rid, room.t) }); } handleLeaveTeam = async(selected) => { logEvent(events.RA_LEAVE_TEAM); try { const { room } = this.state; const { navigation, isMasterDetail } = this.props; const result = await RocketChat.leaveTeam({ teamName: room.name, ...(selected && { rooms: selected }) }); if (result.success) { if (isMasterDetail) { navigation.navigate('DrawerNavigator'); } else { navigation.navigate('RoomsListView'); } } } catch (e) { logEvent(events.RA_LEAVE_TEAM_F); log(e); showErrorAlert( e.data.error ? I18n.t(e.data.error) : I18n.t('There_was_an_error_while_action', { action: I18n.t('leaving_team') }), I18n.t('Cannot_leave') ); } } leaveTeam = async() => { const { room } = this.state; const { navigation } = this.props; try { const db = database.active; const subCollection = db.get('subscriptions'); const rooms = await subCollection.query( Q.where('team_id', Q.eq(room.teamId)), Q.where('team_main', Q.notEq(true)) ); if (rooms.length) { const teamChannels = rooms.map(r => ({ rid: r.id, name: r.name, teamId: r.teamId, alert: r.isLastOwner })); navigation.navigate('SelectListView', { title: 'Leave_Team', data: teamChannels, infoText: 'Select_Team_Channels', nextAction: data => this.handleLeaveTeam(data), showAlert: () => showErrorAlert(I18n.t('Last_owner_team_room'), I18n.t('Cannot_leave')) }); } else { showConfirmationAlert({ message: I18n.t('You_are_leaving_the_team', { team: RocketChat.getRoomTitle(room) }), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('leave') }), onPress: () => this.handleLeaveTeam() }); } } catch (e) { showConfirmationAlert({ message: I18n.t('You_are_leaving_the_team', { team: RocketChat.getRoomTitle(room) }), confirmationText: I18n.t('Yes_action_it', { action: I18n.t('leave') }), onPress: () => this.handleLeaveTeam() }); } } handleConvertToTeam = async() => { logEvent(events.RA_CONVERT_TO_TEAM); try { const { room } = this.state; const { navigation } = this.props; const result = await RocketChat.convertChannelToTeam({ rid: room.rid, name: room.name, type: room.t }); 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) => { logEvent(events.RA_MOVE_TO_TEAM); try { const { room } = this.state; const { navigation } = this.props; const result = await RocketChat.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', Q.notEq(null)) ); if (teamRooms.length) { const data = teamRooms.map(team => ({ rid: team.teamId, 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) => { logEvent(events.RA_SEARCH_TEAM); try { const { addTeamChannelPermission, createTeamPermission } = this.props; const QUERY_SIZE = 50; const db = database.active; const teams = await db.collections .get('subscriptions') .query( Q.where('team_main', Q.notEq(null)), Q.where('name', Q.like(`%${ onChangeText }%`)), Q.experimentalTake(QUERY_SIZE), Q.experimentalSortBy('room_updated_at', Q.desc) ); const asyncFilter = async(teamArray) => { const results = await Promise.all(teamArray.map(async(team) => { const permissions = await RocketChat.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); } } renderRoomInfo = () => { const { room, member } = this.state; const { rid, name, t, topic } = room; const { theme, fontScale } = this.props; const avatar = RocketChat.getRoomAvatar(room); const isGroupChat = RocketChat.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')} accessibilityTraits='button' enabled={!isGroupChat} testID='room-actions-info' theme={theme} > {t === 'd' && member._id ? ( ) : null } {room.t === 'd' ? {room.fname} : ( {RocketChat.getRoomTitle(room)} ) } {room.t === 'd' && } {isGroupChat ? null : } ); } renderJitsi = () => { const { room } = this.state; const { jitsiEnabled } = this.props; if (!jitsiEnabled || room.teamMain) { return null; } return ( RocketChat.callJitsi(room, true)} testID='room-actions-voice' left={() => } showActionIndicator /> RocketChat.callJitsi(room)} 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') { 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} /> ); } } teamChannelActions = (t, room) => { 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} ); } render() { const { room, membersCount, canViewMembers, canAddUser, canInviteUser, joined, canAutoTranslate, canForwardGuest, canReturnQueue } = this.state; const { rid, t } = room; const isGroupChat = RocketChat.isGroupChat(room); return ( {this.renderRoomInfo()} {this.renderJitsi()} {this.renderE2EEncryption()} {(['c', 'p'].includes(t) && canViewMembers) || isGroupChat ? ( <> 0 ? `${ membersCount } ${ I18n.t('members') }` : null} onPress={() => this.onPressTouchable({ route: 'RoomMembersView', params: { rid, room } })} testID='room-actions-members' left={() => } showActionIndicator translateSubtitle={false} /> ) : null} {['c', 'p'].includes(t) && canAddUser ? ( <> this.onPressTouchable({ route: 'SelectedUsersView', params: { rid, title: I18n.t('Add_users'), nextAction: this.addUser } })} testID='room-actions-add-user' left={() => } showActionIndicator /> ) : null} {['c', 'p'].includes(t) && canInviteUser ? ( <> this.onPressTouchable({ route: 'InviteUsersView', params: { rid } })} testID='room-actions-invite-user' 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) } {['l'].includes(t) && !this.isOmnichannelPreview ? ( <> this.onPressTouchable({ event: this.closeLivechat })} left={() => } showActionIndicator /> ) : null} {['l'].includes(t) && !this.isOmnichannelPreview && canForwardGuest ? ( <> this.onPressTouchable({ route: 'ForwardLivechatView', params: { rid } })} left={() => } showActionIndicator /> ) : null} {['l'].includes(t) && !this.isOmnichannelPreview && canReturnQueue ? ( <> this.onPressTouchable({ event: this.returnLivechat })} left={() => } showActionIndicator /> ) : null} {['l'].includes(t) && !this.isOmnichannelPreview ? ( <> this.onPressTouchable({ route: 'VisitorNavigationView', params: { rid } })} left={() => } showActionIndicator /> ) : null} {this.renderLastSection()} ); } } const mapStateToProps = state => ({ jitsiEnabled: state.settings.Jitsi_Enabled || false, encryptionEnabled: state.encryption.enabled, serverVersion: state.server.version, isMasterDetail: state.app.isMasterDetail, addUserToJoinedRoomPermission: state.permissions['add-user-to-joined-room'], addUserToAnyCRoomPermission: state.permissions['add-user-to-any-c-room'], addUserToAnyPRoomPermission: state.permissions['add-user-to-any-p-room'], createInviteLinksPermission: state.permissions['create-invite-links'], editRoomPermission: state.permissions['edit-room'], toggleRoomE2EEncryptionPermission: state.permissions['toggle-room-e2e-encryption'], viewBroadcastMemberListPermission: state.permissions['view-broadcast-member-list'], transferLivechatGuestPermission: state.permissions['transfer-livechat-guest'], createTeamPermission: state.permissions['create-team'], addTeamChannelPermission: state.permissions['add-team-channel'] }); const mapDispatchToProps = dispatch => ({ leaveRoom: (rid, t) => dispatch(leaveRoomAction(rid, t)), closeRoom: rid => dispatch(closeRoomAction(rid)), setLoadingInvite: loading => dispatch(setLoadingAction(loading)) }); export default connect(mapStateToProps, mapDispatchToProps)(withTheme(withDimensions(RoomActionsView)));