add hook to listen for video call events

pull/2015/head
Audric Ackermann 3 years ago
parent 6a1f575c46
commit a0b33fbbbb
No known key found for this signature in database
GPG Key ID: 999F434D76324AD4

@ -46,7 +46,7 @@ import { loadDefaultRooms } from '../../opengroup/opengroupV2/ApiUtil';
import { ActionPanelOnionStatusLight } from '../dialog/OnionStatusPathDialog';
import { switchHtmlToDarkTheme, switchHtmlToLightTheme } from '../../state/ducks/SessionTheme';
import { DraggableCallContainer } from './calling/CallContainer';
import { DraggableCallContainer } from './calling/DraggableCallContainer';
import { IncomingCallDialog } from './calling/IncomingCallDialog';
import { CallInFullScreenContainer } from './calling/CallInFullScreenContainer';

@ -1,4 +1,4 @@
import React, { useCallback, useEffect, useRef, useState } from 'react';
import React, { useEffect, useRef, useState } from 'react';
import { useSelector } from 'react-redux';
import Draggable, { DraggableData, DraggableEvent } from 'react-draggable';
@ -11,8 +11,8 @@ import {
} from '../../../state/selectors/conversations';
import { openConversationWithMessages } from '../../../state/ducks/conversations';
import { Avatar, AvatarSize } from '../../Avatar';
import { getConversationController } from '../../../session/conversations';
import { useVideoCallEventsListener } from '../../../hooks/useVideoEventListener';
import { useAvatarPath, useConversationUsername } from '../../../hooks/useParamSelector';
export const DraggableCallWindow = styled.div`
position: absolute;
@ -74,9 +74,12 @@ export const DraggableCallContainer = () => {
const ongoingCallPubkey = ongoingCallProps?.id;
const { remoteStreamVideoIsMuted, remoteStream } = useVideoCallEventsListener(
'DraggableCallContainer'
'DraggableCallContainer',
false
);
const videoRefRemote = useRef<any>(undefined);
const ongoingCallUsername = useConversationUsername(ongoingCallPubkey);
const avatarPath = useAvatarPath(ongoingCallPubkey);
const videoRefRemote = useRef<HTMLVideoElement>(null);
function onWindowResize() {
if (positionY + 50 > window.innerHeight || positionX + 50 > window.innerWidth) {
@ -95,24 +98,26 @@ export const DraggableCallContainer = () => {
if (videoRefRemote?.current?.srcObject && remoteStream) {
videoRefRemote.current.srcObject = remoteStream;
videoRefRemote.current.load();
}
const openCallingConversation = useCallback(() => {
useEffect(() => {
if (videoRefRemote?.current) {
videoRefRemote.current.srcObject = remoteStream;
videoRefRemote.current.load();
}
}, [remoteStream, videoRefRemote, videoRefRemote?.current]);
const openCallingConversation = () => {
if (ongoingCallPubkey && ongoingCallPubkey !== selectedConversationKey) {
void openConversationWithMessages({ conversationKey: ongoingCallPubkey });
}
}, [ongoingCallPubkey, selectedConversationKey]);
};
if (!hasOngoingCall || !ongoingCallProps || ongoingCallPubkey === selectedConversationKey) {
return null;
}
const ongoingCallUsername = ongoingCallProps?.profileName || ongoingCallProps?.name;
const avatarPath = ongoingCallPubkey
? getConversationController()
.get(ongoingCallPubkey)
.getAvatarPath()
: undefined;
return (
<Draggable

@ -14,11 +14,15 @@ import { SessionIconButton } from '../icon';
import { animation, contextMenu, Item, Menu } from 'react-contexify';
import { InputItem } from '../../../session/utils/CallManager';
import { DropDownAndToggleButton } from '../icon/DropDownAndToggleButton';
import { StyledVideoElement } from './CallContainer';
import { StyledVideoElement } from './DraggableCallContainer';
import { Avatar, AvatarSize } from '../../Avatar';
import { getConversationController } from '../../../session/conversations';
import { setFullScreenCall } from '../../../state/ducks/conversations';
import { useVideoCallEventsListener } from '../../../hooks/useVideoEventListener';
import {
useAvatarPath,
useOurAvatarPath,
useOurConversationUsername,
} from '../../../hooks/useParamSelector';
const VideoContainer = styled.div`
height: 100%;
@ -233,13 +237,10 @@ const handleMicrophoneToggle = async (
return;
}
console.warn('onclick', isAudioMuted);
if (isAudioMuted) {
// selects the first one
await CallManager.selectAudioInputByDeviceId(currentConnectedAudioInputs[0].deviceId);
} else {
console.warn('onclick was not muted so muting it now');
await CallManager.selectAudioInputByDeviceId(CallManager.INPUT_DISABLED_DEVICE_ID);
}
};
@ -252,23 +253,15 @@ export const InConversationCallContainer = () => {
const ongoingCallPubkey = useSelector(getHasOngoingCallWithPubkey);
const ongoingCallUsername = ongoingCallProps?.profileName || ongoingCallProps?.name;
const videoRefRemote = useRef<any>();
const videoRefLocal = useRef<any>();
const remoteAvatarPath = ongoingCallPubkey
? getConversationController()
.get(ongoingCallPubkey)
.getAvatarPath()
: undefined;
const videoRefRemote = useRef<HTMLVideoElement>(null);
const videoRefLocal = useRef<HTMLVideoElement>(null);
const ourPubkey = UserUtils.getOurPubKeyStrFromCache();
const ourUsername = getConversationController()
.get(ourPubkey)
.getProfileName();
const ourAvatarPath = getConversationController()
.get(ourPubkey)
.getAvatarPath();
const remoteAvatarPath = useAvatarPath(ongoingCallPubkey);
const ourAvatarPath = useOurAvatarPath();
const ourUsername = useOurConversationUsername();
const {
currentConnectedAudioInputs,
@ -278,7 +271,7 @@ export const InConversationCallContainer = () => {
remoteStream,
remoteStreamVideoIsMuted,
isAudioMuted,
} = useVideoCallEventsListener('InConversationCallContainer');
} = useVideoCallEventsListener('InConversationCallContainer', true);
if (videoRefRemote?.current && videoRefLocal?.current) {
videoRefRemote.current.srcObject = remoteStream;

@ -3,6 +3,7 @@ import { useSelector } from 'react-redux';
import styled from 'styled-components';
import _ from 'underscore';
import { useAvatarPath, useConversationUsername } from '../../../hooks/useParamSelector';
import { CallManager } from '../../../session/utils';
import { getHasIncomingCall, getHasIncomingCallFrom } from '../../../state/selectors/conversations';
import { Avatar, AvatarSize } from '../../Avatar';
@ -26,44 +27,38 @@ const IncomingCallAvatatContainer = styled.div`
padding: 0 0 2rem 0;
`;
// TODO:
/**
* Add mute input, deafen, end call, possibly add person to call
* duration - look at how duration calculated for recording.
*/
export const IncomingCallDialog = () => {
const hasIncomingCall = useSelector(getHasIncomingCall);
const incomingCallProps = useSelector(getHasIncomingCallFrom);
const incomingCallFromPubkey = useSelector(getHasIncomingCallFrom);
//#region input handlers
const handleAcceptIncomingCall = async () => {
if (incomingCallProps?.id) {
await CallManager.USER_acceptIncomingCallRequest(incomingCallProps.id);
if (incomingCallFromPubkey) {
await CallManager.USER_acceptIncomingCallRequest(incomingCallFromPubkey);
}
};
const handleDeclineIncomingCall = async () => {
// close the modal
if (incomingCallProps?.id) {
await CallManager.USER_rejectIncomingCallRequest(incomingCallProps.id);
if (incomingCallFromPubkey) {
await CallManager.USER_rejectIncomingCallRequest(incomingCallFromPubkey);
}
};
const from = useConversationUsername(incomingCallFromPubkey);
const incomingAvatar = useAvatarPath(incomingCallFromPubkey);
if (!hasIncomingCall) {
return null;
}
const from = incomingCallProps?.profileName || incomingCallProps?.name || incomingCallProps?.id;
if (hasIncomingCall) {
return (
<SessionWrapperModal title={window.i18n('incomingCallFrom', from)}>
<IncomingCallAvatatContainer>
<Avatar
size={AvatarSize.XL}
avatarPath={incomingCallProps?.avatarPath}
name={incomingCallProps?.profileName}
pubkey={incomingCallProps?.id}
avatarPath={incomingAvatar}
name={from}
pubkey={incomingCallFromPubkey}
/>
</IncomingCallAvatatContainer>
<div className="session-modal__button-group">

@ -0,0 +1,30 @@
import { useSelector } from 'react-redux';
import { UserUtils } from '../session/utils';
import { StateType } from '../state/reducer';
export function useAvatarPath(pubkey: string | undefined) {
return useSelector((state: StateType) => {
if (!pubkey) {
return undefined;
}
return state.conversations.conversationLookup[pubkey]?.avatarPath;
});
}
export function useOurAvatarPath() {
return useAvatarPath(UserUtils.getOurPubKeyStrFromCache());
}
export function useConversationUsername(pubkey: string | undefined) {
return useSelector((state: StateType) => {
if (!pubkey) {
return undefined;
}
const convo = state.conversations.conversationLookup[pubkey];
return convo?.profileName || convo?.name || convo.id;
});
}
export function useOurConversationUsername() {
return useConversationUsername(UserUtils.getOurPubKeyStrFromCache());
}

@ -10,7 +10,7 @@ import {
getSelectedConversationKey,
} from '../state/selectors/conversations';
export function useVideoCallEventsListener(uniqueId: string) {
export function useVideoCallEventsListener(uniqueId: string, onSame: boolean) {
const selectedConversationKey = useSelector(getSelectedConversationKey);
const ongoingCallPubkey = useSelector(getHasOngoingCallWithPubkey);
const isFullScreen = useSelector(getCallIsInFullScreen);
@ -27,7 +27,10 @@ export function useVideoCallEventsListener(uniqueId: string) {
[]
);
useEffect(() => {
if (ongoingCallPubkey === selectedConversationKey) {
if (
(onSame && ongoingCallPubkey === selectedConversationKey) ||
(!onSame && ongoingCallPubkey !== selectedConversationKey)
) {
CallManager.addVideoEventsListener(uniqueId, (options: CallManagerOptionsType) => {
const {
audioInputsList,

@ -459,10 +459,10 @@ function onDataChannelReceivedMessage(ev: MessageEvent<string>) {
if (parsed.video !== undefined) {
remoteVideoStreamIsMuted = !Boolean(parsed.video);
}
callVideoListeners();
} catch (e) {
window.log.warn('onDataChannelReceivedMessage Could not parse data in event', ev);
}
callVideoListeners();
}
function onDataChannelOnOpen() {
window.log.info('onDataChannelOnOpen: sending video status');

@ -49,6 +49,7 @@ export const getConversationLookup = createSelector(
export const getConversationsCount = createSelector(getConversationLookup, (state): number => {
return Object.values(state).length;
});
export const getBlockedPubkeys = createSelector(
// make sure to extends this selector to we are rerun on conversation changes
getConversationLookup,
@ -81,9 +82,22 @@ export const getSelectedConversationIsPublic = createSelector(
}
);
const getConversationId = (_whatever: any, id: string) => id;
export const getConversationById = createSelector(
getConversations,
getConversationId,
(
state: ConversationsStateType,
convoId: string | undefined
): ReduxConversationType | undefined => {
return convoId ? state.conversationLookup[convoId] : undefined;
}
);
export const getHasIncomingCallFrom = createSelector(
getConversations,
(state: ConversationsStateType): ReduxConversationType | undefined => {
(state: ConversationsStateType): string | undefined => {
const foundEntry = Object.entries(state.conversationLookup).find(
([_convoKey, convo]) => convo.callState === 'incoming'
);
@ -91,7 +105,7 @@ export const getHasIncomingCallFrom = createSelector(
if (!foundEntry) {
return undefined;
}
return foundEntry[1];
return foundEntry[1].id;
}
);
@ -114,7 +128,7 @@ export const getHasOngoingCallWith = createSelector(
export const getHasIncomingCall = createSelector(
getHasIncomingCallFrom,
(withConvo: ReduxConversationType | undefined): boolean => !!withConvo
(withConvo: string | undefined): boolean => !!withConvo
);
export const getHasOngoingCall = createSelector(

Loading…
Cancel
Save