import React, { Component } from 'react'; import PropTypes from 'prop-types'; import { throttle } from 'lodash'; import { defineMessages, injectIntl } from 'react-intl'; import Modal 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 '../toast/container'; import ModalContainer from '../modal/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 WaitingNotifierContainer from '/imports/ui/components/waiting-users/alert/container'; import LockNotifier from '/imports/ui/components/lock-viewers/notify/container'; import StatusNotifier from '/imports/ui/components/status-notifier/container'; import MediaService from '/imports/ui/components/media/service'; import ManyWebcamsNotifier from '/imports/ui/components/video-provider/many-users-notify/container'; import UploaderContainer from '/imports/ui/components/presentation/presentation-uploader/container'; import RandomUserSelectContainer from '/imports/ui/components/modal/random-user/container'; import NewWebcamContainer from '../webcam/container'; import PresentationAreaContainer from '../presentation/presentation-area/container'; import ScreenshareContainer from '../screenshare/container'; import ExternalVideoContainer from '../external-video-player/container'; import Styled from './styles'; import { DEVICE_TYPE, ACTIONS } 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 LayoutService from '/imports/ui/components/layout/service'; import { registerTitleView } from '/imports/utils/dom-utils'; import GlobalStyles from '/imports/ui/stylesheets/styled-components/globalStyles'; 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 OVERRIDE_LOCALE = APP_CONFIG.defaultSettings.application.overrideLocale; 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', }, iOSWarning: { id: 'app.iOSWarning.label', description: 'message indicating to upgrade ios version', }, clearedEmoji: { id: 'app.toast.clearedEmoji.label', description: 'message for cleared emoji status', }, 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', }, 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, locale: PropTypes.string, }; const defaultProps = { actionsbar: null, captions: null, locale: OVERRIDE_LOCALE || navigator.language, }; const LAYERED_BREAKPOINT = 640; const isLayeredView = window.matchMedia(`(max-width: ${LAYERED_BREAKPOINT}px)`); class App extends Component { static renderWebcamsContainer() { return ; } constructor(props) { super(props); this.state = { enableResize: !window.matchMedia(MOBILE_MEDIA).matches, }; this.handleWindowResize = throttle(this.handleWindowResize).bind(this); this.shouldAriaHide = this.shouldAriaHide.bind(this); this.renderWebcamsContainer = App.renderWebcamsContainer.bind(this); this.throttledDeviceType = throttle(() => this.setDeviceType(), 50, { trailing: true, leading: true }).bind(this); } componentDidMount() { const { locale, notify, intl, validIOSVersion, layoutContextDispatch, meetingLayout, settingsLayout, isRTL, hidePresentation, } = this.props; const { browserName } = browserInfo; const { osName } = deviceInfo; registerTitleView(intl.formatMessage(intlMessages.defaultViewLabel)); layoutContextDispatch({ type: ACTIONS.SET_IS_RTL, value: isRTL, }); layoutContextDispatch({ type: ACTIONS.SET_PRESENTATION_IS_OPEN, value: !hidePresentation, }); MediaService.setSwapLayout(layoutContextDispatch); Modal.setAppElement('#app'); const fontSize = isMobile() ? MOBILE_FONT_SIZE : DESKTOP_FONT_SIZE; document.getElementsByTagName('html')[0].lang = locale; document.getElementsByTagName('html')[0].style.fontSize = fontSize; layoutContextDispatch({ type: ACTIONS.SET_FONT_SIZE, value: parseInt(fontSize.slice(0, -2), 10), }); const currentLayout = settingsLayout || meetingLayout; Settings.application.selectedLayout = currentLayout; Settings.save(); const body = document.getElementsByTagName('body')[0]; if (browserName) { body.classList.add(`browser-${browserName.split(' ').pop() .toLowerCase()}`); } body.classList.add(`os-${osName.split(' ').shift().toLowerCase()}`); if (!validIOSVersion()) { notify( intl.formatMessage(intlMessages.iOSWarning), 'error', 'warning', ); } 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 (isMobile()) makeCall('setMobileUser'); ConnectionStatusService.startRoundTripTime(); logger.info({ logCode: 'app_component_componentdidmount' }, 'Client loaded successfully'); } componentDidUpdate(prevProps) { const { meetingMuted, notify, currentUserEmoji, currentUserRole, intl, hasPublishedPoll, mountModal, deviceType, meetingLayout, selectedLayout, // full layout name settingsLayout, // shortened layout name (without Push) layoutType, pushLayoutToEveryone, // is layout pushed layoutContextDispatch, mountRandomUserModal, } = this.props; if (meetingLayout !== prevProps.meetingLayout) { layoutContextDispatch({ type: ACTIONS.SET_LAYOUT_TYPE, value: meetingLayout, }); Settings.application.selectedLayout = meetingLayout; Settings.save(); } if (selectedLayout !== prevProps.selectedLayout || settingsLayout !== layoutType) { layoutContextDispatch({ type: ACTIONS.SET_LAYOUT_TYPE, value: settingsLayout, }); if (pushLayoutToEveryone) { LayoutService.setMeetingLayout(settingsLayout); } } if (mountRandomUserModal) mountModal(); if (prevProps.currentUserEmoji.status !== currentUserEmoji.status) { const formattedEmojiStatus = intl.formatMessage({ id: `app.actionsBar.emojiMenu.${currentUserEmoji.status}Label` }) || currentUserEmoji.status; const raisedHand = currentUserEmoji.status === 'raiseHand'; let statusLabel = ''; if (currentUserEmoji.status === 'none') { statusLabel = prevProps.currentUserEmoji.status === 'raiseHand' ? intl.formatMessage(intlMessages.loweredHand) : intl.formatMessage(intlMessages.clearedEmoji); } else { statusLabel = raisedHand ? intl.formatMessage(intlMessages.raisedHand) : intl.formatMessage(intlMessages.setEmoji, ({ 0: formattedEmojiStatus })); } notify( statusLabel, 'info', currentUserEmoji.status === 'none' ? 'clear_status' : 'user', ); } if (!prevProps.meetingMuted && meetingMuted) { notify( intl.formatMessage(intlMessages.meetingMuteOn), 'info', 'mute', ); } if (prevProps.meetingMuted && !meetingMuted) { notify( intl.formatMessage(intlMessages.meetingMuteOff), 'info', 'unmute', ); } if (!prevProps.hasPublishedPoll && hasPublishedPoll) { notify( intl.formatMessage(intlMessages.pollPublishedLabel), 'info', 'polling', ); } if (prevProps.currentUserRole === 'VIEWER' && currentUserRole === 'MODERATOR') { notify( intl.formatMessage(intlMessages.promotedLabel), 'info', 'user', ); } if (prevProps.currentUserRole === 'MODERATOR' && currentUserRole === 'VIEWER') { notify( intl.formatMessage(intlMessages.demotedLabel), 'info', 'user', ); } if (deviceType === null || prevProps.deviceType !== deviceType) this.throttledDeviceType(); } componentWillUnmount() { window.removeEventListener('resize', this.handleWindowResize, false); ConnectionStatusService.stopRoundTripTime(); } 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} ); } renderActionsBar() { const { actionsbar, intl, actionsBarStyle, hideActionsBar, } = this.props; if (!actionsbar || hideActionsBar) return null; return ( {actionsbar} ); } renderActivityCheck() { const { User } = this.props; const { inactivityCheck, responseDelay } = User; return (inactivityCheck ? ( ) : null); } renderUserInformation() { const { UserInfo, User } = this.props; return (UserInfo.length > 0 ? ( ) : null); } render() { const { customStyle, customStyleUrl, audioAlertEnabled, pushAlertEnabled, shouldShowPresentation, shouldShowScreenshare, shouldShowExternalVideo, isPresenter, layoutType, } = this.props; return ( <> {this.renderActivityCheck()} {this.renderUserInformation()} {this.renderWebcamsContainer()} {shouldShowPresentation ? : null} {shouldShowScreenshare ? : null} { shouldShowExternalVideo ? : null } {this.renderCaptions()} {(audioAlertEnabled || pushAlertEnabled) && ( )} {this.renderActionsBar()} {customStyleUrl ? : null} {customStyle ? : null} ); } } App.propTypes = propTypes; App.defaultProps = defaultProps; export default injectIntl(App);