2020-03-30 15:26:15 +00:00
|
|
|
// @flow
|
|
|
|
|
|
|
|
import React, { PureComponent } from 'react';
|
2021-04-06 10:07:24 +00:00
|
|
|
import { KeyboardAvoidingView, Platform, SafeAreaView } from 'react-native';
|
2020-03-30 15:26:15 +00:00
|
|
|
|
|
|
|
import { ColorSchemeRegistry } from '../../color-scheme';
|
|
|
|
import { HeaderWithNavigation, SlidingView } from '../../react';
|
|
|
|
import { connect } from '../../redux';
|
|
|
|
import { StyleType } from '../../styles';
|
|
|
|
import { setActiveModalId } from '../actions';
|
|
|
|
|
|
|
|
import styles from './styles';
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The color schemed style of the common header component.
|
|
|
|
*/
|
|
|
|
_headerStyles: StyleType,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* True if the modal should be shown, false otherwise.
|
|
|
|
*/
|
|
|
|
_show: boolean,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The color schemed style of the modal.
|
|
|
|
*/
|
|
|
|
_styles: StyleType,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The children component(s) of the Modal, to be rendered.
|
|
|
|
*/
|
|
|
|
children: React$Node,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The Redux Dispatch function.
|
|
|
|
*/
|
|
|
|
dispatch: Function,
|
|
|
|
|
|
|
|
/**
|
2020-04-06 15:14:32 +00:00
|
|
|
* Optional function that renders a footer component, if needed.
|
2020-03-30 15:26:15 +00:00
|
|
|
*/
|
2020-04-06 15:14:32 +00:00
|
|
|
footerComponent?: Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Props to be passed over to the header.
|
|
|
|
*
|
|
|
|
* See {@code HeaderWithNavigation} for more details.
|
|
|
|
*/
|
|
|
|
headerProps: Object,
|
2020-03-30 15:26:15 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The ID of the modal that is being rendered. This is used to show/hide the modal.
|
|
|
|
*/
|
|
|
|
modalId: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback to be invoked when the modal closes.
|
|
|
|
*/
|
|
|
|
onClose?: Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The position from where the modal should be opened. This is derived from the
|
|
|
|
* props of the {@code SlidingView} with the same name.
|
|
|
|
*/
|
2020-04-06 15:14:32 +00:00
|
|
|
position?: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Additional style to be appended to the View containing the content of the modal.
|
|
|
|
*/
|
|
|
|
style?: StyleType
|
2020-03-30 15:26:15 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements a custom Jitsi Modal that doesn't use the built in native
|
|
|
|
* Modal component of React Native.
|
|
|
|
*/
|
|
|
|
class JitsiModal extends PureComponent<Props> {
|
|
|
|
static defaultProps = {
|
|
|
|
position: 'bottom'
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Instantiates a new component.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
constructor(props: Props) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
this._onRequestClose = this._onRequestClose.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements {@code PureComponent#render}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
render() {
|
2020-04-06 15:14:32 +00:00
|
|
|
const { _headerStyles, _show, _styles, children, footerComponent, headerProps, position, style } = this.props;
|
2020-03-30 15:26:15 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<SlidingView
|
|
|
|
onHide = { this._onRequestClose }
|
|
|
|
position = { position }
|
|
|
|
show = { _show }>
|
2020-04-06 15:14:32 +00:00
|
|
|
<KeyboardAvoidingView
|
2021-04-06 10:07:24 +00:00
|
|
|
behavior =
|
|
|
|
{
|
|
|
|
Platform.OS === 'ios'
|
|
|
|
? 'padding' : 'height'
|
|
|
|
}
|
|
|
|
enabled = { true }
|
2020-03-30 15:26:15 +00:00
|
|
|
style = { [
|
|
|
|
_headerStyles.page,
|
2020-04-06 15:14:32 +00:00
|
|
|
_styles.page,
|
|
|
|
style
|
2020-03-30 15:26:15 +00:00
|
|
|
] }>
|
|
|
|
<HeaderWithNavigation
|
2020-04-06 15:14:32 +00:00
|
|
|
{ ...headerProps }
|
2020-03-30 15:26:15 +00:00
|
|
|
onPressBack = { this._onRequestClose } />
|
|
|
|
<SafeAreaView style = { styles.safeArea }>
|
|
|
|
{ children }
|
|
|
|
</SafeAreaView>
|
2020-04-06 15:14:32 +00:00
|
|
|
{ footerComponent && footerComponent() }
|
|
|
|
</KeyboardAvoidingView>
|
2020-03-30 15:26:15 +00:00
|
|
|
</SlidingView>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
_onRequestClose: () => boolean;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback to be invoked when the SlidingView requests closing.
|
|
|
|
*
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
_onRequestClose() {
|
|
|
|
const { _show, dispatch, onClose } = this.props;
|
2020-04-06 15:14:32 +00:00
|
|
|
let shouldCloseModal = true;
|
2020-03-30 15:26:15 +00:00
|
|
|
|
|
|
|
if (_show) {
|
|
|
|
if (typeof onClose === 'function') {
|
2020-04-06 15:14:32 +00:00
|
|
|
shouldCloseModal = onClose();
|
2020-03-30 15:26:15 +00:00
|
|
|
}
|
2020-04-06 15:14:32 +00:00
|
|
|
shouldCloseModal && dispatch(setActiveModalId());
|
2020-03-30 15:26:15 +00:00
|
|
|
|
2020-04-06 15:14:32 +00:00
|
|
|
return shouldCloseModal;
|
2020-03-30 15:26:15 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps part of the Redix state to the props of this component.
|
|
|
|
*
|
|
|
|
* @param {Object} state - The Redux state.
|
|
|
|
* @param {Props} ownProps - The own props of the component.
|
|
|
|
* @returns {Props}
|
|
|
|
*/
|
|
|
|
function _mapStateToProps(state, ownProps): $Shape<Props> {
|
|
|
|
return {
|
|
|
|
_headerStyles: ColorSchemeRegistry.get(state, 'Header'),
|
|
|
|
_show: state['features/base/modal'].activeModalId === ownProps.modalId,
|
|
|
|
_styles: ColorSchemeRegistry.get(state, 'Modal')
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export default connect(_mapStateToProps)(JitsiModal);
|