import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { throttle } from '/imports/utils/throttle'; import { defineMessages, injectIntl } from 'react-intl'; import ReactModal from 'react-modal'; import browserInfo from '/imports/utils/browserInfo'; import deviceInfo from '/imports/utils/deviceInfo'; import PollingContainer from '/imports/ui/components/polling/container'; import logger from '/imports/startup/client/logger'; import ActivityCheckContainer from '/imports/ui/components/activity-check/container'; import UserInfoContainer from '/imports/ui/components/user-info/container'; import BreakoutRoomInvitation from '/imports/ui/components/breakout-room/invitation/container'; import { Meteor } from 'meteor/meteor'; import ToastContainer from '/imports/ui/components/common/toast/container'; import PadsSessionsContainer from '/imports/ui/components/pads/sessions/container'; import WakeLockContainer from '../wake-lock/container'; import NotificationsBarContainer from '../notifications-bar/container'; import AudioContainer from '../audio/container'; import ChatAlertContainer from '../chat/alert/container'; import BannerBarContainer from '/imports/ui/components/banner-bar/container'; import RaiseHandNotifier from '/imports/ui/components/raisehand-notifier/container'; import ManyWebcamsNotifier from '/imports/ui/components/video-provider/many-users-notify/container'; import AudioCaptionsSpeechContainer from '/imports/ui/components/audio/captions/speech/container'; import UploaderContainer from '/imports/ui/components/presentation/presentation-uploader/container'; import CaptionsSpeechContainer from '/imports/ui/components/captions/speech/container'; import RandomUserSelectContainer from '/imports/ui/components/common/modal/random-user/container'; import ScreenReaderAlertContainer from '../screenreader-alert/container'; import WebcamContainer from '../webcam/container'; import PresentationAreaContainer from '../presentation/presentation-area/container'; import ScreenshareContainer from '../screenshare/container'; import ExternalVideoContainer from '../external-video-player/container'; import EmojiRainContainer from '../emoji-rain/container'; import Styled from './styles'; import { DEVICE_TYPE, ACTIONS, SMALL_VIEWPORT_BREAKPOINT, PANELS } from '../layout/enums'; import { isMobile, isTablet, isTabletPortrait, isTabletLandscape, isDesktop, } from '../layout/utils'; import LayoutEngine from '../layout/layout-manager/layoutEngine'; import NavBarContainer from '../nav-bar/container'; import SidebarNavigationContainer from '../sidebar-navigation/container'; import SidebarContentContainer from '../sidebar-content/container'; import { makeCall } from '/imports/ui/services/api'; import ConnectionStatusService from '/imports/ui/components/connection-status/service'; import Settings from '/imports/ui/services/settings'; import { registerTitleView } from '/imports/utils/dom-utils'; import Notifications from '../notifications/container'; import GlobalStyles from '/imports/ui/stylesheets/styled-components/globalStyles'; import ActionsBarContainer from '../actions-bar/container'; import PushLayoutEngine from '../layout/push-layout/pushLayoutEngine'; import AudioService from '/imports/ui/components/audio/service'; import NotesContainer from '/imports/ui/components/notes/container'; import DEFAULT_VALUES from '../layout/defaultValues'; import AppService from '/imports/ui/components/app/service'; import TimerService from '/imports/ui/components/timer/service'; import SpeechService from '/imports/ui/components/audio/captions/speech/service'; const MOBILE_MEDIA = 'only screen and (max-width: 40em)'; const APP_CONFIG = Meteor.settings.public.app; const DESKTOP_FONT_SIZE = APP_CONFIG.desktopFontSize; const MOBILE_FONT_SIZE = APP_CONFIG.mobileFontSize; const LAYOUT_CONFIG = Meteor.settings.public.layout; const CONFIRMATION_ON_LEAVE = Meteor.settings.public.app.askForConfirmationOnLeave; const intlMessages = defineMessages({ userListLabel: { id: 'app.userList.label', description: 'Aria-label for Userlist Nav', }, chatLabel: { id: 'app.chat.label', description: 'Aria-label for Chat Section', }, actionsBarLabel: { id: 'app.actionsBar.label', description: 'Aria-label for ActionsBar Section', }, clearedEmoji: { id: 'app.toast.clearedEmoji.label', description: 'message for cleared emoji status', }, clearedReaction: { id: 'app.toast.clearedReactions.label', description: 'message for cleared reactions', }, setEmoji: { id: 'app.toast.setEmoji.label', description: 'message when a user emoji has been set', }, raisedHand: { id: 'app.toast.setEmoji.raiseHand', description: 'toast message for raised hand notification', }, loweredHand: { id: 'app.toast.setEmoji.lowerHand', description: 'toast message for lowered hand notification', }, away: { id: 'app.toast.setEmoji.away', description: 'toast message for set away notification', }, notAway: { id: 'app.toast.setEmoji.notAway', description: 'toast message for remove away notification', }, meetingMuteOn: { id: 'app.toast.meetingMuteOn.label', description: 'message used when meeting has been muted', }, meetingMuteOff: { id: 'app.toast.meetingMuteOff.label', description: 'message used when meeting has been unmuted', }, pollPublishedLabel: { id: 'app.whiteboard.annotations.poll', description: 'message displayed when a poll is published', }, defaultViewLabel: { id: 'app.title.defaultViewLabel', description: 'view name apended to document title', }, promotedLabel: { id: 'app.toast.promotedLabel', description: 'notification message when promoted', }, demotedLabel: { id: 'app.toast.demotedLabel', description: 'notification message when demoted', }, }); const propTypes = { actionsbar: PropTypes.element, captions: PropTypes.element, darkTheme: PropTypes.bool.isRequired, }; const defaultProps = { actionsbar: null, captions: null, }; const isLayeredView = window.matchMedia(`(max-width: ${SMALL_VIEWPORT_BREAKPOINT}px)`); class App extends Component { constructor(props) { super(props); this.state = { enableResize: !window.matchMedia(MOBILE_MEDIA).matches, isAudioModalOpen: false, isRandomUserSelectModalOpen: false, isVideoPreviewModalOpen: false, presentationFitToWidth: false, }; this.isTimerEnabled = TimerService.isEnabled(); this.timeOffsetInterval = null; this.setPresentationFitToWidth = this.setPresentationFitToWidth.bind(this); this.handleWindowResize = throttle(this.handleWindowResize).bind(this); this.shouldAriaHide = this.shouldAriaHide.bind(this); this.setAudioModalIsOpen = this.setAudioModalIsOpen.bind(this); this.setRandomUserSelectModalIsOpen = this.setRandomUserSelectModalIsOpen.bind(this); this.setVideoPreviewModalIsOpen = this.setVideoPreviewModalIsOpen.bind(this); this.throttledDeviceType = throttle(() => this.setDeviceType(), 50, { trailing: true, leading: true }).bind(this); } componentDidMount() { const { notify, intl, layoutContextDispatch, isRTL, transcriptionSettings, } = this.props; const { browserName } = browserInfo; const { osName } = deviceInfo; registerTitleView(intl.formatMessage(intlMessages.defaultViewLabel)); layoutContextDispatch({ type: ACTIONS.SET_IS_RTL, value: isRTL, }); ReactModal.setAppElement('#app'); const fontSize = isMobile() ? MOBILE_FONT_SIZE : DESKTOP_FONT_SIZE; document.getElementsByTagName('html')[0].style.fontSize = fontSize; layoutContextDispatch({ type: ACTIONS.SET_FONT_SIZE, value: parseInt(fontSize.slice(0, -2), 10), }); const body = document.getElementsByTagName('body')[0]; if (browserName) { body.classList.add(`browser-${browserName.split(' ').pop() .toLowerCase()}`); } body.classList.add(`os-${osName.split(' ').shift().toLowerCase()}`); this.handleWindowResize(); window.addEventListener('resize', this.handleWindowResize, false); window.addEventListener('localeChanged', () => { layoutContextDispatch({ type: ACTIONS.SET_IS_RTL, value: Settings.application.isRTL, }); }); window.ondragover = (e) => { e.preventDefault(); }; window.ondrop = (e) => { e.preventDefault(); }; if (CONFIRMATION_ON_LEAVE) { window.onbeforeunload = (event) => { AudioService.muteMicrophone(); event.stopImmediatePropagation(); event.preventDefault(); // eslint-disable-next-line no-param-reassign event.returnValue = ''; }; } if (deviceInfo.isMobile) makeCall('setMobileUser'); ConnectionStatusService.startRoundTripTime(); if (this.isTimerEnabled) { TimerService.fetchTimeOffset(); this.timeOffsetInterval = setInterval(TimerService.fetchTimeOffset, TimerService.OFFSET_INTERVAL); } if (transcriptionSettings) { const { partialUtterances, minUtteranceLength } = transcriptionSettings; if (partialUtterances !== undefined || minUtteranceLength !== undefined) { logger.info({ logCode: 'app_component_set_speech_options' }, 'Setting initial speech options'); Settings.transcription.partialUtterances = partialUtterances ? true : false; Settings.transcription.minUtteranceLength = parseInt(minUtteranceLength, 10); SpeechService.setSpeechOptions(Settings.transcription.partialUtterances, Settings.transcription.minUtteranceLength); } } logger.info({ logCode: 'app_component_componentdidmount' }, 'Client loaded successfully'); } componentDidUpdate(prevProps) { const { notify, currentUserEmoji, currentUserAway, currentUserRaiseHand, intl, deviceType, mountRandomUserModal, selectedLayout, sidebarContentIsOpen, layoutContextDispatch, numCameras, presentationIsOpen, } = this.props; this.renderDarkMode(); if (mountRandomUserModal) this.setRandomUserSelectModalIsOpen(true); if (prevProps.currentUserEmoji.status !== currentUserEmoji.status && currentUserEmoji.status !== 'raiseHand' && currentUserEmoji.status !== 'away' ) { const formattedEmojiStatus = intl.formatMessage({ id: `app.actionsBar.emojiMenu.${currentUserEmoji.status}Label` }) || currentUserEmoji.status; if (currentUserEmoji.status === 'none') { notify( intl.formatMessage(intlMessages.clearedEmoji), 'info', 'clear_status', ); } else { notify( intl.formatMessage(intlMessages.setEmoji, ({ 0: formattedEmojiStatus })), 'info', 'user', ); } } if (prevProps.currentUserAway !== currentUserAway) { if (currentUserAway === true) { notify(intl.formatMessage(intlMessages.away), 'info', 'user'); } else { notify(intl.formatMessage(intlMessages.notAway), 'info', 'clear_status'); } } if (prevProps.currentUserRaiseHand !== currentUserRaiseHand) { if (currentUserRaiseHand === true) { notify(intl.formatMessage(intlMessages.raisedHand), 'info', 'user'); } else { notify(intl.formatMessage(intlMessages.loweredHand), 'info', 'clear_status'); } } if (deviceType === null || prevProps.deviceType !== deviceType) this.throttledDeviceType(); if ( selectedLayout !== prevProps.selectedLayout && selectedLayout?.toLowerCase?.()?.includes?.('focus') && !sidebarContentIsOpen && deviceType !== DEVICE_TYPE.MOBILE && numCameras > 0 && presentationIsOpen ) { setTimeout(() => { layoutContextDispatch({ type: ACTIONS.SET_SIDEBAR_CONTENT_IS_OPEN, value: true, }); layoutContextDispatch({ type: ACTIONS.SET_ID_CHAT_OPEN, value: DEFAULT_VALUES.idChatOpen, }); layoutContextDispatch({ type: ACTIONS.SET_SIDEBAR_CONTENT_PANEL, value: PANELS.CHAT, }); }, 0); } } componentWillUnmount() { window.removeEventListener('resize', this.handleWindowResize, false); window.onbeforeunload = null; ConnectionStatusService.stopRoundTripTime(); if (this.timeOffsetInterval) { clearInterval(this.timeOffsetInterval); } } setPresentationFitToWidth(presentationFitToWidth) { this.setState({ presentationFitToWidth }); } handleWindowResize() { const { enableResize } = this.state; const shouldEnableResize = !window.matchMedia(MOBILE_MEDIA).matches; if (enableResize === shouldEnableResize) return; this.setState({ enableResize: shouldEnableResize }); this.throttledDeviceType(); } setDeviceType() { const { deviceType, layoutContextDispatch } = this.props; let newDeviceType = null; if (isMobile()) newDeviceType = DEVICE_TYPE.MOBILE; if (isTablet()) newDeviceType = DEVICE_TYPE.TABLET; if (isTabletPortrait()) newDeviceType = DEVICE_TYPE.TABLET_PORTRAIT; if (isTabletLandscape()) newDeviceType = DEVICE_TYPE.TABLET_LANDSCAPE; if (isDesktop()) newDeviceType = DEVICE_TYPE.DESKTOP; if (newDeviceType !== deviceType) { layoutContextDispatch({ type: ACTIONS.SET_DEVICE_TYPE, value: newDeviceType, }); } } shouldAriaHide() { const { sidebarNavigationIsOpen, sidebarContentIsOpen, isPhone } = this.props; return sidebarNavigationIsOpen && sidebarContentIsOpen && (isPhone || isLayeredView.matches); } renderCaptions() { const { captions, captionsStyle, } = this.props; if (!captions) return null; return ( {captions} ); } renderAudioCaptions() { const { audioCaptions, captionsStyle, } = this.props; if (!audioCaptions) return null; return ( {audioCaptions} ); } renderActionsBar() { const { intl, actionsBarStyle, hideActionsBar, setPushLayout, setMeetingLayout, presentationIsOpen, selectedLayout, } = this.props; const { showPushLayoutButton } = LAYOUT_CONFIG; if (hideActionsBar) return null; return ( ); } renderActivityCheck() { const { User } = this.props; const { inactivityCheck, responseDelay } = User; return (inactivityCheck ? ( ) : null); } renderUserInformation() { const { UserInfo, User } = this.props; return (UserInfo.length > 0 ? ( ) : null); } renderDarkMode() { const { darkTheme } = this.props; AppService.setDarkTheme(darkTheme); } mountPushLayoutEngine() { const { cameraWidth, cameraHeight, cameraIsResizing, cameraPosition, focusedCamera, horizontalPosition, isMeetingLayoutResizing, isPresenter, isModerator, layoutContextDispatch, meetingLayout, meetingLayoutCameraPosition, meetingLayoutFocusedCamera, meetingLayoutVideoRate, meetingPresentationIsOpen, meetingLayoutUpdatedAt, presentationIsOpen, presentationVideoRate, pushLayout, pushLayoutMeeting, selectedLayout, setMeetingLayout, setPushLayout, shouldShowScreenshare, shouldShowExternalVideo, } = this.props; return ( ); } setAudioModalIsOpen(value) { this.setState({isAudioModalOpen: value}); } setVideoPreviewModalIsOpen(value) { this.setState({isVideoPreviewModalOpen: value}); } setRandomUserSelectModalIsOpen(value) { const {setMountRandomUserModal} = this.props; this.setState({isRandomUserSelectModalOpen: value}); setMountRandomUserModal(false); } render() { const { customStyle, customStyleUrl, audioAlertEnabled, pushAlertEnabled, shouldShowPresentation, shouldShowScreenshare, shouldShowExternalVideo, shouldShowSharedNotes, isPresenter, selectedLayout, presentationIsOpen, darkTheme, } = this.props; const { isAudioModalOpen, isRandomUserSelectModalOpen, isVideoPreviewModalOpen, presentationFitToWidth } = this.state; return ( <> {this.mountPushLayoutEngine()} {selectedLayout ? : null} {this.renderActivityCheck()} {this.renderUserInformation()} {shouldShowPresentation ? : null} {shouldShowScreenshare ? : null} { shouldShowExternalVideo ? : null } {shouldShowSharedNotes ? ( ) : null} {this.renderCaptions()} {this.renderAudioCaptions()} {(audioAlertEnabled || pushAlertEnabled) && ( )} {this.renderActionsBar()} {customStyleUrl ? : null} {customStyle ? : null} {isRandomUserSelectModalOpen ? this.setRandomUserSelectModalIsOpen(false), priority: "low", setIsOpen: this.setRandomUserSelectModalIsOpen, isOpen: isRandomUserSelectModalOpen, }} /> : null} ); } } App.propTypes = propTypes; App.defaultProps = defaultProps; export default injectIntl(App);