2020-04-15 13:13:43 +00:00
|
|
|
// @flow
|
|
|
|
|
2022-01-20 14:26:03 +00:00
|
|
|
import React, { PureComponent } from 'react';
|
2020-04-15 13:13:43 +00:00
|
|
|
import { ScrollView, Text, View, TouchableOpacity } from 'react-native';
|
|
|
|
|
|
|
|
import { Avatar } from '../../../base/avatar';
|
|
|
|
import { translate } from '../../../base/i18n';
|
2022-01-20 14:26:03 +00:00
|
|
|
import { isLocalParticipantModerator } from '../../../base/participants';
|
2020-04-15 13:13:43 +00:00
|
|
|
import { connect } from '../../../base/redux';
|
2022-03-03 17:29:38 +00:00
|
|
|
import { handleLobbyChatInitialized } from '../../../chat/actions.any';
|
|
|
|
import { navigate } from '../../../mobile/navigation/components/conference/ConferenceNavigationContainerRef';
|
|
|
|
import { screen } from '../../../mobile/navigation/routes';
|
2022-01-20 14:26:03 +00:00
|
|
|
import { setKnockingParticipantApproval } from '../../actions';
|
2021-04-20 13:05:49 +00:00
|
|
|
import { HIDDEN_EMAILS } from '../../constants';
|
2022-03-03 17:29:38 +00:00
|
|
|
import { showLobbyChatButton, getKnockingParticipants, getLobbyEnabled } from '../../functions';
|
2020-04-15 13:13:43 +00:00
|
|
|
|
|
|
|
import styles from './styles';
|
|
|
|
|
2022-01-20 14:26:03 +00:00
|
|
|
/**
|
|
|
|
* Props type of the component.
|
|
|
|
*/
|
|
|
|
export type Props = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The list of participants.
|
|
|
|
*/
|
|
|
|
_participants: Array<Object>,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if the list should be rendered.
|
|
|
|
*/
|
|
|
|
_visible: boolean,
|
|
|
|
|
2022-03-03 17:29:38 +00:00
|
|
|
/**
|
|
|
|
* True if the polls feature is disabled.
|
|
|
|
*/
|
|
|
|
_isPollsDisabled: boolean,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if the lobby chat button should be shown.
|
|
|
|
*/
|
|
|
|
_showChatButton: Function,
|
|
|
|
|
2022-01-20 14:26:03 +00:00
|
|
|
/**
|
|
|
|
* The Redux Dispatch function.
|
|
|
|
*/
|
|
|
|
dispatch: Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function to be used to translate i18n labels.
|
|
|
|
*/
|
|
|
|
t: Function
|
|
|
|
};
|
|
|
|
|
2020-04-15 13:13:43 +00:00
|
|
|
/**
|
|
|
|
* Component to render a list for the actively knocking participants.
|
|
|
|
*/
|
2022-01-20 14:26:03 +00:00
|
|
|
class KnockingParticipantList extends PureComponent<Props> {
|
|
|
|
/**
|
|
|
|
* Instantiates a new component.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The read-only properties with which the new
|
|
|
|
* instance is to be initialized.
|
|
|
|
*/
|
|
|
|
constructor(props: Props) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this._onRespondToParticipant = this._onRespondToParticipant.bind(this);
|
|
|
|
}
|
|
|
|
|
2020-04-15 13:13:43 +00:00
|
|
|
/**
|
|
|
|
* Implements {@code PureComponent#render}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
render() {
|
2022-03-03 17:29:38 +00:00
|
|
|
const { _participants, _visible, _showChatButton, t } = this.props;
|
2020-04-15 13:13:43 +00:00
|
|
|
|
2020-05-20 08:25:31 +00:00
|
|
|
if (!_visible) {
|
|
|
|
return null;
|
|
|
|
}
|
2020-04-15 13:13:43 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<ScrollView
|
|
|
|
style = { styles.knockingParticipantList }>
|
2020-05-20 08:25:31 +00:00
|
|
|
{ _participants.map(p => (
|
2020-04-15 13:13:43 +00:00
|
|
|
<View
|
|
|
|
key = { p.id }
|
|
|
|
style = { styles.knockingParticipantListEntry }>
|
|
|
|
<Avatar
|
|
|
|
displayName = { p.name }
|
|
|
|
size = { 48 }
|
|
|
|
url = { p.loadableAvatarUrl } />
|
|
|
|
<View style = { styles.knockingParticipantListDetails }>
|
|
|
|
<Text style = { styles.knockingParticipantListText }>
|
|
|
|
{ p.name }
|
|
|
|
</Text>
|
2021-04-20 13:05:49 +00:00
|
|
|
{ p.email && !HIDDEN_EMAILS.includes(p.email) && (
|
2020-04-15 13:13:43 +00:00
|
|
|
<Text style = { styles.knockingParticipantListText }>
|
|
|
|
{ p.email }
|
|
|
|
</Text>
|
|
|
|
) }
|
|
|
|
</View>
|
|
|
|
<TouchableOpacity
|
|
|
|
onPress = { this._onRespondToParticipant(p.id, true) }
|
|
|
|
style = { [
|
|
|
|
styles.knockingParticipantListButton,
|
|
|
|
styles.knockingParticipantListPrimaryButton
|
|
|
|
] }>
|
|
|
|
<Text style = { styles.knockingParticipantListText }>
|
|
|
|
{ t('lobby.allow') }
|
|
|
|
</Text>
|
|
|
|
</TouchableOpacity>
|
|
|
|
<TouchableOpacity
|
|
|
|
onPress = { this._onRespondToParticipant(p.id, false) }
|
|
|
|
style = { [
|
|
|
|
styles.knockingParticipantListButton,
|
|
|
|
styles.knockingParticipantListSecondaryButton
|
|
|
|
] }>
|
|
|
|
<Text style = { styles.knockingParticipantListText }>
|
|
|
|
{ t('lobby.reject') }
|
|
|
|
</Text>
|
|
|
|
</TouchableOpacity>
|
2022-03-03 17:29:38 +00:00
|
|
|
{_showChatButton(p) ? (
|
|
|
|
<TouchableOpacity
|
|
|
|
onPress = { this._onInitializeLobbyChat(p.id) }
|
|
|
|
style = { [
|
|
|
|
styles.knockingParticipantListButton,
|
|
|
|
styles.knockingParticipantListSecondaryButton
|
|
|
|
] }>
|
|
|
|
<Text style = { styles.knockingParticipantListText }>
|
|
|
|
{ t('lobby.chat') }
|
|
|
|
</Text>
|
|
|
|
</TouchableOpacity>
|
|
|
|
) : null}
|
2020-04-15 13:13:43 +00:00
|
|
|
</View>
|
|
|
|
)) }
|
|
|
|
</ScrollView>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onRespondToParticipant: (string, boolean) => Function;
|
2022-01-20 14:26:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function that constructs a callback for the response handler button.
|
|
|
|
*
|
|
|
|
* @param {string} id - The id of the knocking participant.
|
|
|
|
* @param {boolean} approve - The response for the knocking.
|
|
|
|
* @returns {Function}
|
|
|
|
*/
|
|
|
|
_onRespondToParticipant(id, approve) {
|
|
|
|
return () => {
|
|
|
|
this.props.dispatch(setKnockingParticipantApproval(id, approve));
|
|
|
|
};
|
|
|
|
}
|
2022-03-03 17:29:38 +00:00
|
|
|
|
|
|
|
_onInitializeLobbyChat: (string) => Function;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function that constructs a callback for the lobby chat button.
|
|
|
|
*
|
|
|
|
* @param {string} id - The id of the knocking participant.
|
|
|
|
* @returns {Function}
|
|
|
|
*/
|
|
|
|
_onInitializeLobbyChat(id) {
|
|
|
|
return () => {
|
|
|
|
this.props.dispatch(handleLobbyChatInitialized(id));
|
|
|
|
if (this.props._isPollsDisabled) {
|
|
|
|
return navigate(screen.conference.chat);
|
|
|
|
}
|
|
|
|
navigate(screen.conference.chatandpolls.main);
|
|
|
|
};
|
|
|
|
}
|
2020-04-15 13:13:43 +00:00
|
|
|
}
|
|
|
|
|
2020-05-20 08:25:31 +00:00
|
|
|
/**
|
|
|
|
* Maps part of the Redux state to the props of this component.
|
|
|
|
*
|
|
|
|
* @param {Object} state - The Redux state.
|
|
|
|
* @returns {Props}
|
|
|
|
*/
|
2022-01-20 14:26:03 +00:00
|
|
|
function _mapStateToProps(state): Object {
|
|
|
|
const lobbyEnabled = getLobbyEnabled(state);
|
|
|
|
const knockingParticipants = getKnockingParticipants(state);
|
2022-03-03 17:29:38 +00:00
|
|
|
const { disablePolls } = state['features/base/config'];
|
2020-05-20 08:25:31 +00:00
|
|
|
|
|
|
|
return {
|
2022-01-20 14:26:03 +00:00
|
|
|
_visible: lobbyEnabled && isLocalParticipantModerator(state),
|
2022-03-03 17:29:38 +00:00
|
|
|
_showChatButton: participant => showLobbyChatButton(participant)(state),
|
|
|
|
_isPollsDisabled: disablePolls,
|
2020-05-20 08:25:31 +00:00
|
|
|
|
|
|
|
// On mobile we only show a portion of the list for screen real estate reasons
|
2022-01-20 14:26:03 +00:00
|
|
|
_participants: knockingParticipants.slice(0, 2)
|
2020-05-20 08:25:31 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export default translate(connect(_mapStateToProps)(KnockingParticipantList));
|