blob: ec68765168d136ecbef89b2d76db8f0488105530 [file] [log] [blame]
/*
* Copyright (C) 2022 Savoir-faire Linux Inc.
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as
* published by the Free Software Foundation; either version 3 of the
* License, or (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public
* License along with this program. If not, see
* <https://www.gnu.org/licenses/>.
*/
import { CallBegin, ConversationInfos, WebSocketMessageType } from 'jami-web-common';
import { createContext, useCallback, useContext, useEffect, useMemo, useState } from 'react';
import { useTranslation } from 'react-i18next';
import { useNavigate } from 'react-router-dom';
import { RemoteVideoOverlay } from '../components/VideoOverlay';
import { useUrlParams } from '../hooks/useUrlParams';
import { ConversationMember } from '../models/conversation-member';
import { ConversationRouteParams } from '../router';
import { useConversationInfosQuery, useMembersQuery } from '../services/conversationQueries';
import { SetState, WithChildren } from '../utils/utils';
import { AlertSnackbarContext } from './AlertSnackbarProvider';
import CallProvider, { CallRole } from './CallProvider';
import { useWebSocketContext } from './WebSocketProvider';
export type CallData = {
conversationId: string;
role: CallRole;
withVideoOn?: boolean;
};
type ICallManagerContext = {
callData: CallData | undefined;
callConversationInfos: ConversationInfos | undefined;
callMembers: ConversationMember[] | undefined;
startCall: SetState<CallData | undefined>;
exitCall: () => void;
};
const defaultCallManagerContext: ICallManagerContext = {
callData: undefined,
callConversationInfos: undefined,
callMembers: undefined,
startCall: () => {},
exitCall: () => {},
};
export const CallManagerContext = createContext<ICallManagerContext>(defaultCallManagerContext);
CallManagerContext.displayName = 'CallManagerContext';
export default ({ children }: WithChildren) => {
const [callData, setCallData] = useState<CallData>();
const { setAlertContent } = useContext(AlertSnackbarContext);
const webSocket = useWebSocketContext();
const navigate = useNavigate();
const { data: conversationInfos } = useConversationInfosQuery(callData?.conversationId);
const { data: members } = useMembersQuery(callData?.conversationId);
const { urlParams } = useUrlParams<ConversationRouteParams>();
const { t } = useTranslation();
const failStartCall = useCallback(() => {
throw new Error('Cannot start call: Already in a call');
}, []);
const startCall = !callData ? setCallData : failStartCall;
const exitCall = useCallback(() => {
if (!callData) {
return;
}
setCallData(undefined);
// TODO: write in chat that the call ended
}, [callData]);
useEffect(() => {
const callBeginListener = ({ conversationId, withVideoOn }: CallBegin) => {
if (callData) {
// TODO: Currently, we display a notification if already in a call.
// In the future, we should handle receiving a call while already in another.
setAlertContent({
messageI18nKey: 'missed_incoming_call',
messageI18nContext: { conversationId },
severity: 'info',
alertOpen: true,
});
return;
}
startCall({ conversationId: conversationId, role: 'receiver', withVideoOn });
navigate(`/conversation/${conversationId}`);
};
webSocket.bind(WebSocketMessageType.CallBegin, callBeginListener);
return () => {
webSocket.unbind(WebSocketMessageType.CallBegin, callBeginListener);
};
}, [webSocket, navigate, startCall, callData, setAlertContent, t]);
const value = useMemo(
() => ({
startCall,
callData,
callConversationInfos: conversationInfos,
callMembers: members,
exitCall,
}),
[startCall, callData, conversationInfos, members, exitCall]
);
return (
<>
<CallManagerContext.Provider value={value}>
<CallProvider>
{callData && callData.conversationId !== urlParams.conversationId && (
<RemoteVideoOverlay callConversationId={callData.conversationId} />
)}
{children}
</CallProvider>
</CallManagerContext.Provider>
</>
);
};