import React, { memo, useEffect, useState } from 'react'; import { FetchResult } from '@apollo/client'; import ButtonEmoji from '/imports/ui/components/common/button/button-emoji/ButtonEmoji'; import { IntlShape, defineMessages, injectIntl } from 'react-intl'; import deviceInfo from '/imports/utils/deviceInfo'; import { debounce } from '/imports/utils/debounce'; import BBBMenu from '/imports/ui/components/common/menu/component'; import Button from '/imports/ui/components/common/button/component'; import VideoPreviewContainer from '/imports/ui/components/video-preview/container'; import { CameraSettingsDropdownItemType } from 'bigbluebutton-html-plugin-sdk/dist/cjs/extensible-areas/camera-settings-dropdown-item/enums'; import { getSettingsSingletonInstance } from '/imports/ui/services/settings'; import { CameraSettingsDropdownInterface } from 'bigbluebutton-html-plugin-sdk'; import VideoService from '../service'; import Styled from './styles'; const intlMessages = defineMessages({ videoSettings: { id: 'app.video.videoSettings', description: 'Open video settings', }, visualEffects: { id: 'app.video.visualEffects', description: 'Visual effects label', }, joinVideo: { id: 'app.video.joinVideo', description: 'Join video button label', }, leaveVideo: { id: 'app.video.leaveVideo', description: 'Leave video button label', }, advancedVideo: { id: 'app.video.advancedVideo', description: 'Open advanced video label', }, videoLocked: { id: 'app.video.videoLocked', description: 'video disabled label', }, videoConnecting: { id: 'app.video.connecting', description: 'video connecting label', }, camCapReached: { id: 'app.video.meetingCamCapReached', description: 'meeting camera cap label', }, meteorDisconnected: { id: 'app.video.clientDisconnected', description: 'Meteor disconnected label', }, }); const JOIN_VIDEO_DELAY_MILLISECONDS = 500; interface JoinVideoButtonProps { cameraSettingsDropdownItems: CameraSettingsDropdownInterface[]; hasVideoStream: boolean; updateSettings: ( obj: object, msgDescriptor: object | null, mutation: (settings: Record) => void, ) => void; disableReason: string | undefined; status: string; setLocalSettings: (settings: Record) => Promise>; exitVideo: () => void; stopVideo: (cameraId?: string | undefined) => void; intl: IntlShape; } const JoinVideoButton: React.FC = ({ intl, hasVideoStream, status, disableReason, updateSettings, cameraSettingsDropdownItems, setLocalSettings, exitVideo: exit, stopVideo, }) => { const { isMobile } = deviceInfo; const isMobileSharingCamera = hasVideoStream && isMobile; const isDesktopSharingCamera = hasVideoStream && !isMobile; const ENABLE_WEBCAM_SELECTOR_BUTTON = window.meetingClientSettings.public.app.enableWebcamSelectorButton; const shouldEnableWebcamSelectorButton = ENABLE_WEBCAM_SELECTOR_BUTTON && isDesktopSharingCamera; const exitVideo = () => isDesktopSharingCamera && (!VideoService.isMultipleCamerasEnabled() || shouldEnableWebcamSelectorButton); const [propsToPassModal, setPropsToPassModal] = useState<{ isVisualEffects?: boolean }>({}); const [forceOpen, setForceOpen] = useState(false); const [isVideoPreviewModalOpen, setVideoPreviewModalIsOpen] = useState(false); const [wasSelfViewDisabled, setWasSelfViewDisabled] = useState(false); useEffect(() => { const Settings = getSettingsSingletonInstance(); const isSelfViewDisabled = Settings.application.selfViewDisable; if (isVideoPreviewModalOpen && isSelfViewDisabled) { setWasSelfViewDisabled(true); const obj = { application: { ...Settings.application, selfViewDisable: false }, }; updateSettings(obj, null, setLocalSettings); } }, [isVideoPreviewModalOpen]); const handleOnClick = debounce(() => { switch (status) { case 'videoConnecting': stopVideo(); break; case 'connected': default: if (exitVideo()) { exit(); } else { setForceOpen(isMobileSharingCamera); setVideoPreviewModalIsOpen(true); } } }, JOIN_VIDEO_DELAY_MILLISECONDS); const handleOpenAdvancedOptions = (callback?: () => void) => { if (callback) callback(); setForceOpen(isDesktopSharingCamera); setVideoPreviewModalIsOpen(true); }; const getMessageFromStatus = () => { let statusMessage = status; if (status !== 'videoConnecting') { statusMessage = exitVideo() ? 'leaveVideo' : 'joinVideo'; } return statusMessage; }; const label = disableReason ? intl.formatMessage(intlMessages[disableReason as keyof typeof intlMessages]) : intl.formatMessage(intlMessages[getMessageFromStatus() as keyof typeof intlMessages]); const renderUserActions = () => { const actions = []; if (shouldEnableWebcamSelectorButton) { actions.push( { key: 'advancedVideo', label: intl.formatMessage(intlMessages.advancedVideo), onClick: () => handleOpenAdvancedOptions(), dataTest: 'advancedVideoSettingsButton', }, ); } if (actions.length === 0) return null; const customStyles = { top: '-3.6rem' }; cameraSettingsDropdownItems.forEach((plugin) => { switch (plugin.type) { case CameraSettingsDropdownItemType.OPTION: actions.push({ key: plugin.id, // @ts-expect-error -> Plugin-related. label: plugin.label, // @ts-expect-error -> Plugin-related. onClick: plugin.onClick, // @ts-expect-error -> Plugin-related. icon: plugin.icon, }); break; case CameraSettingsDropdownItemType.SEPARATOR: actions.push({ key: plugin.id, isSeparator: true, }); break; default: break; } }); return ( )} actions={actions} opts={{ id: 'video-dropdown-menu', keepMounted: true, transitionDuration: 0, elevation: 3, getcontentanchorel: null, fullwidth: 'true', anchorOrigin: { vertical: 'top', horizontal: 'center' }, transformOrigin: { vertical: 'top', horizontal: 'center' }, }} /> ); }; return ( <>