signal-desktop/ts/components/CallsList.tsx

1091 lines
32 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 type { ChangeEvent } from 'react';
import React, {
useCallback,
useEffect,
useMemo,
useRef,
useState,
} from 'react';
import type { Index, IndexRange, ListRowProps } from 'react-virtualized';
import { InfiniteLoader, List } from 'react-virtualized';
import classNames from 'classnames';
import type { LocalizerType } from '../types/I18N';
import { ListTile } from './ListTile';
import { Avatar, AvatarSize } from './Avatar';
import { SearchInput } from './SearchInput';
import type {
CallHistoryFilterOptions,
CallHistoryGroup,
CallHistoryPagination,
} from '../types/CallDisposition';
import {
CallHistoryFilterStatus,
CallDirection,
CallType,
DirectCallStatus,
GroupCallStatus,
isSameCallHistoryGroup,
CallMode,
2023-08-09 00:53:06 +00:00
} from '../types/CallDisposition';
import { formatDateTimeShort, isMoreRecentThan } from '../util/timestamp';
2023-08-09 00:53:06 +00:00
import type { ConversationType } from '../state/ducks/conversations';
import * as log from '../logging/log';
import { refMerger } from '../util/refMerger';
import { drop } from '../util/drop';
import { strictAssert } from '../util/assert';
import { UserText } from './UserText';
2024-05-15 21:48:02 +00:00
import { I18n } from './I18n';
2024-08-13 23:34:42 +00:00
import { NavSidebarSearchHeader, NavSidebarEmpty } from './NavSidebar';
2023-08-09 00:53:06 +00:00
import { SizeObserver } from '../hooks/useSizeObserver';
import {
formatCallHistoryGroup,
getCallIdFromEra,
} from '../util/callDisposition';
import { CallsNewCallButton } from './CallsNewCallButton';
import { Tooltip, TooltipPlacement } from './Tooltip';
import { Theme } from '../util/theme';
2024-04-01 19:19:35 +00:00
import type { CallingConversationType } from '../types/Calling';
import type { CallLinkType } from '../types/CallLink';
2024-04-25 17:09:05 +00:00
import {
callLinkToConversation,
getPlaceholderCallLinkConversation,
} from '../util/callLinks';
2024-05-22 16:24:27 +00:00
import type { CallsTabSelectedView } from './CallsTab';
import type { CallStateType } from '../state/selectors/calling';
import {
isGroupOrAdhocCallMode,
isGroupOrAdhocCallState,
} from '../util/isGroupOrAdhocCall';
import { isAnybodyInGroupCall } from '../state/ducks/callingHelpers';
import type {
ActiveCallStateType,
PeekNotConnectedGroupCallType,
} from '../state/ducks/calling';
import { DAY, MINUTE, SECOND } from '../util/durations';
import type { StartCallData } from './ConfirmLeaveCallModal';
2024-11-13 20:24:00 +00:00
import { Button, ButtonVariant } from './Button';
import type { ICUJSXMessageParamsByKeyType } from '../types/Util';
2023-08-09 00:53:06 +00:00
function Timestamp({
i18n,
timestamp,
}: {
i18n: LocalizerType;
timestamp: number;
}): JSX.Element {
const [now, setNow] = useState(() => Date.now());
useEffect(() => {
const timer = setInterval(() => {
setNow(Date.now());
}, 1_000);
return () => {
clearInterval(timer);
};
}, []);
const dateTime = useMemo(() => {
return new Date(timestamp).toISOString();
}, [timestamp]);
const formatted = useMemo(() => {
void now; // Use this as a dep so we update
return formatDateTimeShort(i18n, timestamp);
}, [i18n, timestamp, now]);
return <time dateTime={dateTime}>{formatted}</time>;
}
type SearchResults = Readonly<{
count: number;
items: ReadonlyArray<CallHistoryGroup>;
}>;
type SearchState = Readonly<{
state: 'init' | 'pending' | 'rejected' | 'fulfilled';
// Note these fields shouldnt be updated until the search is fulfilled or rejected.
options: null | { query: string; status: CallHistoryFilterStatus };
results: null | SearchResults;
}>;
const defaultInitState: SearchState = {
state: 'init',
options: null,
results: null,
};
const defaultPendingState: SearchState = {
state: 'pending',
options: null,
results: {
count: 100,
items: [],
},
};
type CallsListProps = Readonly<{
activeCall: ActiveCallStateType | undefined;
2024-06-10 15:23:43 +00:00
canCreateCallLinks: boolean;
2023-08-09 00:53:06 +00:00
getCallHistoryGroupsCount: (
options: CallHistoryFilterOptions
) => Promise<number>;
getCallHistoryGroups: (
options: CallHistoryFilterOptions,
pagination: CallHistoryPagination
) => Promise<Array<CallHistoryGroup>>;
callHistoryEdition: number;
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;
2023-08-09 00:53:06 +00:00
i18n: LocalizerType;
selectedCallHistoryGroup: CallHistoryGroup | null;
2024-06-10 15:23:43 +00:00
onCreateCallLink: () => void;
onOutgoingAudioCallInConversation: (conversationId: string) => void;
onOutgoingVideoCallInConversation: (conversationId: string) => void;
2024-05-22 16:24:27 +00:00
onChangeCallsTabSelectedView: (selectedView: CallsTabSelectedView) => void;
peekNotConnectedGroupCall: (options: PeekNotConnectedGroupCallType) => void;
startCallLinkLobbyByRoomId: (options: { roomId: string }) => void;
toggleConfirmLeaveCallModal: (options: StartCallData | null) => void;
togglePip: () => void;
2023-08-09 00:53:06 +00:00
}>;
2024-11-13 20:24:00 +00:00
const FILTER_HEADER_ROW_HEIGHT = 50;
const CALL_LIST_ITEM_ROW_HEIGHT = 62;
const INACTIVE_CALL_LINKS_TO_PEEK = 10;
const INACTIVE_CALL_LINK_AGE_THRESHOLD = 10 * DAY;
const INACTIVE_CALL_LINK_PEEK_INTERVAL = 5 * MINUTE;
const PEEK_BATCH_COUNT = 10;
const PEEK_QUEUE_INTERVAL = 30 * SECOND;
2023-08-17 00:48:20 +00:00
function isSameOptions(
a: CallHistoryFilterOptions,
b: CallHistoryFilterOptions
) {
return a.query === b.query && a.status === b.status;
}
2024-11-13 20:24:00 +00:00
type SpecialRows =
| 'CreateCallLink'
| 'EmptyState'
| 'FilterHeader'
| 'ClearFilterButton';
2024-06-10 15:23:43 +00:00
type Row = CallHistoryGroup | SpecialRows;
2023-08-09 00:53:06 +00:00
export function CallsList({
activeCall,
2024-06-10 15:23:43 +00:00
canCreateCallLinks,
2023-08-09 00:53:06 +00:00
getCallHistoryGroupsCount,
getCallHistoryGroups,
callHistoryEdition,
getAdhocCall,
getCall,
2024-04-01 19:19:35 +00:00
getCallLink,
2023-08-09 00:53:06 +00:00
getConversation,
i18n,
selectedCallHistoryGroup,
2024-06-10 15:23:43 +00:00
onCreateCallLink,
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
2024-05-22 16:24:27 +00:00
onChangeCallsTabSelectedView,
peekNotConnectedGroupCall,
2024-04-01 19:19:35 +00:00
startCallLinkLobbyByRoomId,
toggleConfirmLeaveCallModal,
togglePip,
2023-08-09 00:53:06 +00:00
}: CallsListProps): JSX.Element {
const infiniteLoaderRef = useRef<InfiniteLoader>(null);
const listRef = useRef<List>(null);
const [queryInput, setQueryInput] = useState('');
2024-06-10 15:23:43 +00:00
const [statusInput, setStatusInput] = useState(CallHistoryFilterStatus.All);
2023-08-09 00:53:06 +00:00
const [searchState, setSearchState] = useState(defaultInitState);
2023-08-17 00:48:20 +00:00
const prevOptionsRef = useRef<CallHistoryFilterOptions | null>(null);
const getCallHistoryGroupsCountRef = useRef(getCallHistoryGroupsCount);
const getCallHistoryGroupsRef = useRef(getCallHistoryGroups);
2024-06-10 15:23:43 +00:00
const searchStateQuery = searchState.options?.query ?? '';
const searchStateStatus =
searchState.options?.status ?? CallHistoryFilterStatus.All;
2024-08-13 23:34:42 +00:00
const hasSearchStateQuery = searchStateQuery !== '';
2024-11-13 20:24:00 +00:00
const hasMissedCallFilter =
searchStateStatus === CallHistoryFilterStatus.Missed;
const searchFiltering = hasSearchStateQuery || hasMissedCallFilter;
2024-06-10 15:23:43 +00:00
const searchPending = searchState.state === 'pending';
2024-08-13 23:34:42 +00:00
const isEmpty = !searchState.results?.items?.length;
2024-06-10 15:23:43 +00:00
2024-11-13 20:24:00 +00:00
const rows = useMemo<ReadonlyArray<Row>>(() => {
const results: ReadonlyArray<Row> = searchState.results?.items ?? [];
if (results.length === 0 && searchFiltering) {
return hasMissedCallFilter
? ['FilterHeader', 'EmptyState', 'ClearFilterButton']
: ['EmptyState'];
2024-06-10 15:23:43 +00:00
}
2024-11-13 20:24:00 +00:00
2024-06-10 15:23:43 +00:00
if (!searchFiltering && canCreateCallLinks) {
2024-11-13 20:24:00 +00:00
return ['CreateCallLink', ...results];
}
if (hasMissedCallFilter) {
return ['FilterHeader', ...results, 'ClearFilterButton'];
2024-06-10 15:23:43 +00:00
}
return results;
2024-08-13 23:34:42 +00:00
}, [
searchState.results?.items,
searchFiltering,
canCreateCallLinks,
2024-11-13 20:24:00 +00:00
hasMissedCallFilter,
2024-08-13 23:34:42 +00:00
]);
2024-06-10 15:23:43 +00:00
const rowCount = rows.length;
const searchStateItemsRef = useRef<ReadonlyArray<CallHistoryGroup> | null>(
null
);
const peekQueueRef = useRef<Set<string>>(new Set());
const peekQueueArgsRef = useRef<Map<string, PeekNotConnectedGroupCallType>>(
new Map()
);
const inactiveCallLinksPeekedAtRef = useRef<Map<string, number>>(new Map());
const peekQueueTimerRef = useRef<NodeJS.Timeout | null>(null);
2024-06-10 15:23:43 +00:00
useEffect(() => {
return () => {
2024-06-10 15:23:43 +00:00
if (peekQueueTimerRef.current != null) {
clearInterval(peekQueueTimerRef.current);
peekQueueTimerRef.current = null;
}
};
}, []);
useEffect(() => {
getCallHistoryGroupsCountRef.current = getCallHistoryGroupsCount;
getCallHistoryGroupsRef.current = getCallHistoryGroups;
}, [getCallHistoryGroupsCount, getCallHistoryGroups]);
const getConversationForItem = useCallback(
(item: CallHistoryGroup | null): CallingConversationType | null => {
if (!item) {
return null;
}
const isAdhoc = item?.type === CallType.Adhoc;
if (isAdhoc) {
const callLink = isAdhoc ? getCallLink(item.peerId) : null;
if (callLink) {
return callLinkToConversation(callLink, i18n);
}
return getPlaceholderCallLinkConversation(item.peerId, i18n);
}
return getConversation(item.peerId) ?? null;
},
[getCallLink, getConversation, i18n]
);
const getCallByPeerId = useCallback(
({
mode,
peerId,
}: {
mode: CallMode | undefined;
peerId: string | undefined;
}): CallStateType | undefined => {
if (!peerId || !mode) {
return;
}
if (mode === CallMode.Adhoc) {
return getAdhocCall(peerId);
}
const conversation = getConversation(peerId);
if (!conversation) {
return;
}
return getCall(conversation.id);
},
[getAdhocCall, getCall, getConversation]
);
const getIsCallActive = useCallback(
({
callHistoryGroup,
}: {
callHistoryGroup: CallHistoryGroup | null;
}): boolean => {
if (!callHistoryGroup) {
return false;
}
const { mode, peerId } = callHistoryGroup;
const call = getCallByPeerId({ mode, peerId });
if (!call || !isGroupOrAdhocCallState(call)) {
// We can't tell from CallHistory alone whether a 1:1 call is active
return false;
}
// eraId indicates a group/call link call is active.
const eraId = call.peekInfo?.eraId;
if (!eraId) {
return false;
}
// Group calls have multiple entries sharing a peerId. To distinguish them we need
// to compare the active callId (derived from eraId) with this item's callId set.
if (mode === CallMode.Group) {
const callId = getCallIdFromEra(eraId);
return callHistoryGroup.children.some(
groupItem => groupItem.callId === callId
);
}
// Call links only show once in the calls list, so we can just return active.
return true;
},
[getCallByPeerId]
);
const getIsAnybodyInCall = useCallback(
({
callHistoryGroup,
}: {
callHistoryGroup: CallHistoryGroup | null;
}): boolean => {
if (!callHistoryGroup) {
return false;
}
const { mode, peerId } = callHistoryGroup;
const call = getCallByPeerId({ mode, peerId });
if (!call || !isGroupOrAdhocCallState(call)) {
return false;
}
return isAnybodyInGroupCall(call.peekInfo);
},
[getCallByPeerId]
);
const getIsInCall = useCallback(
({
activeCallConversationId,
callHistoryGroup,
conversation,
isActive,
}: {
activeCallConversationId: string | undefined;
callHistoryGroup: CallHistoryGroup | null;
conversation: CallingConversationType | null;
isActive: boolean;
}): boolean => {
if (!callHistoryGroup) {
return false;
}
const { mode, peerId } = callHistoryGroup;
if (mode === CallMode.Adhoc) {
return peerId === activeCallConversationId;
}
// For direct conversations, we know the call is active if it's the active call!
if (mode === CallMode.Direct) {
return Boolean(
conversation && conversation?.id === activeCallConversationId
);
}
// For group and adhoc calls
return Boolean(
isActive &&
conversation &&
conversation?.id === activeCallConversationId
);
},
[]
);
// If the call is already enqueued then this is a no op.
const maybeEnqueueCallPeek = useCallback((item: CallHistoryGroup): void => {
const { mode: callMode, peerId } = item;
const queue = peekQueueRef.current;
if (queue.has(peerId)) {
return;
}
if (isGroupOrAdhocCallMode(callMode)) {
peekQueueArgsRef.current.set(peerId, {
callMode,
conversationId: peerId,
});
queue.add(peerId);
} else {
log.error(`Trying to peek unsupported call mode ${callMode}`);
}
}, []);
// Get the oldest inserted peerIds by iterating the Set in insertion order.
const getPeerIdsToPeek = useCallback((): ReadonlyArray<string> => {
const peerIds: Array<string> = [];
for (const peerId of peekQueueRef.current) {
peerIds.push(peerId);
if (peerIds.length === PEEK_BATCH_COUNT) {
return peerIds;
}
}
return peerIds;
}, []);
const doCallPeeks = useCallback((): void => {
const peerIds = getPeerIdsToPeek();
for (const peerId of peerIds) {
const peekArgs = peekQueueArgsRef.current.get(peerId);
if (peekArgs) {
inactiveCallLinksPeekedAtRef.current.set(peerId, new Date().getTime());
peekNotConnectedGroupCall(peekArgs);
}
peekQueueRef.current.delete(peerId);
peekQueueArgsRef.current.delete(peerId);
}
}, [getPeerIdsToPeek, peekNotConnectedGroupCall]);
const enqueueCallPeeks = useCallback(
(callItems: ReadonlyArray<CallHistoryGroup>, isFirstRun: boolean): void => {
let peekCount = 0;
let inactiveCallLinksToPeek = 0;
for (const item of callItems) {
const { mode } = item;
if (isGroupOrAdhocCallMode(mode)) {
const isActive = getIsCallActive({
callHistoryGroup: item,
});
if (isActive) {
// Don't peek if you're already in the call.
const activeCallConversationId = activeCall?.conversationId;
if (activeCallConversationId) {
const conversation = getConversationForItem(item);
const isInCall = getIsInCall({
activeCallConversationId,
callHistoryGroup: item,
conversation,
isActive,
});
if (isInCall) {
continue;
}
}
maybeEnqueueCallPeek(item);
peekCount += 1;
continue;
}
if (
mode === CallMode.Adhoc &&
isFirstRun &&
inactiveCallLinksToPeek < INACTIVE_CALL_LINKS_TO_PEEK &&
isMoreRecentThan(item.timestamp, INACTIVE_CALL_LINK_AGE_THRESHOLD)
) {
const peekedAt = inactiveCallLinksPeekedAtRef.current.get(
item.peerId
);
if (
peekedAt &&
isMoreRecentThan(peekedAt, INACTIVE_CALL_LINK_PEEK_INTERVAL)
) {
continue;
}
maybeEnqueueCallPeek(item);
inactiveCallLinksToPeek += 1;
peekCount += 1;
}
}
}
if (peekCount === 0) {
return;
}
log.info(`Found ${peekCount} calls to peek.`);
if (peekQueueTimerRef.current != null) {
return;
}
log.info('Starting background call peek.');
peekQueueTimerRef.current = setInterval(() => {
if (searchStateItemsRef.current) {
enqueueCallPeeks(searchStateItemsRef.current, false);
}
if (peekQueueRef.current.size > 0) {
doCallPeeks();
}
}, PEEK_QUEUE_INTERVAL);
doCallPeeks();
},
[
activeCall?.conversationId,
doCallPeeks,
getConversationForItem,
getIsCallActive,
getIsInCall,
maybeEnqueueCallPeek,
]
);
2023-08-09 00:53:06 +00:00
useEffect(() => {
const controller = new AbortController();
async function search() {
2023-08-17 00:48:20 +00:00
const options: CallHistoryFilterOptions = {
2023-08-09 00:53:06 +00:00
query: queryInput.toLowerCase().normalize().trim(),
2024-06-10 15:23:43 +00:00
status: statusInput,
2023-08-09 00:53:06 +00:00
};
let timer = setTimeout(() => {
setSearchState(prevSearchState => {
if (prevSearchState.state === 'init') {
return defaultPendingState;
}
return prevSearchState;
});
timer = setTimeout(() => {
// Show loading indicator after a delay
setSearchState(defaultPendingState);
}, 300);
}, 50);
let results: SearchResults | null = null;
try {
const [count, items] = await Promise.all([
getCallHistoryGroupsCountRef.current(options),
getCallHistoryGroupsRef.current(options, {
2023-08-09 00:53:06 +00:00
offset: 0,
limit: 100, // preloaded rows
}),
]);
results = { count, items };
} catch (error) {
log.error('CallsList#fetchTotal error fetching', error);
}
// Clear the loading indicator timeout
clearTimeout(timer);
// Ignore old requests
if (controller.signal.aborted) {
return;
}
if (results) {
enqueueCallPeeks(results.items, true);
searchStateItemsRef.current = results.items;
}
2023-08-09 00:53:06 +00:00
// Only commit the new search state once the results are ready
setSearchState({
state: results == null ? 'rejected' : 'fulfilled',
options,
results,
});
2023-08-17 00:48:20 +00:00
const isUpdatingSameSearch =
prevOptionsRef.current != null &&
isSameOptions(options, prevOptionsRef.current);
// Commit only at the end in case the search was aborted.
prevOptionsRef.current = options;
// Only reset the scroll position to the top when the user has changed the
// search parameters
if (!isUpdatingSameSearch) {
infiniteLoaderRef.current?.resetLoadMoreRowsCache(true);
listRef.current?.scrollToPosition(0);
}
2023-08-09 00:53:06 +00:00
}
drop(search());
return () => {
controller.abort();
};
2024-06-10 15:23:43 +00:00
}, [queryInput, statusInput, callHistoryEdition, enqueueCallPeeks]);
2023-08-09 00:53:06 +00:00
const loadMoreRows = useCallback(
async (props: IndexRange) => {
const { state, options } = searchState;
if (state !== 'fulfilled') {
return;
}
strictAssert(
options != null,
'options should never be null when status is fulfilled'
);
let { startIndex, stopIndex } = props;
if (startIndex > stopIndex) {
// flip
[startIndex, stopIndex] = [stopIndex, startIndex];
}
const offset = startIndex;
const limit = stopIndex - startIndex + 1;
try {
const groups = await getCallHistoryGroupsRef.current(options, {
offset,
limit,
});
2023-08-09 00:53:06 +00:00
if (searchState.options !== options) {
return;
}
enqueueCallPeeks(groups, false);
2023-08-09 00:53:06 +00:00
setSearchState(prevSearchState => {
strictAssert(
prevSearchState.results != null,
'results should never be null here'
);
const newItems = prevSearchState.results.items.slice();
newItems.splice(startIndex, stopIndex, ...groups);
searchStateItemsRef.current = newItems;
2023-08-09 00:53:06 +00:00
return {
...prevSearchState,
results: {
...prevSearchState.results,
items: newItems,
},
};
});
} catch (error) {
log.error('CallsList#loadMoreRows error fetching', error);
}
},
[enqueueCallPeeks, searchState]
2024-04-25 17:09:05 +00:00
);
2023-08-09 00:53:06 +00:00
const isRowLoaded = useCallback(
(props: Index) => {
return searchState.results?.items[props.index] != null;
},
[searchState]
);
2024-06-10 15:23:43 +00:00
const rowHeight = useCallback(
({ index }: Index) => {
const item = rows.at(index) ?? null;
2024-11-13 20:24:00 +00:00
if (item === 'FilterHeader') {
return FILTER_HEADER_ROW_HEIGHT;
2024-06-10 15:23:43 +00:00
}
return CALL_LIST_ITEM_ROW_HEIGHT;
},
[rows]
);
2023-08-09 00:53:06 +00:00
const rowRenderer = useCallback(
({ key, index, style }: ListRowProps) => {
2024-06-10 15:23:43 +00:00
const item = rows.at(index) ?? null;
if (item === 'CreateCallLink') {
return (
<div key={key} style={style}>
<ListTile
moduleClassName="CallsList__ItemTile"
title={
<span className="CallsList__ItemTitle">
{i18n('icu:CallsList__CreateCallLink')}
</span>
}
leading={
<i className="ComposeStepButton__icon ComposeStepButton__icon--call-link" />
2024-06-10 15:23:43 +00:00
}
onClick={onCreateCallLink}
/>
</div>
);
}
if (item === 'EmptyState') {
2024-11-13 20:24:00 +00:00
let i18nId: keyof ICUJSXMessageParamsByKeyType;
if (hasSearchStateQuery && hasMissedCallFilter) {
i18nId = 'icu:CallsList__EmptyState--hasQueryAndMissedCalls';
} else if (hasSearchStateQuery) {
i18nId = 'icu:CallsList__EmptyState--hasQuery';
} else if (hasMissedCallFilter) {
i18nId = 'icu:CallsList__EmptyState--missedCalls';
} else {
// This should never happen
i18nId = 'icu:CallsList__EmptyState--hasQuery';
}
2024-06-10 15:23:43 +00:00
return (
<div key={key} className="CallsList__EmptyState" style={style}>
2024-08-13 23:34:42 +00:00
<I18n
i18n={i18n}
2024-11-13 20:24:00 +00:00
id={i18nId}
2024-08-13 23:34:42 +00:00
components={{
query: <UserText text={searchStateQuery} />,
}}
/>
2024-06-10 15:23:43 +00:00
</div>
);
}
2024-11-13 20:24:00 +00:00
if (item === 'FilterHeader') {
return (
<div key={key} style={style} className="CallsList__FilterHeader">
{i18n('icu:CallsList__FilteredByMissedHeader')}
</div>
);
}
if (item === 'ClearFilterButton') {
return (
<div key={key} style={style} className="ClearFilterButton">
<Button
variant={ButtonVariant.SecondaryAffirmative}
className={classNames('ClearFilterButton__inner', {
// The clear filter button should be closer to the emty state
// text than to the search results.
'ClearFilterButton__inner-vertical-center': !isEmpty,
})}
onClick={() => setStatusInput(CallHistoryFilterStatus.All)}
>
{i18n('icu:clearFilterButton')}
</Button>
</div>
);
}
2024-04-25 17:09:05 +00:00
const conversation = getConversationForItem(item);
const activeCallConversationId = activeCall?.conversationId;
const isActive = getIsCallActive({
callHistoryGroup: item,
});
// After everyone leaves a call, it remains active on the server for a little bit.
// We don't need to show the active call join button in this case.
const isAnybodyInCall =
isActive &&
getIsAnybodyInCall({
callHistoryGroup: item,
});
const isInCall = getIsInCall({
activeCallConversationId,
callHistoryGroup: item,
conversation,
isActive,
});
2024-04-25 17:09:05 +00:00
const isAdhoc = item?.type === CallType.Adhoc;
const isCallButtonVisible = Boolean(
2024-04-25 17:09:05 +00:00
!isAdhoc || (isAdhoc && getCallLink(item.peerId))
);
const isActiveVisible = Boolean(
isCallButtonVisible && item && isAnybodyInCall
);
2023-08-09 00:53:06 +00:00
2024-06-10 15:23:43 +00:00
if (searchPending || item == null || conversation == null) {
2023-08-09 00:53:06 +00:00
return (
<div key={key} style={style}>
<ListTile
moduleClassName="CallsList__ItemTile"
2023-08-09 00:53:06 +00:00
leading={<div className="CallsList__LoadingAvatar" />}
title={
<span className="CallsList__LoadingText CallsList__LoadingText--title" />
}
subtitleMaxLines={1}
2023-08-09 00:53:06 +00:00
subtitle={
<span className="CallsList__LoadingText CallsList__LoadingText--subtitle" />
}
/>
</div>
);
}
const isSelected =
selectedCallHistoryGroup != null &&
isSameCallHistoryGroup(item, selectedCallHistoryGroup);
const wasMissed =
item.direction === CallDirection.Incoming &&
(item.status === DirectCallStatus.Missed ||
item.status === GroupCallStatus.Missed);
2024-07-30 23:21:33 +00:00
const wasDeclined =
item.direction === CallDirection.Incoming &&
(item.status === DirectCallStatus.Declined ||
item.status === GroupCallStatus.Declined);
2023-08-09 00:53:06 +00:00
let statusText;
if (wasMissed) {
statusText = i18n('icu:CallsList__ItemCallInfo--Missed');
2024-07-30 23:21:33 +00:00
} else if (wasDeclined) {
statusText = i18n('icu:CallsList__ItemCallInfo--Declined');
2024-04-25 17:09:05 +00:00
} else if (isAdhoc) {
2024-04-01 19:19:35 +00:00
statusText = i18n('icu:CallsList__ItemCallInfo--CallLink');
2023-08-09 00:53:06 +00:00
} else if (item.direction === CallDirection.Outgoing) {
statusText = i18n('icu:CallsList__ItemCallInfo--Outgoing');
} else if (item.direction === CallDirection.Incoming) {
statusText = i18n('icu:CallsList__ItemCallInfo--Incoming');
} else {
strictAssert(false, 'Cannot format call');
}
const inCallAndNotThisOne = !isInCall && activeCall;
const callButton = (
<CallsNewCallButton
callType={item.type}
isActive={isActiveVisible}
isInCall={isInCall}
isEnabled={!inCallAndNotThisOne}
onClick={() => {
if (isInCall) {
togglePip();
} else if (activeCall) {
if (isAdhoc) {
toggleConfirmLeaveCallModal({
type: 'adhoc-roomId',
roomId: item.peerId,
});
} else {
toggleConfirmLeaveCallModal({
type: 'conversation',
conversationId: conversation.id,
isVideoCall: item.type !== CallType.Audio,
});
}
} else if (isAdhoc) {
startCallLinkLobbyByRoomId({ roomId: item.peerId });
} else if (conversation) {
if (item.type === CallType.Audio) {
onOutgoingAudioCallInConversation(conversation.id);
} else {
onOutgoingVideoCallInConversation(conversation.id);
}
}
}}
i18n={i18n}
/>
);
2023-08-09 00:53:06 +00:00
return (
<div
key={key}
style={style}
2024-06-10 15:23:43 +00:00
data-type={item.type}
2024-10-25 20:31:30 +00:00
data-testid={item.peerId}
2023-08-09 00:53:06 +00:00
className={classNames('CallsList__Item', {
'CallsList__Item--selected': isSelected,
'CallsList__Item--missed': wasMissed,
2024-07-30 23:21:33 +00:00
'CallsList__Item--declined': wasDeclined,
2023-08-09 00:53:06 +00:00
})}
>
<ListTile
moduleClassName="CallsList__ItemTile"
aria-selected={isSelected}
leading={
<Avatar
acceptedMessageRequest
2024-07-11 19:44:09 +00:00
avatarUrl={conversation.avatarUrl}
color={conversation.color}
2024-04-01 19:19:35 +00:00
conversationType={conversation.type}
2023-08-09 00:53:06 +00:00
i18n={i18n}
isMe={false}
title={conversation.title}
sharedGroupNames={[]}
size={AvatarSize.THIRTY_SIX}
2023-08-09 00:53:06 +00:00
badge={undefined}
className="CallsList__ItemAvatar"
/>
}
trailing={isCallButtonVisible ? callButton : undefined}
2023-08-09 00:53:06 +00:00
title={
<span
className="CallsList__ItemTitle"
data-call={formatCallHistoryGroup(item)}
>
<UserText text={conversation.title} />
</span>
}
subtitleMaxLines={1}
2023-08-09 00:53:06 +00:00
subtitle={
<span className="CallsList__ItemCallInfo">
2023-08-09 00:53:06 +00:00
{item.children.length > 1 ? `(${item.children.length}) ` : ''}
{statusText} &middot;{' '}
{isActiveVisible ? (
i18n('icu:CallsList__ItemCallInfo--Active')
) : (
<Timestamp i18n={i18n} timestamp={item.timestamp} />
)}
2023-08-09 00:53:06 +00:00
</span>
}
onClick={() => {
2024-04-25 17:09:05 +00:00
if (isAdhoc) {
2024-05-22 16:24:27 +00:00
onChangeCallsTabSelectedView({
type: 'callLink',
roomId: item.peerId,
callHistoryGroup: item,
});
2024-04-01 19:19:35 +00:00
return;
}
2024-04-25 17:09:05 +00:00
if (conversation == null) {
return;
}
2024-05-22 16:24:27 +00:00
onChangeCallsTabSelectedView({
type: 'conversation',
conversationId: conversation.id,
callHistoryGroup: item,
});
2023-08-09 00:53:06 +00:00
}}
/>
</div>
);
},
[
activeCall,
2024-06-10 15:23:43 +00:00
rows,
searchStateQuery,
searchPending,
2024-04-01 19:19:35 +00:00
getCallLink,
2024-04-25 17:09:05 +00:00
getConversationForItem,
getIsAnybodyInCall,
getIsCallActive,
getIsInCall,
2024-11-13 20:24:00 +00:00
hasMissedCallFilter,
hasSearchStateQuery,
2023-08-09 00:53:06 +00:00
selectedCallHistoryGroup,
2024-05-22 16:24:27 +00:00
onChangeCallsTabSelectedView,
2024-06-10 15:23:43 +00:00
onCreateCallLink,
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
2024-04-01 19:19:35 +00:00
startCallLinkLobbyByRoomId,
toggleConfirmLeaveCallModal,
togglePip,
2023-08-09 00:53:06 +00:00
i18n,
2024-11-13 20:24:00 +00:00
isEmpty,
2023-08-09 00:53:06 +00:00
]
);
const handleSearchInputChange = useCallback(
(event: ChangeEvent<HTMLInputElement>) => {
setQueryInput(event.target.value);
},
[]
);
const handleSearchInputClear = useCallback(() => {
setQueryInput('');
}, []);
const handleStatusToggle = useCallback(() => {
2024-06-10 15:23:43 +00:00
setStatusInput(prevStatus => {
2023-08-09 00:53:06 +00:00
return prevStatus === CallHistoryFilterStatus.All
? CallHistoryFilterStatus.Missed
: CallHistoryFilterStatus.All;
});
}, []);
return (
<>
2024-08-13 23:34:42 +00:00
{isEmpty && !searchFiltering && (
<NavSidebarEmpty
title={i18n('icu:CallsList__EmptyState--noQuery__title')}
subtitle={i18n('icu:CallsList__EmptyState--noQuery__subtitle')}
/>
)}
2023-08-09 00:53:06 +00:00
<NavSidebarSearchHeader>
<SearchInput
i18n={i18n}
2024-11-13 20:24:00 +00:00
placeholder={
searchFiltering
? i18n('icu:CallsList__SearchInputPlaceholder--missed-calls')
: i18n('icu:CallsList__SearchInputPlaceholder')
}
2023-08-09 00:53:06 +00:00
onChange={handleSearchInputChange}
onClear={handleSearchInputClear}
value={queryInput}
/>
<Tooltip
direction={TooltipPlacement.Bottom}
content={i18n('icu:CallsList__ToggleFilterByMissedLabel')}
theme={Theme.Dark}
delay={600}
2023-08-09 00:53:06 +00:00
>
<button
className={classNames('CallsList__ToggleFilterByMissed', {
2024-11-13 20:24:00 +00:00
'CallsList__ToggleFilterByMissed--pressed': hasMissedCallFilter,
})}
type="button"
2024-11-13 20:24:00 +00:00
aria-pressed={hasMissedCallFilter}
aria-roledescription={i18n(
'icu:CallsList__ToggleFilterByMissed__RoleDescription'
)}
onClick={handleStatusToggle}
>
<span className="CallsList__ToggleFilterByMissedLabel">
{i18n('icu:CallsList__ToggleFilterByMissedLabel')}
</span>
</button>
</Tooltip>
2023-08-09 00:53:06 +00:00
</NavSidebarSearchHeader>
<SizeObserver>
{(ref, size) => {
return (
<div className="CallsList__ListContainer" ref={ref}>
{size != null && (
<InfiniteLoader
ref={infiniteLoaderRef}
isRowLoaded={isRowLoaded}
loadMoreRows={loadMoreRows}
rowCount={searchState.results?.count ?? Infinity}
2023-08-09 00:53:06 +00:00
minimumBatchSize={100}
threshold={30}
>
{({ onRowsRendered, registerChild }) => {
return (
<List
className={classNames('CallsList__List', {
2024-06-10 15:23:43 +00:00
'CallsList__List--disableScrolling':
searchState.results == null ||
searchState.results.count === 0,
2023-08-09 00:53:06 +00:00
})}
ref={refMerger(listRef, registerChild)}
width={size.width}
height={size.height}
2024-06-10 15:23:43 +00:00
rowCount={rowCount}
2023-08-09 00:53:06 +00:00
rowHeight={rowHeight}
rowRenderer={rowRenderer}
onRowsRendered={onRowsRendered}
/>
);
}}
</InfiniteLoader>
)}
</div>
);
}}
</SizeObserver>
</>
);
}