| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267 | // @flow
import InlineDialog from '@atlaskit/inline-dialog';
import React, { Component } from 'react';
import type { Dispatch } from 'redux';
import { createToolbarEvent, sendAnalytics } from '../../../../analytics';
import { openDialog } from '../../../../base/dialog';
import { translate } from '../../../../base/i18n';
import { IconInfo } from '../../../../base/icons';
import { JitsiRecordingConstants } from '../../../../base/lib-jitsi-meet';
import { getParticipantCount } from '../../../../base/participants';
import { OverflowMenuItem } from '../../../../base/toolbox';
import { connect } from '../../../../base/redux';
import { getActiveSession } from '../../../../recording';
import { ToolbarButton } from '../../../../toolbox';
import { updateDialInNumbers } from '../../../actions';
import InfoDialog from './InfoDialog';
/**
 * The type of the React {@code Component} props of {@link InfoDialogButton}.
 */
type Props = {
    /**
     * The redux state representing the dial-in numbers feature.
     */
    _dialIn: Object,
    /**
     * Whether or not the {@code InfoDialog} should display automatically when
     * in a lonely call.
     */
    _disableAutoShow: boolean,
    /**
     * Whether or not the local participant has joined a
     * {@code JitsiConference}. Used to trigger auto showing of the
     * {@code InfoDialog}.
     */
    _isConferenceJoined: Boolean,
    /**
     * The URL for a currently active live broadcast
     */
    _liveStreamViewURL: ?string,
    /**
     * The number of real participants in the call. If in a lonely call, the
     * {@code InfoDialog} will be automatically shown.
     */
    _participantCount: number,
    /**
     * Whether or not the toolbox, in which this component exists, is visible.
     */
    _toolboxVisible: boolean,
    /**
     * Invoked to toggle display of the info dialog.
     */
    dispatch: Dispatch<any>,
    /**
     * Whether to show the label or not.
     */
    showLabel: boolean,
    /**
     * Invoked to obtain translated strings.
     */
    t: Function
};
/**
 * The type of the React {@code Component} state of {@link InfoDialogButton}.
 */
type State = {
    /**
     * Cache the conference connection state to derive when transitioning from
     * not joined to join, in order to auto-show the InfoDialog.
     */
    hasConnectedToConference: boolean,
    /**
     * Whether or not {@code InfoDialog} should be visible.
     */
    showDialog: boolean
};
/**
 * A React Component for displaying a button which opens a dialog with
 * information about the conference and with ways to invite people.
 *
 * @extends Component
 */
class InfoDialogButton extends Component<Props, State> {
    /**
     * Implements React's {@link Component#getDerivedStateFromProps()}.
     *
     * @inheritdoc
     */
    static getDerivedStateFromProps(props, state) {
        return {
            hasConnectedToConference: props._isConferenceJoined,
            showDialog: (props._toolboxVisible && state.showDialog)
                || (!state.hasConnectedToConference
                    && props._isConferenceJoined
                    && props._participantCount < 2
                    && props._toolboxVisible
                    && !props._disableAutoShow)
        };
    }
    /**
     * Initializes new {@code InfoDialogButton} instance.
     *
     * @inheritdoc
     */
    constructor(props) {
        super(props);
        this.state = {
            hasConnectedToConference: props._isConferenceJoined,
            showDialog: false
        };
        // Bind event handlers so they are only bound once for every instance.
        this._onDialogClose = this._onDialogClose.bind(this);
        this._onDialogToggle = this._onDialogToggle.bind(this);
        this._onClickOverflowMenuButton
            = this._onClickOverflowMenuButton.bind(this);
    }
    /**
     * Update dial-in numbers {@code InfoDialog}.
     *
     * @inheritdoc
     */
    componentDidMount() {
        if (!this.props._dialIn.numbers) {
            this.props.dispatch(updateDialInNumbers());
        }
    }
    /**
     * Implements React's {@link Component#render()}.
     *
     * @inheritdoc
     * @returns {ReactElement}
     */
    render() {
        const { _dialIn, _liveStreamViewURL, showLabel, t } = this.props;
        const { showDialog } = this.state;
        if (showLabel) {
            return (
                <OverflowMenuItem
                    accessibilityLabel = { t('info.accessibilityLabel') }
                    icon = 'icon-info'
                    key = 'info-button'
                    onClick = { this._onClickOverflowMenuButton }
                    text = { t('info.label') } />
            );
        }
        return (
            <div className = 'toolbox-button-wth-dialog'>
                <InlineDialog
                    content = {
                        <InfoDialog
                            dialIn = { _dialIn }
                            isInlineDialog = { true }
                            liveStreamViewURL = { _liveStreamViewURL }
                            onClose = { this._onDialogClose } /> }
                    isOpen = { showDialog }
                    onClose = { this._onDialogClose }
                    position = { 'top right' }>
                    <ToolbarButton
                        accessibilityLabel = { t('info.accessibilityLabel') }
                        icon = { IconInfo }
                        onClick = { this._onDialogToggle }
                        tooltip = { t('info.tooltip') } />
                </InlineDialog>
            </div>
        );
    }
    _onDialogClose: () => void;
    /**
     * Hides {@code InfoDialog}.
     *
     * @private
     * @returns {void}
     */
    _onDialogClose() {
        this.setState({ showDialog: false });
    }
    _onClickOverflowMenuButton: () => void;
    /**
     * Opens the Info dialog.
     *
     * @returns {void}
     */
    _onClickOverflowMenuButton() {
        const { _dialIn, _liveStreamViewURL } = this.props;
        this.props.dispatch(openDialog(InfoDialog, {
            dialIn: _dialIn,
            liveStreamViewURL: _liveStreamViewURL,
            isInlineDialog: false
        }));
    }
    _onDialogToggle: () => void;
    /**
     * Toggles the display of {@code InfoDialog}.
     *
     * @private
     * @returns {void}
     */
    _onDialogToggle() {
        sendAnalytics(createToolbarEvent('info'));
        this.setState({ showDialog: !this.state.showDialog });
    }
}
/**
 * Maps (parts of) the Redux state to the associated {@code InfoDialogButton}
 * component's props.
 *
 * @param {Object} state - The Redux state.
 * @private
 * @returns {{
 *     _dialIn: Object,
 *     _disableAutoShow: boolean,
 *     _isConferenceIsJoined: boolean,
 *     _liveStreamViewURL: string,
 *     _participantCount: number,
 *     _toolboxVisible: boolean
 * }}
 */
function _mapStateToProps(state) {
    const currentLiveStreamingSession
        = getActiveSession(state, JitsiRecordingConstants.mode.STREAM);
    return {
        _dialIn: state['features/invite'],
        _disableAutoShow: state['features/base/config'].iAmRecorder,
        _isConferenceJoined:
            Boolean(state['features/base/conference'].conference),
        _liveStreamViewURL:
            currentLiveStreamingSession
                && currentLiveStreamingSession.liveStreamViewURL,
        _participantCount: getParticipantCount(state),
        _toolboxVisible: state['features/toolbox'].visible
    };
}
export default translate(connect(_mapStateToProps)(InfoDialogButton));
 |