import React, { useEffect, useRef, useState } from 'react'; import { withTracker } from 'meteor/react-meteor-data'; import Auth from '/imports/ui/services/auth'; import Users from '/imports/api/users'; import Meetings, { LayoutMeetings } from '/imports/api/meetings'; import AudioCaptionsLiveContainer from '/imports/ui/components/audio/captions/live/container'; import AudioCaptionsService from '/imports/ui/components/audio/captions/service'; import { notify } from '/imports/ui/services/notification'; import CaptionsContainer from '/imports/ui/components/captions/live/container'; import CaptionsService from '/imports/ui/components/captions/service'; import getFromUserSettings from '/imports/ui/services/users-settings'; import deviceInfo from '/imports/utils/deviceInfo'; import UserInfos from '/imports/api/users-infos'; import Settings from '/imports/ui/services/settings'; import MediaService from '/imports/ui/components/media/service'; import LayoutService from '/imports/ui/components/layout/service'; import { isPresentationEnabled, isExternalVideoEnabled } from '/imports/ui/services/features'; import { layoutSelect, layoutSelectInput, layoutSelectOutput, layoutDispatch, } from '../layout/context'; import { isEqual } from 'radash'; import useCurrentUser from '/imports/ui/core/hooks/useCurrentUser'; import useMeeting from '/imports/ui/core/hooks/useMeeting'; import { LAYOUT_TYPE } from '/imports/ui/components/layout/enums'; import { useMutation } from '@apollo/client'; import { SET_MOBILE_FLAG } from '/imports/ui/core/graphql/mutations/userMutations'; import { getFontSize, getBreakoutRooms, } from './service'; import App from './component'; const CUSTOM_STYLE_URL = Meteor.settings.public.app.customStyleUrl; const endMeeting = (code, ejectedReason) => { Session.set('codeError', code); Session.set('errorMessageDescription', ejectedReason); Session.set('isMeetingEnded', true); }; const AppContainer = (props) => { function usePrevious(value) { const ref = useRef(); useEffect(() => { ref.current = value; }); return ref.current; } const layoutType = useRef(null); const { actionsbar, selectedLayout, pushLayout, pushLayoutMeeting, currentUserId, shouldShowScreenshare: propsShouldShowScreenshare, shouldShowSharedNotes, presentationRestoreOnUpdate, randomlySelectedUser, isModalOpen, meetingLayout, meetingLayoutUpdatedAt, meetingPresentationIsOpen, isMeetingLayoutResizing, meetingLayoutCameraPosition, meetingLayoutFocusedCamera, meetingLayoutVideoRate, isSharedNotesPinned, viewScreenshare, ...otherProps } = props; const sidebarContent = layoutSelectInput((i) => i.sidebarContent); const sidebarNavigation = layoutSelectInput((i) => i.sidebarNavigation); const actionsBarStyle = layoutSelectOutput((i) => i.actionBar); const captionsStyle = layoutSelectOutput((i) => i.captions); const cameraDock = layoutSelectOutput((i) => i.cameraDock); const cameraDockInput = layoutSelectInput((i) => i.cameraDock); const presentation = layoutSelectInput((i) => i.presentation); const deviceType = layoutSelect((i) => i.deviceType); const layoutContextDispatch = layoutDispatch(); const [setMobileFlag] = useMutation(SET_MOBILE_FLAG); const setMobileUser = (mobile) => { setMobileFlag({ variables: { mobile, }, }); }; const { data: currentUserData } = useCurrentUser((user) => ({ enforceLayout: user.enforceLayout, isModerator: user.isModerator, presenter: user.presenter, })); const isModerator = currentUserData?.isModerator; const isPresenter = currentUserData?.presenter; const { sidebarContentPanel, isOpen: sidebarContentIsOpen } = sidebarContent; const { sidebarNavPanel, isOpen: sidebarNavigationIsOpen } = sidebarNavigation; const { isOpen } = presentation; const presentationIsOpen = isOpen; const { focusedId } = cameraDock; useEffect(() => { if ( layoutContextDispatch && (typeof meetingLayout !== 'undefined') && (layoutType.current !== meetingLayout) && isSharedNotesPinned ) { layoutType.current = meetingLayout; MediaService.setPresentationIsOpen(layoutContextDispatch, true); } }, [meetingLayout, layoutContextDispatch, layoutType]); const horizontalPosition = cameraDock.position === 'contentLeft' || cameraDock.position === 'contentRight'; // this is not exactly right yet let presentationVideoRate; if (horizontalPosition) { presentationVideoRate = cameraDock.width / window.innerWidth; } else { presentationVideoRate = cameraDock.height / window.innerHeight; } presentationVideoRate = parseFloat(presentationVideoRate.toFixed(2)); const prevRandomUser = usePrevious(randomlySelectedUser); const [mountRandomUserModal, setMountRandomUserModal] = useState(false); useEffect(() => { setMountRandomUserModal(!isPresenter && !isEqual(prevRandomUser, randomlySelectedUser) && randomlySelectedUser.length > 0 && !isModalOpen); }, [isPresenter, prevRandomUser, randomlySelectedUser, isModalOpen]); const setPushLayout = () => { LayoutService.setPushLayout(pushLayout); } const setMeetingLayout = () => { const { isResizing } = cameraDockInput; LayoutService.setMeetingLayout({ layout: selectedLayout, presentationIsOpen, isResizing, cameraPosition: cameraDock.position, focusedCamera: focusedId, presentationVideoRate, pushLayout, }); }; const { data: currentMeeting } = useMeeting((m) => ({ externalVideo: m.externalVideo, })); const isSharingVideo = !!currentMeeting?.externalVideo?.externalVideoUrl; useEffect(() => { MediaService.buildLayoutWhenPresentationAreaIsDisabled(layoutContextDispatch, isSharingVideo); }); const shouldShowExternalVideo = isExternalVideoEnabled() && isSharingVideo; const validateEnforceLayout = (currentUser) => { const layoutTypes = Object.values(LAYOUT_TYPE); const enforceLayout = currentUser?.enforceLayout; return enforceLayout && layoutTypes.includes(enforceLayout) ? enforceLayout : null; }; const shouldShowScreenshare = propsShouldShowScreenshare && (viewScreenshare || isPresenter); const shouldShowPresentation = (!shouldShowScreenshare && !shouldShowSharedNotes && !shouldShowExternalVideo && (presentationIsOpen || presentationRestoreOnUpdate)) && isPresentationEnabled(); return currentUserId ? ( ) : null; }; const currentUserEmoji = (currentUser) => (currentUser ? { status: currentUser.emoji, changedAt: currentUser.emojiTime, } : { status: 'none', changedAt: null, } ); export default withTracker(() => { Users.find({ userId: Auth.userID, meetingId: Auth.meetingID }).observe({ removed(userData) { // wait 3secs (before endMeeting), client will try to authenticate again const delayForReconnection = userData.ejected ? 0 : 3000; setTimeout(() => { const queryCurrentUser = Users.find({ userId: Auth.userID, meetingId: Auth.meetingID }); if (queryCurrentUser.count() === 0) { if (userData.ejected) { endMeeting('403', userData.ejectedReason); } else { // Either authentication process hasn't finished yet or user did authenticate but Users // collection is unsynchronized. In both cases user may be able to rejoin. const description = Auth.isAuthenticating || Auth.loggedIn ? 'able_to_rejoin_user_disconnected_reason' : null; endMeeting('503', description); } } }, delayForReconnection); }, }); const currentUser = Users.findOne( { userId: Auth.userID }, { fields: { approved: 1, emoji: 1, raiseHand: 1, away: 1, userId: 1, role: 1, }, }, ); const currentMeeting = Meetings.findOne({ meetingId: Auth.meetingID }, { fields: { randomlySelectedUser: 1, layout: 1, }, }); const { randomlySelectedUser, } = currentMeeting; const meetingLayoutObj = LayoutMeetings.findOne({ meetingId: Auth.meetingID }) || {}; const { layout: meetingLayout, pushLayout: pushLayoutMeeting, layoutUpdatedAt: meetingLayoutUpdatedAt, presentationIsOpen: meetingPresentationIsOpen, isResizing: isMeetingLayoutResizing, cameraPosition: meetingLayoutCameraPosition, focusedCamera: meetingLayoutFocusedCamera, presentationVideoRate: meetingLayoutVideoRate, } = meetingLayoutObj; const UserInfo = UserInfos.find({ meetingId: Auth.meetingID, requesterUserId: Auth.userID, }).fetch(); const AppSettings = Settings.application; const { selectedLayout, pushLayout } = AppSettings; const { viewScreenshare } = Settings.dataSaving; const shouldShowSharedNotes = MediaService.shouldShowSharedNotes(); const shouldShowScreenshare = MediaService.shouldShowScreenshare(); let customStyleUrl = getFromUserSettings('bbb_custom_style_url', false); if (!customStyleUrl && CUSTOM_STYLE_URL) { customStyleUrl = CUSTOM_STYLE_URL; } const LAYOUT_CONFIG = Meteor.settings.public.layout; return { captions: CaptionsService.isCaptionsActive() ? : null, audioCaptions: AudioCaptionsService.getAudioCaptions() ? : null, fontSize: getFontSize(), hasBreakoutRooms: getBreakoutRooms().length > 0, customStyle: getFromUserSettings('bbb_custom_style', false), customStyleUrl, UserInfo, notify, isPhone: deviceInfo.isPhone, isRTL: document.documentElement.getAttribute('dir') === 'rtl', currentUserEmoji: currentUserEmoji(currentUser), currentUserAway: currentUser.away, currentUserRaiseHand: currentUser.raiseHand, randomlySelectedUser, currentUserId: currentUser?.userId, meetingLayout, meetingLayoutUpdatedAt, meetingPresentationIsOpen, isMeetingLayoutResizing, meetingLayoutCameraPosition, meetingLayoutFocusedCamera, meetingLayoutVideoRate, selectedLayout, pushLayout, pushLayoutMeeting, audioAlertEnabled: AppSettings.chatAudioAlerts, pushAlertEnabled: AppSettings.chatPushAlerts, darkTheme: AppSettings.darkTheme, shouldShowScreenshare, viewScreenshare, shouldShowSharedNotes, isLargeFont: Session.get('isLargeFont'), presentationRestoreOnUpdate: getFromUserSettings( 'bbb_force_restore_presentation_on_new_events', Meteor.settings.public.presentation.restoreOnUpdate, ), hidePresentationOnJoin: getFromUserSettings('bbb_hide_presentation_on_join', LAYOUT_CONFIG.hidePresentationOnJoin), hideActionsBar: getFromUserSettings('bbb_hide_actions_bar', false), hideNavBar: getFromUserSettings('bbb_hide_nav_bar', false), ignorePollNotifications: Session.get('ignorePollNotifications'), isSharedNotesPinned: MediaService.shouldShowSharedNotes(), }; })(AppContainer);