You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			235 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			235 lines
		
	
	
		
			6.8 KiB
		
	
	
	
		
			TypeScript
		
	
| import { useRef, useState } from 'react';
 | |
| import { useSelector } from 'react-redux';
 | |
| 
 | |
| import moment from 'moment';
 | |
| import useInterval from 'react-use/lib/useInterval';
 | |
| import styled from 'styled-components';
 | |
| import { CallManager, UserUtils } from '../../session/utils';
 | |
| import {
 | |
|   getCallIsInFullScreen,
 | |
|   getCallWithFocusedConvoIsOffering,
 | |
|   getCallWithFocusedConvosIsConnected,
 | |
|   getCallWithFocusedConvosIsConnecting,
 | |
|   getHasOngoingCallWithFocusedConvo,
 | |
|   getHasOngoingCallWithPubkey,
 | |
| } from '../../state/selectors/call';
 | |
| import { Avatar, AvatarSize } from '../avatar/Avatar';
 | |
| import { StyledVideoElement } from './DraggableCallContainer';
 | |
| 
 | |
| import { useModuloWithTripleDots } from '../../hooks/useModuloWithTripleDots';
 | |
| import { useVideoCallEventsListener } from '../../hooks/useVideoEventListener';
 | |
| import { DEVICE_DISABLED_DEVICE_ID } from '../../session/utils/calling/CallManager';
 | |
| import { CallWindowControls } from './CallButtons';
 | |
| 
 | |
| import { SessionSpinner } from '../loading';
 | |
| 
 | |
| const VideoContainer = styled.div`
 | |
|   height: 100%;
 | |
|   width: 50%;
 | |
|   z-index: 0;
 | |
|   padding-top: 30px; // leave some space at the top for the connecting/duration of the current call
 | |
| `;
 | |
| 
 | |
| const InConvoCallWindow = styled.div`
 | |
|   padding: 1rem;
 | |
|   display: flex;
 | |
| 
 | |
|   background-color: var(--in-call-container-background-color);
 | |
| 
 | |
|   flex-shrink: 1;
 | |
|   min-height: 80px;
 | |
|   align-items: center;
 | |
|   flex-grow: 1;
 | |
| `;
 | |
| 
 | |
| const RelativeCallWindow = styled.div`
 | |
|   position: relative;
 | |
|   height: 100%;
 | |
|   display: flex;
 | |
|   flex-grow: 1;
 | |
| `;
 | |
| 
 | |
| const CenteredAvatarInConversation = styled.div`
 | |
|   top: -50%;
 | |
|   transform: translateY(-50%);
 | |
|   position: relative;
 | |
|   bottom: 0;
 | |
|   left: 0;
 | |
|   right: 50%;
 | |
| 
 | |
|   display: flex;
 | |
|   justify-content: center;
 | |
|   align-items: center;
 | |
| `;
 | |
| 
 | |
| const StyledCenteredLabel = styled.div`
 | |
|   position: absolute;
 | |
|   left: 50%;
 | |
|   transform: translateX(-50%);
 | |
|   height: min-content;
 | |
|   white-space: nowrap;
 | |
|   color: var(--in-call-container-text-color);
 | |
|   z-index: 5;
 | |
| `;
 | |
| 
 | |
| const RingingLabel = () => {
 | |
|   const ongoingCallWithFocusedIsRinging = useSelector(getCallWithFocusedConvoIsOffering);
 | |
| 
 | |
|   const modulatedStr = useModuloWithTripleDots(window.i18n('ringing'), 3, 1000);
 | |
|   if (!ongoingCallWithFocusedIsRinging) {
 | |
|     return null;
 | |
|   }
 | |
|   return <StyledCenteredLabel>{modulatedStr}</StyledCenteredLabel>;
 | |
| };
 | |
| 
 | |
| const ConnectingLabel = () => {
 | |
|   const ongoingCallWithFocusedIsConnecting = useSelector(getCallWithFocusedConvosIsConnecting);
 | |
| 
 | |
|   const modulatedStr = useModuloWithTripleDots(window.i18n('establishingConnection'), 3, 1000);
 | |
| 
 | |
|   if (!ongoingCallWithFocusedIsConnecting) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   return <StyledCenteredLabel>{modulatedStr}</StyledCenteredLabel>;
 | |
| };
 | |
| 
 | |
| const DurationLabel = () => {
 | |
|   const [callDuration, setCallDuration] = useState<undefined | number>(undefined);
 | |
|   const ongoingCallWithFocusedIsConnected = useSelector(getCallWithFocusedConvosIsConnected);
 | |
| 
 | |
|   useInterval(() => {
 | |
|     const duration = CallManager.getCurrentCallDuration();
 | |
|     if (duration) {
 | |
|       setCallDuration(duration);
 | |
|     }
 | |
|   }, 100);
 | |
| 
 | |
|   if (!ongoingCallWithFocusedIsConnected || !callDuration || callDuration < 0) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   const ms = callDuration * 1000;
 | |
|   const d = moment.duration(ms);
 | |
| 
 | |
|   const dateString = Math.floor(d.asHours()) + moment.utc(ms).format(':mm:ss');
 | |
|   return <StyledCenteredLabel>{dateString}</StyledCenteredLabel>;
 | |
| };
 | |
| 
 | |
| const StyledSpinner = styled.div<{ fullWidth: boolean }>`
 | |
|   height: 100%;
 | |
|   width: ${props => (props.fullWidth ? '100%' : '50%')};
 | |
|   display: flex;
 | |
|   justify-content: center;
 | |
|   align-items: center;
 | |
|   position: absolute;
 | |
|   z-index: -1;
 | |
| `;
 | |
| 
 | |
| export const VideoLoadingSpinner = (props: { fullWidth: boolean }) => {
 | |
|   return (
 | |
|     <StyledSpinner fullWidth={props.fullWidth}>
 | |
|       <SessionSpinner loading={true} />
 | |
|     </StyledSpinner>
 | |
|   );
 | |
| };
 | |
| 
 | |
| export const InConversationCallContainer = () => {
 | |
|   const isInFullScreen = useSelector(getCallIsInFullScreen);
 | |
| 
 | |
|   const ongoingCallPubkey = useSelector(getHasOngoingCallWithPubkey);
 | |
|   const ongoingCallWithFocused = useSelector(getHasOngoingCallWithFocusedConvo);
 | |
|   const videoRefRemote = useRef<HTMLVideoElement>(null);
 | |
|   const videoRefLocal = useRef<HTMLVideoElement>(null);
 | |
| 
 | |
|   const ourPubkey = UserUtils.getOurPubKeyStrFromCache();
 | |
| 
 | |
|   const {
 | |
|     currentConnectedAudioInputs,
 | |
|     currentConnectedCameras,
 | |
|     currentConnectedAudioOutputs,
 | |
|     currentSelectedAudioOutput,
 | |
|     localStream,
 | |
|     localStreamVideoIsMuted,
 | |
|     remoteStream,
 | |
|     remoteStreamVideoIsMuted,
 | |
|     isAudioMuted,
 | |
|     isAudioOutputMuted,
 | |
|   } = useVideoCallEventsListener('InConversationCallContainer', true);
 | |
| 
 | |
|   if (videoRefRemote?.current && videoRefLocal?.current) {
 | |
|     if (videoRefRemote.current.srcObject !== remoteStream) {
 | |
|       videoRefRemote.current.srcObject = remoteStream;
 | |
|     }
 | |
| 
 | |
|     if (videoRefLocal.current.srcObject !== localStream) {
 | |
|       videoRefLocal.current.srcObject = localStream;
 | |
|     }
 | |
| 
 | |
|     if (videoRefRemote.current) {
 | |
|       if (currentSelectedAudioOutput === DEVICE_DISABLED_DEVICE_ID) {
 | |
|         videoRefRemote.current.muted = true;
 | |
|       } else {
 | |
|         void (videoRefRemote.current as any)?.setSinkId(currentSelectedAudioOutput);
 | |
|         videoRefRemote.current.muted = false;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (isInFullScreen && videoRefRemote.current) {
 | |
|     // disable this video element so the one in fullscreen is the only one playing audio
 | |
|     videoRefRemote.current.muted = true;
 | |
|   }
 | |
| 
 | |
|   if (!ongoingCallWithFocused || !ongoingCallPubkey) {
 | |
|     return null;
 | |
|   }
 | |
| 
 | |
|   return (
 | |
|     <InConvoCallWindow>
 | |
|       <RelativeCallWindow>
 | |
|         <RingingLabel />
 | |
|         <ConnectingLabel />
 | |
|         <DurationLabel />
 | |
|         <VideoContainer>
 | |
|           <VideoLoadingSpinner fullWidth={false} />
 | |
|           <StyledVideoElement
 | |
|             ref={videoRefRemote}
 | |
|             autoPlay={true}
 | |
|             isVideoMuted={remoteStreamVideoIsMuted}
 | |
|           />
 | |
|           {remoteStreamVideoIsMuted && (
 | |
|             <CenteredAvatarInConversation>
 | |
|               <Avatar size={AvatarSize.XL} pubkey={ongoingCallPubkey} />
 | |
|             </CenteredAvatarInConversation>
 | |
|           )}
 | |
|         </VideoContainer>
 | |
|         <VideoContainer>
 | |
|           <StyledVideoElement
 | |
|             ref={videoRefLocal}
 | |
|             autoPlay={true}
 | |
|             muted={true}
 | |
|             isVideoMuted={localStreamVideoIsMuted}
 | |
|           />
 | |
|           {localStreamVideoIsMuted && (
 | |
|             <CenteredAvatarInConversation>
 | |
|               <Avatar size={AvatarSize.XL} pubkey={ourPubkey} />
 | |
|             </CenteredAvatarInConversation>
 | |
|           )}
 | |
|         </VideoContainer>
 | |
| 
 | |
|         <CallWindowControls
 | |
|           currentConnectedAudioInputs={currentConnectedAudioInputs}
 | |
|           currentConnectedCameras={currentConnectedCameras}
 | |
|           isAudioMuted={isAudioMuted}
 | |
|           currentConnectedAudioOutputs={currentConnectedAudioOutputs}
 | |
|           isAudioOutputMuted={isAudioOutputMuted}
 | |
|           localStreamVideoIsMuted={localStreamVideoIsMuted}
 | |
|           remoteStreamVideoIsMuted={remoteStreamVideoIsMuted}
 | |
|           isFullScreen={false}
 | |
|         />
 | |
|       </RelativeCallWindow>
 | |
|     </InConvoCallWindow>
 | |
|   );
 | |
| };
 |