123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392 |
- // @flow
-
- import { withStyles } from '@material-ui/core';
- import React, { Component } from 'react';
-
- import participantsPaneTheme from '../../../base/components/themes/participantsPaneTheme.json';
- import { openDialog } from '../../../base/dialog';
- import { translate } from '../../../base/i18n';
- import { Icon, IconClose, IconHorizontalPoints } from '../../../base/icons';
- import { isLocalParticipantModerator } from '../../../base/participants';
- import { connect } from '../../../base/redux';
- import { getBreakoutRoomsConfig } from '../../../breakout-rooms/functions';
- import { MuteEveryoneDialog } from '../../../video-menu/components/';
- import { close } from '../../actions';
- import { findAncestorByClass, getParticipantsPaneOpen } from '../../functions';
- import { AddBreakoutRoomButton } from '../breakout-rooms/components/web/AddBreakoutRoomButton';
- import { RoomList } from '../breakout-rooms/components/web/RoomList';
-
- import FooterButton from './FooterButton';
- import { FooterContextMenu } from './FooterContextMenu';
- import LobbyParticipants from './LobbyParticipants';
- import MeetingParticipants from './MeetingParticipants';
-
- /**
- * The type of the React {@code Component} props of {@link ParticipantsPane}.
- */
- type Props = {
-
- /**
- * Whether there is backend support for Breakout Rooms.
- */
- _isBreakoutRoomsSupported: Boolean,
-
- /**
- * Whether to display the context menu as a drawer.
- */
- _overflowDrawer: boolean,
-
- /**
- * Should the add breakout room button be displayed?
- */
- _showAddRoomButton: boolean,
-
- /**
- * Is the participants pane open.
- */
- _paneOpen: boolean,
-
- /**
- * Whether to show the footer menu.
- */
- _showFooter: boolean,
-
- /**
- * The Redux dispatch function.
- */
- dispatch: Function,
-
- /**
- * An object containing the CSS classes.
- */
- classes: Object,
-
- /**
- * The i18n translate function.
- */
- t: Function
- };
-
- /**
- * The type of the React {@code Component} state of {@link ParticipantsPane}.
- */
- type State = {
-
- /**
- * Indicates if the footer context menu is open.
- */
- contextOpen: boolean,
-
- /**
- * Participants search string.
- */
- searchString: string
- };
-
- const styles = theme => {
- return {
- container: {
- boxSizing: 'border-box',
- flex: 1,
- overflowY: 'auto',
- position: 'relative',
- padding: `0 ${participantsPaneTheme.panePadding}px`,
-
- [`& > * + *:not(.${participantsPaneTheme.ignoredChildClassName})`]: {
- marginTop: theme.spacing(3)
- },
-
- '&::-webkit-scrollbar': {
- display: 'none'
- }
- },
-
- closeButton: {
- alignItems: 'center',
- cursor: 'pointer',
- display: 'flex',
- justifyContent: 'center'
- },
-
- header: {
- alignItems: 'center',
- boxSizing: 'border-box',
- display: 'flex',
- height: `${participantsPaneTheme.headerSize}px`,
- padding: '0 20px',
- justifyContent: 'flex-end'
- },
-
- antiCollapse: {
- fontSize: 0,
-
- '&:first-child': {
- display: 'none'
- },
-
- '&:first-child + *': {
- marginTop: 0
- }
- },
-
- footer: {
- display: 'flex',
- justifyContent: 'flex-end',
- padding: `${theme.spacing(4)}px ${participantsPaneTheme.panePadding}px`,
-
- '& > *:not(:last-child)': {
- marginRight: `${theme.spacing(3)}px`
- }
- },
-
- footerMoreContainer: {
- position: 'relative'
- }
- };
- };
-
- /**
- * Implements the participants list.
- */
- class ParticipantsPane extends Component<Props, State> {
- /**
- * Initializes a new {@code ParticipantsPane} instance.
- *
- * @inheritdoc
- */
- constructor(props) {
- super(props);
-
- this.state = {
- contextOpen: false,
- searchString: ''
- };
-
- // Bind event handlers so they are only bound once per instance.
- this._onClosePane = this._onClosePane.bind(this);
- this._onDrawerClose = this._onDrawerClose.bind(this);
- this._onKeyPress = this._onKeyPress.bind(this);
- this._onMuteAll = this._onMuteAll.bind(this);
- this._onToggleContext = this._onToggleContext.bind(this);
- this._onWindowClickListener = this._onWindowClickListener.bind(this);
- this.setSearchString = this.setSearchString.bind(this);
- }
-
-
- /**
- * Implements React's {@link Component#componentDidMount()}.
- *
- * @inheritdoc
- */
- componentDidMount() {
- window.addEventListener('click', this._onWindowClickListener);
- }
-
- /**
- * Implements React's {@link Component#componentWillUnmount()}.
- *
- * @inheritdoc
- */
- componentWillUnmount() {
- window.removeEventListener('click', this._onWindowClickListener);
- }
-
- /**
- * Implements React's {@link Component#render}.
- *
- * @inheritdoc
- */
- render() {
- const {
- _isBreakoutRoomsSupported,
- _paneOpen,
- _showAddRoomButton,
- _showFooter,
- classes,
- t
- } = this.props;
- const { contextOpen, searchString } = this.state;
-
- // when the pane is not open optimize to not
- // execute the MeetingParticipantList render for large list of participants
- if (!_paneOpen) {
- return null;
- }
-
- return (
- <div className = 'participants_pane'>
- <div className = 'participants_pane-content'>
- <div className = { classes.header }>
- <div
- aria-label = { t('participantsPane.close', 'Close') }
- className = { classes.closeButton }
- onClick = { this._onClosePane }
- onKeyPress = { this._onKeyPress }
- role = 'button'
- tabIndex = { 0 }>
- <Icon
- size = { 24 }
- src = { IconClose } />
- </div>
- </div>
- <div className = { classes.container }>
- <LobbyParticipants />
- <br className = { classes.antiCollapse } />
- <MeetingParticipants
- searchString = { searchString }
- setSearchString = { this.setSearchString } />
- {_isBreakoutRoomsSupported && <RoomList searchString = { searchString } />}
- {_showAddRoomButton && <AddBreakoutRoomButton />}
- </div>
- {_showFooter && (
- <div className = { classes.footer }>
- <FooterButton
- accessibilityLabel = { t('participantsPane.actions.muteAll') }
- onClick = { this._onMuteAll }>
- {t('participantsPane.actions.muteAll')}
- </FooterButton>
- <div className = { classes.footerMoreContainer }>
- <FooterButton
- accessibilityLabel = { t('participantsPane.actions.moreModerationActions') }
- id = 'participants-pane-context-menu'
- isIconButton = { true }
- onClick = { this._onToggleContext }>
- <Icon src = { IconHorizontalPoints } />
- </FooterButton>
- <FooterContextMenu
- isOpen = { contextOpen }
- onDrawerClose = { this._onDrawerClose }
- onMouseLeave = { this._onToggleContext } />
- </div>
- </div>
- )}
- </div>
- </div>
- );
- }
-
- setSearchString: (string) => void;
-
- /**
- * Sets the search string.
- *
- * @param {string} newSearchString - The new search string.
- * @returns {void}
- */
- setSearchString(newSearchString) {
- this.setState({
- searchString: newSearchString
- });
- }
-
- _onClosePane: () => void;
-
- /**
- * Callback for closing the participant pane.
- *
- * @private
- * @returns {void}
- */
- _onClosePane() {
- this.props.dispatch(close());
- }
-
- _onDrawerClose: () => void;
-
- /**
- * Callback for closing the drawer.
- *
- * @private
- * @returns {void}
- */
- _onDrawerClose() {
- this.setState({
- contextOpen: false
- });
- }
-
- _onKeyPress: (Object) => void;
-
- /**
- * KeyPress handler for accessibility for closing the participants pane.
- *
- * @param {Object} e - The key event to handle.
- *
- * @returns {void}
- */
- _onKeyPress(e) {
- if (e.key === ' ' || e.key === 'Enter') {
- e.preventDefault();
- this._onClosePane();
- }
- }
-
- _onMuteAll: () => void;
-
- /**
- * The handler for clicking mute all button.
- *
- * @returns {void}
- */
- _onMuteAll() {
- this.props.dispatch(openDialog(MuteEveryoneDialog));
- }
-
- _onToggleContext: () => void;
-
- /**
- * Handler for toggling open/close of the footer context menu.
- *
- * @returns {void}
- */
- _onToggleContext() {
- this.setState({
- contextOpen: !this.state.contextOpen
- });
- }
-
- _onWindowClickListener: (event: Object) => void;
-
- /**
- * Window click event listener.
- *
- * @param {Event} e - The click event.
- * @returns {void}
- */
- _onWindowClickListener(e) {
- if (this.state.contextOpen && !findAncestorByClass(e.target, this.props.classes.footerMoreContainer)) {
- this.setState({
- contextOpen: false
- });
- }
- }
-
-
- }
-
- /**
- * Maps (parts of) the redux state to the React {@code Component} props of
- * {@code ParticipantsPane}.
- *
- * @param {Object} state - The redux state.
- * @protected
- * @returns {Props}
- */
- function _mapStateToProps(state: Object) {
- const isPaneOpen = getParticipantsPaneOpen(state);
- const { hideAddRoomButton } = getBreakoutRoomsConfig(state);
- const { conference } = state['features/base/conference'];
-
- // $FlowExpectedError
- const _isBreakoutRoomsSupported = conference?.getBreakoutRooms()?.isSupported();
- const _isLocalParticipantModerator = isLocalParticipantModerator(state);
-
- return {
- _isBreakoutRoomsSupported,
- _paneOpen: isPaneOpen,
- _showAddRoomButton: _isBreakoutRoomsSupported && !hideAddRoomButton && _isLocalParticipantModerator,
- _showFooter: isPaneOpen && isLocalParticipantModerator(state)
- };
- }
-
- export default translate(connect(_mapStateToProps)(withStyles(styles)(ParticipantsPane)));
|