import React from 'react';
import { connect } from 'react-redux';
import { ScrollView, Switch, Text } from 'react-native';
import { StackNavigationOptions } from '@react-navigation/stack';
import Loading from '../../containers/Loading';
import KeyboardView from '../../presentation/KeyboardView';
import scrollPersistTaps from '../../utils/scrollPersistTaps';
import I18n from '../../i18n';
import * as HeaderButton from '../../containers/HeaderButton';
import StatusBar from '../../containers/StatusBar';
import { SWITCH_TRACK_COLOR, themes } from '../../constants/colors';
import { withTheme } from '../../theme';
import { getUserSelector } from '../../selectors/login';
import TextInput from '../../containers/TextInput';
import RocketChat from '../../lib/rocketchat';
import Navigation from '../../lib/Navigation';
import { createDiscussionRequest } from '../../actions/createDiscussion';
import { showErrorAlert } from '../../utils/info';
import SafeAreaView from '../../containers/SafeAreaView';
import { goRoom } from '../../utils/goRoom';
import { events, logEvent } from '../../utils/log';
import { E2E_ROOM_TYPES } from '../../lib/encryption/constants';
import styles from './styles';
import SelectUsers from './SelectUsers';
import SelectChannel from './SelectChannel';
import { ICreateChannelViewProps, IResult, IError } from './interfaces';
import { IApplicationState } from '../../definitions';
class CreateChannelView extends React.Component<ICreateChannelViewProps, any> {
private channel: any;
constructor(props: ICreateChannelViewProps) {
super(props);
const { route } = props;
this.channel = route.params?.channel;
const message: any = route.params?.message ?? {};
this.state = {
channel: this.channel,
message,
name: message?.msg || '',
users: [],
reply: '',
encrypted: props.encryptionEnabled
};
this.setHeader();
}
componentDidUpdate(prevProps: any, prevState: any) {
const { channel, name } = this.state;
const { loading, failure, error, result, isMasterDetail } = this.props;
if (channel?.rid !== prevState.channel?.rid || name !== prevState.name) {
if (!loading && loading !== prevProps.loading) {
setTimeout(() => {
if (failure) {
const msg = error.reason || I18n.t('There_was_an_error_while_action', { action: I18n.t('creating_discussion') });
showErrorAlert(msg);
} else {
const { rid, t, prid } = result;
if (isMasterDetail) {
Navigation.navigate('DrawerNavigator');
Navigation.navigate('RoomsListView');
const item = {
rid,
name: RocketChat.getRoomTitle(result),
t,
prid
goRoom({ item, isMasterDetail });
}, 300);
setHeader = () => {
const { navigation, route } = this.props;
const showCloseModal = route.params?.showCloseModal;
navigation.setOptions({
title: I18n.t('Create_Discussion'),
headerRight: this.valid()
? () => (
<HeaderButton.Container>
<HeaderButton.Item title={I18n.t('Create')} onPress={this.submit} testID='create-discussion-submit' />
</HeaderButton.Container>
)
: null,
headerLeft: showCloseModal ? () => <HeaderButton.CloseModal navigation={navigation} /> : undefined
} as StackNavigationOptions);
submit = () => {
const {
name: t_name,
channel: { prid, rid },
message: { id: pmid },
reply,
users,
encrypted
} = this.state;
const { dispatch } = this.props;
const params: any = {
prid: prid || rid,
pmid,
t_name,
users
if (this.isEncryptionEnabled) {
params.encrypted = encrypted ?? false;
dispatch(createDiscussionRequest(params));
valid = () => {
return channel && channel.rid && channel.rid.trim().length && name.trim().length;
selectChannel = ({ value }: any) => {
logEvent(events.CD_SELECT_CHANNEL);
this.setState({ channel: value, encrypted: value?.encrypted });
selectUsers = ({ value }: any) => {
logEvent(events.CD_SELECT_USERS);
this.setState({ users: value });
get isEncryptionEnabled() {
const { channel } = this.state;
const { encryptionEnabled } = this.props;
// TODO: remove this ts-ignore when migrate the file: app/lib/encryption/constants.js
// @ts-ignore
return encryptionEnabled && E2E_ROOM_TYPES[channel?.t];
onEncryptedChange = (value: any) => {
logEvent(events.CD_TOGGLE_ENCRY);
this.setState({ encrypted: value });
render() {
const { name, users, encrypted } = this.state;
const { server, user, loading, blockUnauthenticatedAccess, theme, serverVersion } = this.props;
return (
<KeyboardView
style={{ backgroundColor: themes[theme].auxiliaryBackground }}
contentContainerStyle={styles.container}
keyboardVerticalOffset={128}>
<StatusBar />
<SafeAreaView testID='create-discussion-view' style={styles.container}>
<ScrollView {...scrollPersistTaps}>
<Text style={[styles.description, { color: themes[theme].auxiliaryText }]}>{I18n.t('Discussion_Desc')}</Text>
<SelectChannel
server={server}
userId={user.id}
token={user.token}
initial={this.channel && { text: RocketChat.getRoomTitle(this.channel) }}
onChannelSelect={this.selectChannel}
blockUnauthenticatedAccess={blockUnauthenticatedAccess}
serverVersion={serverVersion}
theme={theme}
/>
<TextInput
label={I18n.t('Discussion_name')}
testID='multi-select-discussion-name'
placeholder={I18n.t('A_meaningful_name_for_the_discussion_room')}
containerStyle={styles.inputStyle}
/* @ts-ignore*/
defaultValue={name}
onChangeText={(text: string) => this.setState({ name: text })}
<SelectUsers
selected={users}
onUserSelect={this.selectUsers}
{this.isEncryptionEnabled ? (
<>
<Text style={[styles.label, { color: themes[theme].titleText }]}>{I18n.t('Encrypted')}</Text>
<Switch value={encrypted} onValueChange={this.onEncryptedChange} trackColor={SWITCH_TRACK_COLOR} />
</>
) : null}
<Loading visible={loading} />
</ScrollView>
</SafeAreaView>
</KeyboardView>
);
const mapStateToProps = (state: IApplicationState) => ({
user: getUserSelector(state),
server: state.server.server,
error: state.createDiscussion.error as IError,
failure: state.createDiscussion.failure,
loading: state.createDiscussion.isFetching,
result: state.createDiscussion.result as IResult,
blockUnauthenticatedAccess: !!state.settings.Accounts_AvatarBlockUnauthenticatedAccess ?? true,
serverVersion: state.server.version as string,
isMasterDetail: state.app.isMasterDetail,
encryptionEnabled: state.encryption.enabled
});
export default connect(mapStateToProps)(withTheme(CreateChannelView));