import React from 'react'; import PropTypes from 'prop-types'; import { Text, View, LayoutAnimation, ActivityIndicator, Platform } from 'react-native'; import { connect, Provider } from 'react-redux'; import { RectButton, gestureHandlerRootHOC } from 'react-native-gesture-handler'; import { Navigation } from 'react-native-navigation'; import SafeAreaView from 'react-native-safe-area-view'; import { openRoom as openRoomAction, closeRoom as closeRoomAction, setLastOpen as setLastOpenAction } from '../../actions/room'; import { toggleReactionPicker as toggleReactionPickerAction, actionsShow as actionsShowAction } from '../../actions/messages'; import LoggedView from '../View'; import { List } from './ListView'; import database from '../../lib/realm'; import RocketChat from '../../lib/rocketchat'; import Message from '../../containers/message'; import MessageActions from '../../containers/MessageActions'; import MessageErrorActions from '../../containers/MessageErrorActions'; import MessageBox from '../../containers/MessageBox'; import ReactionPicker from './ReactionPicker'; import UploadProgress from './UploadProgress'; import styles from './styles'; import log from '../../utils/log'; import I18n from '../../i18n'; import { iconsMap } from '../../Icons'; import store from '../../lib/createStore'; import ConnectionBadge from '../../containers/ConnectionBadge'; import { DEFAULT_HEADER } from '../../constants/headerOptions'; let RoomActionsView = null; @connect(state => ({ user: { id: state.login.user && state.login.user.id, username: state.login.user && state.login.user.username, token: state.login.user && state.login.user.token }, actionMessage: state.messages.actionMessage, showActions: state.messages.showActions, showErrorActions: state.messages.showErrorActions, appState: state.app.ready && state.app.foreground ? 'foreground' : 'background' }), dispatch => ({ openRoom: room => dispatch(openRoomAction(room)), setLastOpen: date => dispatch(setLastOpenAction(date)), toggleReactionPicker: message => dispatch(toggleReactionPickerAction(message)), actionsShow: actionMessage => dispatch(actionsShowAction(actionMessage)), closeRoom: () => dispatch(closeRoomAction()) })) /** @extends React.Component */ export default class RoomView extends LoggedView { static options() { return { ...DEFAULT_HEADER, topBar: { ...DEFAULT_HEADER.topBar, title: { component: { name: 'RoomHeaderView', alignment: 'left' } }, rightButtons: [{ id: 'more', testID: 'room-view-header-actions', icon: iconsMap.more }, { id: 'star', testID: 'room-view-header-star', icon: iconsMap.starOutline }] }, blurOnUnmount: true }; } static propTypes = { componentId: PropTypes.string, openRoom: PropTypes.func.isRequired, setLastOpen: PropTypes.func.isRequired, user: PropTypes.shape({ id: PropTypes.string.isRequired, username: PropTypes.string.isRequired, token: PropTypes.string.isRequired }), rid: PropTypes.string, name: PropTypes.string, t: PropTypes.string, showActions: PropTypes.bool, showErrorActions: PropTypes.bool, actionMessage: PropTypes.object, appState: PropTypes.string, toggleReactionPicker: PropTypes.func.isRequired, actionsShow: PropTypes.func, closeRoom: PropTypes.func }; constructor(props) { super('RoomView', props); this.rid = props.rid; this.rooms = database.objects('subscriptions').filtered('rid = $0', this.rid); this.state = { loaded: false, joined: this.rooms.length > 0, room: {} }; this.focused = true; this.onReactionPress = this.onReactionPress.bind(this); Navigation.events().bindComponent(this); } componentDidMount() { if (this.rooms.length === 0 && this.rid) { const { rid, name, t } = this.props; this.setState( { room: { rid, name, t } }, () => this.updateRoom() ); } this.rooms.addListener(this.updateRoom); this.internalSetState({ loaded: true }); } shouldComponentUpdate(nextProps, nextState) { const { room, loaded, joined } = this.state; const { showActions, showErrorActions, appState } = this.props; if (room.ro !== nextState.room.ro) { return true; } else if (room.f !== nextState.room.f) { return true; } else if (room.blocked !== nextState.room.blocked) { return true; } else if (room.blocker !== nextState.room.blocker) { return true; } else if (room.archived !== nextState.room.archived) { return true; } else if (loaded !== nextState.loaded) { return true; } else if (joined !== nextState.joined) { return true; } else if (showActions !== nextProps.showActions) { return true; } else if (showErrorActions !== nextProps.showErrorActions) { return true; } else if (appState !== nextProps.appState) { return true; } return false; } componentDidUpdate(prevProps, prevState) { const { room } = this.state; const { componentId, appState } = this.props; if (prevState.room.f !== room.f) { const rightButtons = [{ id: 'star', testID: 'room-view-header-star', icon: room.f ? iconsMap.star : iconsMap.starOutline }]; if (room.t !== 'l') { rightButtons.unshift({ id: 'more', testID: 'room-view-header-actions', icon: iconsMap.more }); } Navigation.mergeOptions(componentId, { topBar: { rightButtons } }); } else if (appState === 'foreground' && appState !== prevProps.appState) { RocketChat.loadMissedMessages(room).catch(e => console.log(e)); RocketChat.readMessages(room.rid).catch(e => console.log(e)); } } componentWillUnmount() { const { closeRoom } = this.props; closeRoom(); this.rooms.removeAllListeners(); } // eslint-disable-next-line componentDidAppear() { this.focused = true; } // eslint-disable-next-line componentDidDisappear() { this.focused = false; } onMessageLongPress = (message) => { const { actionsShow } = this.props; actionsShow(message); } onReactionPress = (shortname, messageId) => { const { actionMessage, toggleReactionPicker } = this.props; try { if (!messageId) { RocketChat.setReaction(shortname, actionMessage._id); return toggleReactionPicker(); } RocketChat.setReaction(shortname, messageId); } catch (e) { log('RoomView.onReactionPress', e); } }; internalSetState = (...args) => { if (Platform.OS === 'ios') { LayoutAnimation.easeInEaseOut(); } this.setState(...args); } navigationButtonPressed = ({ buttonId }) => { const { room } = this.state; const { rid, f } = room; const { componentId } = this.props; if (buttonId === 'more') { if (RoomActionsView == null) { RoomActionsView = require('../RoomActionsView').default; Navigation.registerComponentWithRedux('RoomActionsView', () => gestureHandlerRootHOC(RoomActionsView), Provider, store); } Navigation.push(componentId, { component: { id: 'RoomActionsView', name: 'RoomActionsView', passProps: { rid } } }); } else if (buttonId === 'star') { try { RocketChat.toggleFavorite(rid, !f); } catch (e) { log('toggleFavorite', e); } } } // eslint-disable-next-line react/sort-comp updateRoom = () => { const { openRoom, setLastOpen } = this.props; if (!this.focused) { return; } if (this.rooms.length > 0) { const { room: prevRoom } = this.state; const room = JSON.parse(JSON.stringify(this.rooms[0] || {})); this.internalSetState({ room }); if (!prevRoom._id) { openRoom({ ...room }); if (room.alert || room.unread || room.userMentions) { setLastOpen(room.ls); } else { setLastOpen(null); } } } else { const { room } = this.state; if (room.rid) { openRoom(room); this.internalSetState({ joined: false }); } } } sendMessage = (message) => { const { setLastOpen } = this.props; LayoutAnimation.easeInEaseOut(); RocketChat.sendMessage(this.rid, message).then(() => { setLastOpen(null); }); }; joinRoom = async() => { const { rid } = this.props; try { const result = await RocketChat.joinRoom(rid); if (result.success) { this.internalSetState({ joined: true }); } } catch (e) { log('joinRoom', e); } }; isOwner = () => { const { room } = this.state; return room && room.roles && Array.from(Object.keys(room.roles), i => room.roles[i].value).includes('owner'); } isMuted = () => { const { room } = this.state; const { user } = this.props; return room && room.muted && Array.from(Object.keys(room.muted), i => room.muted[i].value).includes(user.username); } isReadOnly = () => { const { room } = this.state; return room.ro && this.isMuted() && !this.isOwner(); } isBlocked = () => { const { room } = this.state; if (room) { const { t, blocked, blocker } = room; if (t === 'd' && (blocked || blocker)) { return true; } } return false; } renderItem = (item, previousItem) => { const { room } = this.state; const { user } = this.props; return ( ); } renderFooter = () => { const { joined, room } = this.state; if (!joined) { return ( {I18n.t('You_are_in_preview_mode')} {I18n.t('Join')} ); } if (room.archived || this.isReadOnly()) { return ( {I18n.t('This_room_is_read_only')} ); } if (this.isBlocked()) { return ( {I18n.t('This_room_is_blocked')} ); } return ; }; renderList = () => { const { loaded, room } = this.state; if (!loaded || !room.rid) { return ; } return ( [ , this.renderFooter() ] ); } render() { const { room } = this.state; const { user, showActions, showErrorActions } = this.props; return ( {this.renderList()} {room._id && showActions ? : null } {showErrorActions ? : null} ); } }