// @flow import { NativeModules, Platform } from 'react-native'; import { PIP_ENABLED, PIP_WHILE_SCREEN_SHARING_ENABLED, getFeatureFlag } from '../../../base/flags'; import { translate } from '../../../base/i18n'; import { IconArrowDown } from '../../../base/icons'; import { connect } from '../../../base/redux'; import { AbstractButton, type AbstractButtonProps } from '../../../base/toolbox/components'; import { isLocalVideoTrackDesktop } from '../../../base/tracks/functions'; import { enterPictureInPicture } from '../actions'; type Props = AbstractButtonProps & { /** * Whether Picture-in-Picture is enabled or not. */ _enabled: boolean, /** * The redux {@code dispatch} function. */ dispatch: Function }; /** * An implementation of a button for entering Picture-in-Picture mode. */ class PictureInPictureButton extends AbstractButton { accessibilityLabel = 'toolbar.accessibilityLabel.pip'; icon = IconArrowDown; label = 'toolbar.pip'; /** * Handles clicking / pressing the button. * * @protected * @returns {void} */ _handleClick() { this.props.dispatch(enterPictureInPicture()); } /** * Implements React's {@link Component#render()}. * * @inheritdoc * @returns {React$Node} */ render() { return this.props._enabled ? super.render() : null; } } /** * Maps (parts of) the redux state to the associated props for the * {@code PictureInPictureButton} component. * * @param {Object} state - The Redux state. * @private * @returns {{ * _enabled: boolean * }} */ function _mapStateToProps(state): Object { const pipEnabled = Boolean(getFeatureFlag(state, PIP_ENABLED)); const pipWhileScreenSharingEnabled = getFeatureFlag(state, PIP_WHILE_SCREEN_SHARING_ENABLED, false); let enabled = pipEnabled && (!isLocalVideoTrackDesktop(state) || pipWhileScreenSharingEnabled); // Override flag for Android, since it might be unsupported. if (Platform.OS === 'android' && !NativeModules.PictureInPicture.SUPPORTED) { enabled = false; } return { _enabled: enabled }; } export default translate(connect(_mapStateToProps)(PictureInPictureButton));