signal-desktop/ts/components/CallsTab.tsx

379 lines
13 KiB
TypeScript
Raw Normal View History

2023-08-09 00:53:06 +00:00
// Copyright 2023 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import React, { useCallback, useEffect, useState } from 'react';
2023-08-09 00:53:06 +00:00
import type { LocalizerType } from '../types/I18N';
import { NavSidebar, NavSidebarActionButton } from './NavSidebar';
import { CallsList } from './CallsList';
import type { ConversationType } from '../state/ducks/conversations';
import type {
CallHistoryFilterOptions,
CallHistoryGroup,
CallHistoryPagination,
} from '../types/CallDisposition';
import { CallsNewCall } from './CallsNewCallButton';
2023-08-09 00:53:06 +00:00
import { useEscapeHandling } from '../hooks/useEscapeHandling';
import type {
ActiveCallStateType,
PeekNotConnectedGroupCallType,
} from '../state/ducks/calling';
2023-08-09 00:53:06 +00:00
import { ContextMenu } from './ContextMenu';
import { ConfirmationDialog } from './ConfirmationDialog';
import type { UnreadStats } from '../util/countUnreadStats';
import type { WidthBreakpoint } from './_util';
2024-04-01 19:19:35 +00:00
import type { CallLinkType } from '../types/CallLink';
import type { CallStateType } from '../state/selectors/calling';
import type { StartCallData } from './ConfirmLeaveCallModal';
2024-08-13 23:34:42 +00:00
import { I18n } from './I18n';
2023-08-09 00:53:06 +00:00
enum CallsTabSidebarView {
CallsListView,
NewCallView,
}
type CallsTabProps = Readonly<{
activeCall: ActiveCallStateType | undefined;
allConversations: ReadonlyArray<ConversationType>;
2023-08-21 20:12:27 +00:00
otherTabsUnreadStats: UnreadStats;
2023-08-09 00:53:06 +00:00
getCallHistoryGroupsCount: (
options: CallHistoryFilterOptions
) => Promise<number>;
getCallHistoryGroups: (
options: CallHistoryFilterOptions,
pagination: CallHistoryPagination
) => Promise<Array<CallHistoryGroup>>;
callHistoryEdition: number;
2024-06-10 15:23:43 +00:00
canCreateCallLinks: boolean;
getAdhocCall: (roomId: string) => CallStateType | undefined;
getCall: (id: string) => CallStateType | undefined;
2024-04-01 19:19:35 +00:00
getCallLink: (id: string) => CallLinkType | undefined;
2023-08-09 00:53:06 +00:00
getConversation: (id: string) => ConversationType | void;
hangUpActiveCall: (reason: string) => void;
hasFailedStorySends: boolean;
hasPendingUpdate: boolean;
2023-08-09 00:53:06 +00:00
i18n: LocalizerType;
navTabsCollapsed: boolean;
onClearCallHistory: () => void;
onMarkCallHistoryRead: (conversationId: string, callId: string) => void;
2023-08-09 00:53:06 +00:00
onToggleNavTabsCollapse: (navTabsCollapsed: boolean) => void;
2024-06-10 15:23:43 +00:00
onCreateCallLink: () => void;
2023-08-09 00:53:06 +00:00
onOutgoingAudioCallInConversation: (conversationId: string) => void;
onOutgoingVideoCallInConversation: (conversationId: string) => void;
peekNotConnectedGroupCall: (options: PeekNotConnectedGroupCallType) => void;
2023-08-09 00:53:06 +00:00
preferredLeftPaneWidth: number;
2024-05-22 16:24:27 +00:00
renderCallLinkDetails: (
roomId: string,
callHistoryGroup: CallHistoryGroup,
onClose: () => void
2024-05-22 16:24:27 +00:00
) => JSX.Element;
2023-08-09 00:53:06 +00:00
renderConversationDetails: (
conversationId: string,
callHistoryGroup: CallHistoryGroup | null
) => JSX.Element;
renderToastManager: (_: {
containerWidthBreakpoint: WidthBreakpoint;
}) => JSX.Element;
2023-08-09 00:53:06 +00:00
regionCode: string | undefined;
savePreferredLeftPaneWidth: (preferredLeftPaneWidth: number) => void;
startCallLinkLobbyByRoomId: (options: { roomId: string }) => void;
toggleConfirmLeaveCallModal: (options: StartCallData | null) => void;
togglePip: () => void;
2023-08-09 00:53:06 +00:00
}>;
2024-05-22 16:24:27 +00:00
export type CallsTabSelectedView =
| {
type: 'conversation';
conversationId: string;
callHistoryGroup: CallHistoryGroup | null;
}
| {
type: 'callLink';
roomId: string;
callHistoryGroup: CallHistoryGroup;
};
2023-08-09 00:53:06 +00:00
export function CallsTab({
activeCall,
allConversations,
2023-08-21 20:12:27 +00:00
otherTabsUnreadStats,
2023-08-09 00:53:06 +00:00
getCallHistoryGroupsCount,
getCallHistoryGroups,
callHistoryEdition,
2024-06-10 15:23:43 +00:00
canCreateCallLinks,
getAdhocCall,
getCall,
2024-04-01 19:19:35 +00:00
getCallLink,
2023-08-09 00:53:06 +00:00
getConversation,
hangUpActiveCall,
hasFailedStorySends,
hasPendingUpdate,
2023-08-09 00:53:06 +00:00
i18n,
navTabsCollapsed,
onClearCallHistory,
onMarkCallHistoryRead,
2023-08-09 00:53:06 +00:00
onToggleNavTabsCollapse,
2024-06-10 15:23:43 +00:00
onCreateCallLink,
2023-08-09 00:53:06 +00:00
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
peekNotConnectedGroupCall,
2023-08-09 00:53:06 +00:00
preferredLeftPaneWidth,
2024-05-22 16:24:27 +00:00
renderCallLinkDetails,
2023-08-09 00:53:06 +00:00
renderConversationDetails,
renderToastManager,
2023-08-09 00:53:06 +00:00
regionCode,
savePreferredLeftPaneWidth,
2024-04-01 19:19:35 +00:00
startCallLinkLobbyByRoomId,
toggleConfirmLeaveCallModal,
togglePip,
2023-08-09 00:53:06 +00:00
}: CallsTabProps): JSX.Element {
const [sidebarView, setSidebarView] = useState(
CallsTabSidebarView.CallsListView
);
2024-05-22 16:24:27 +00:00
const [selectedView, setSelectedViewInner] =
useState<CallsTabSelectedView | null>(null);
const [selectedViewKey, setSelectedViewKey] = useState(() => 1);
2023-08-09 00:53:06 +00:00
const [
confirmClearCallHistoryDialogOpen,
setConfirmClearCallHistoryDialogOpen,
] = useState(false);
2024-05-22 16:24:27 +00:00
const updateSelectedView = useCallback(
(nextSelected: CallsTabSelectedView | null) => {
setSelectedViewInner(nextSelected);
setSelectedViewKey(key => key + 1);
2023-08-09 00:53:06 +00:00
},
[]
);
2024-05-22 16:24:27 +00:00
const updateSidebarView = useCallback(
(newSidebarView: CallsTabSidebarView) => {
setSidebarView(newSidebarView);
updateSelectedView(null);
2023-08-09 00:53:06 +00:00
},
2024-05-22 16:24:27 +00:00
[updateSelectedView]
2023-08-09 00:53:06 +00:00
);
const onCloseSelectedView = useCallback(() => {
updateSelectedView(null);
}, [updateSelectedView]);
2023-08-09 00:53:06 +00:00
useEscapeHandling(
sidebarView === CallsTabSidebarView.NewCallView
? () => {
updateSidebarView(CallsTabSidebarView.CallsListView);
}
: undefined
);
const handleOpenClearCallHistoryDialog = useCallback(() => {
setConfirmClearCallHistoryDialogOpen(true);
}, []);
const handleCloseClearCallHistoryDialog = useCallback(() => {
setConfirmClearCallHistoryDialogOpen(false);
}, []);
const handleOutgoingAudioCallInConversation = useCallback(
(conversationId: string) => {
onOutgoingAudioCallInConversation(conversationId);
updateSidebarView(CallsTabSidebarView.CallsListView);
},
[updateSidebarView, onOutgoingAudioCallInConversation]
);
const handleOutgoingVideoCallInConversation = useCallback(
(conversationId: string) => {
onOutgoingVideoCallInConversation(conversationId);
updateSidebarView(CallsTabSidebarView.CallsListView);
},
[updateSidebarView, onOutgoingVideoCallInConversation]
);
useEffect(() => {
2024-05-22 16:24:27 +00:00
if (selectedView?.type === 'conversation') {
selectedView.callHistoryGroup?.children.forEach(child => {
onMarkCallHistoryRead(selectedView.conversationId, child.callId);
});
}
2024-05-22 16:24:27 +00:00
}, [selectedView, onMarkCallHistoryRead]);
2023-08-09 00:53:06 +00:00
return (
<>
<div className="CallsTab">
<NavSidebar
i18n={i18n}
title={
sidebarView === CallsTabSidebarView.CallsListView
? i18n('icu:CallsTab__HeaderTitle--CallsList')
: i18n('icu:CallsTab__HeaderTitle--NewCall')
}
2023-08-21 20:12:27 +00:00
otherTabsUnreadStats={otherTabsUnreadStats}
hasFailedStorySends={hasFailedStorySends}
hasPendingUpdate={hasPendingUpdate}
2023-08-09 00:53:06 +00:00
navTabsCollapsed={navTabsCollapsed}
onBack={
sidebarView === CallsTabSidebarView.NewCallView
? () => {
updateSidebarView(CallsTabSidebarView.CallsListView);
}
: null
}
onToggleNavTabsCollapse={onToggleNavTabsCollapse}
requiresFullWidth
preferredLeftPaneWidth={preferredLeftPaneWidth}
savePreferredLeftPaneWidth={savePreferredLeftPaneWidth}
renderToastManager={renderToastManager}
2023-08-09 00:53:06 +00:00
actions={
<>
{sidebarView === CallsTabSidebarView.CallsListView && (
<>
<NavSidebarActionButton
icon={<span className="CallsTab__NewCallActionIcon" />}
label={i18n('icu:CallsTab__NewCallActionLabel')}
onClick={() => {
updateSidebarView(CallsTabSidebarView.NewCallView);
}}
/>
<ContextMenu
i18n={i18n}
menuOptions={[
{
icon: 'CallsTab__ClearCallHistoryIcon',
label: i18n('icu:CallsTab__ClearCallHistoryLabel'),
onClick: handleOpenClearCallHistoryDialog,
},
]}
popperOptions={{
placement: 'bottom',
strategy: 'absolute',
}}
portalToRoot
>
{({ onClick, onKeyDown, ref }) => {
2023-08-09 00:53:06 +00:00
return (
<NavSidebarActionButton
ref={ref}
onClick={onClick}
2023-08-09 00:53:06 +00:00
onKeyDown={onKeyDown}
icon={<span className="CallsTab__MoreActionsIcon" />}
label={i18n('icu:CallsTab__MoreActionsLabel')}
/>
);
}}
</ContextMenu>
</>
)}
</>
}
>
{sidebarView === CallsTabSidebarView.CallsListView && (
<CallsList
key={CallsTabSidebarView.CallsListView}
activeCall={activeCall}
2024-06-10 15:23:43 +00:00
canCreateCallLinks={canCreateCallLinks}
2023-08-09 00:53:06 +00:00
getCallHistoryGroupsCount={getCallHistoryGroupsCount}
getCallHistoryGroups={getCallHistoryGroups}
callHistoryEdition={callHistoryEdition}
getAdhocCall={getAdhocCall}
getCall={getCall}
2024-04-01 19:19:35 +00:00
getCallLink={getCallLink}
2023-08-09 00:53:06 +00:00
getConversation={getConversation}
hangUpActiveCall={hangUpActiveCall}
2023-08-09 00:53:06 +00:00
i18n={i18n}
2024-05-22 16:24:27 +00:00
selectedCallHistoryGroup={selectedView?.callHistoryGroup ?? null}
onChangeCallsTabSelectedView={updateSelectedView}
2024-06-10 15:23:43 +00:00
onCreateCallLink={onCreateCallLink}
onOutgoingAudioCallInConversation={
handleOutgoingAudioCallInConversation
}
onOutgoingVideoCallInConversation={
handleOutgoingVideoCallInConversation
}
peekNotConnectedGroupCall={peekNotConnectedGroupCall}
2024-04-01 19:19:35 +00:00
startCallLinkLobbyByRoomId={startCallLinkLobbyByRoomId}
toggleConfirmLeaveCallModal={toggleConfirmLeaveCallModal}
togglePip={togglePip}
2023-08-09 00:53:06 +00:00
/>
)}
{sidebarView === CallsTabSidebarView.NewCallView && (
<CallsNewCall
key={CallsTabSidebarView.NewCallView}
hasActiveCall={activeCall != null}
2023-08-09 00:53:06 +00:00
allConversations={allConversations}
i18n={i18n}
regionCode={regionCode}
2024-05-22 16:24:27 +00:00
onChangeCallsTabSelectedView={updateSelectedView}
2023-08-09 00:53:06 +00:00
onOutgoingAudioCallInConversation={
handleOutgoingAudioCallInConversation
}
onOutgoingVideoCallInConversation={
handleOutgoingVideoCallInConversation
}
/>
)}
</NavSidebar>
2024-05-22 16:24:27 +00:00
{selectedView == null ? (
2023-08-09 00:53:06 +00:00
<div className="CallsTab__EmptyState">
<div className="CallsTab__EmptyStateIcon" />
<p className="CallsTab__EmptyStateLabel">
2024-08-13 23:34:42 +00:00
<I18n
i18n={i18n}
id="icu:CallsTab__EmptyStateText--with-icon-2"
2024-08-13 23:34:42 +00:00
components={{
// eslint-disable-next-line react/no-unstable-nested-components
newCallButtonIcon: () => {
2024-08-13 23:34:42 +00:00
return (
<span
className="CallsTab__EmptyState__ActionIcon"
aria-label={i18n('icu:CallsTab__NewCallActionLabel')}
2024-08-13 23:34:42 +00:00
/>
);
},
}}
/>
</p>
2023-08-09 00:53:06 +00:00
</div>
) : (
<div
className="CallsTab__ConversationCallDetails"
2024-05-22 16:24:27 +00:00
// Force scrolling to top when selection changes
key={selectedViewKey}
2023-08-09 00:53:06 +00:00
>
2024-05-22 16:24:27 +00:00
{selectedView.type === 'conversation' &&
renderConversationDetails(
selectedView.conversationId,
selectedView.callHistoryGroup
)}
{selectedView.type === 'callLink' &&
renderCallLinkDetails(
selectedView.roomId,
selectedView.callHistoryGroup,
onCloseSelectedView
2024-05-22 16:24:27 +00:00
)}
2023-08-09 00:53:06 +00:00
</div>
)}
</div>
{confirmClearCallHistoryDialogOpen && (
<ConfirmationDialog
dialogName="CallsTab__ConfirmClearCallHistory"
i18n={i18n}
onClose={handleCloseClearCallHistoryDialog}
title={i18n('icu:CallsTab__ConfirmClearCallHistory__Title')}
actions={[
{
style: 'negative',
text: i18n(
'icu:CallsTab__ConfirmClearCallHistory__ConfirmButton'
),
action: onClearCallHistory,
},
]}
>
{i18n('icu:CallsTab__ConfirmClearCallHistory__Body')}
</ConfirmationDialog>
)}
</>
);
}