signal-desktop/ts/components/LeftPane.tsx

825 lines
28 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2019 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import React, { useEffect, useCallback, useMemo, useRef } from 'react';
import classNames from 'classnames';
2023-08-09 00:53:06 +00:00
import { isNumber } from 'lodash';
2019-01-14 21:49:58 +00:00
import type { LeftPaneHelper, ToFindType } from './leftPane/LeftPaneHelper';
import { FindDirection } from './leftPane/LeftPaneHelper';
import type { LeftPaneInboxPropsType } from './leftPane/LeftPaneInboxHelper';
import { LeftPaneInboxHelper } from './leftPane/LeftPaneInboxHelper';
import type { LeftPaneSearchPropsType } from './leftPane/LeftPaneSearchHelper';
import { LeftPaneSearchHelper } from './leftPane/LeftPaneSearchHelper';
import type { LeftPaneArchivePropsType } from './leftPane/LeftPaneArchiveHelper';
import { LeftPaneArchiveHelper } from './leftPane/LeftPaneArchiveHelper';
import type { LeftPaneComposePropsType } from './leftPane/LeftPaneComposeHelper';
import { LeftPaneComposeHelper } from './leftPane/LeftPaneComposeHelper';
import type { LeftPaneFindByUsernamePropsType } from './leftPane/LeftPaneFindByUsernameHelper';
import { LeftPaneFindByUsernameHelper } from './leftPane/LeftPaneFindByUsernameHelper';
import type { LeftPaneFindByPhoneNumberPropsType } from './leftPane/LeftPaneFindByPhoneNumberHelper';
import { LeftPaneFindByPhoneNumberHelper } from './leftPane/LeftPaneFindByPhoneNumberHelper';
import type { LeftPaneChooseGroupMembersPropsType } from './leftPane/LeftPaneChooseGroupMembersHelper';
import { LeftPaneChooseGroupMembersHelper } from './leftPane/LeftPaneChooseGroupMembersHelper';
import type { LeftPaneSetGroupMetadataPropsType } from './leftPane/LeftPaneSetGroupMetadataHelper';
import { LeftPaneSetGroupMetadataHelper } from './leftPane/LeftPaneSetGroupMetadataHelper';
import { LeftPaneMode } from '../types/leftPane';
2021-11-02 23:01:13 +00:00
import type { LocalizerType, ThemeType } from '../types/Util';
import { ScrollBehavior } from '../types/Util';
2021-11-17 21:11:21 +00:00
import type { PreferredBadgeSelectorType } from '../state/selectors/badges';
2021-09-17 22:24:21 +00:00
import { usePrevious } from '../hooks/usePrevious';
import { missingCaseError } from '../util/missingCaseError';
2022-11-16 20:18:02 +00:00
import type { DurationInSeconds } from '../util/durations';
import { WidthBreakpoint, getNavSidebarWidthBreakpoint } from './_util';
import * as KeyboardLayout from '../services/keyboardLayout';
import type { LookupConversationWithoutServiceIdActionsType } from '../util/lookupConversationWithoutServiceId';
2022-06-16 19:12:50 +00:00
import type { ShowConversationType } from '../state/ducks/conversations';
2023-01-18 23:31:10 +00:00
import type { PropsType as UnsupportedOSDialogPropsType } from '../state/smart/UnsupportedOSDialog';
2019-01-14 21:49:58 +00:00
import { ConversationList } from './ConversationList';
2021-03-03 20:09:58 +00:00
import { ContactCheckboxDisabledReason } from './conversationList/ContactCheckbox';
2023-01-18 23:31:10 +00:00
import type { PropsType as DialogExpiredBuildPropsType } from './DialogExpiredBuild';
import { LeftPaneBanner } from './LeftPaneBanner';
2019-03-12 00:20:16 +00:00
import type {
2021-08-06 00:17:05 +00:00
DeleteAvatarFromDiskActionType,
ReplaceAvatarActionType,
SaveAvatarToDiskActionType,
} from '../types/Avatar';
import { useSizeObserver } from '../hooks/useSizeObserver';
2023-08-09 00:53:06 +00:00
import {
NavSidebar,
NavSidebarActionButton,
NavSidebarSearchHeader,
} from './NavSidebar';
import { ContextMenu } from './ContextMenu';
2024-02-06 18:35:59 +00:00
import { EditState as ProfileEditorEditState } from './ProfileEditor';
import type { UnreadStats } from '../util/countUnreadStats';
2021-08-06 00:17:05 +00:00
export type PropsType = {
2023-08-21 20:12:27 +00:00
otherTabsUnreadStats: UnreadStats;
2023-01-18 23:31:10 +00:00
hasExpiredDialog: boolean;
hasFailedStorySends: boolean;
2023-01-18 23:31:10 +00:00
hasNetworkDialog: boolean;
hasPendingUpdate: boolean;
2023-01-18 23:31:10 +00:00
hasRelinkDialog: boolean;
hasUpdateDialog: boolean;
isUpdateDownloaded: boolean;
unsupportedOSDialogType: 'error' | 'warning' | undefined;
usernameCorrupted: boolean;
usernameLinkCorrupted: boolean;
2023-01-18 23:31:10 +00:00
// These help prevent invalid states. For example, we don't need the list of pinned
// conversations if we're trying to start a new conversation. Ideally these would be
// at the top level, but this is not supported by react-redux + TypeScript.
modeSpecificProps:
| ({
mode: LeftPaneMode.Inbox;
} & LeftPaneInboxPropsType)
| ({
mode: LeftPaneMode.Search;
} & LeftPaneSearchPropsType)
| ({
mode: LeftPaneMode.Archive;
} & LeftPaneArchivePropsType)
| ({
mode: LeftPaneMode.Compose;
2021-03-03 20:09:58 +00:00
} & LeftPaneComposePropsType)
| ({
mode: LeftPaneMode.FindByUsername;
} & LeftPaneFindByUsernamePropsType)
| ({
mode: LeftPaneMode.FindByPhoneNumber;
} & LeftPaneFindByPhoneNumberPropsType)
2021-03-03 20:09:58 +00:00
| ({
mode: LeftPaneMode.ChooseGroupMembers;
} & LeftPaneChooseGroupMembersPropsType)
| ({
mode: LeftPaneMode.SetGroupMetadata;
} & LeftPaneSetGroupMetadataPropsType);
2021-11-17 21:11:21 +00:00
getPreferredBadge: PreferredBadgeSelectorType;
2019-01-14 21:49:58 +00:00
i18n: LocalizerType;
2023-01-18 23:31:10 +00:00
isMacOS: boolean;
preferredWidthFromStorage: number;
selectedConversationId: undefined | string;
2023-03-20 22:23:53 +00:00
targetedMessageId: undefined | string;
challengeStatus: 'idle' | 'required' | 'pending';
setChallengeStatus: (status: 'idle') => void;
2022-01-11 20:02:46 +00:00
crashReportCount: number;
2021-11-02 23:01:13 +00:00
theme: ThemeType;
2019-01-14 21:49:58 +00:00
// Action Creators
2023-04-05 20:48:00 +00:00
blockConversation: (conversationId: string) => void;
2022-01-27 22:12:26 +00:00
clearConversationSearch: () => void;
2021-03-03 20:09:58 +00:00
clearGroupCreationError: () => void;
2021-11-01 18:43:02 +00:00
clearSearch: () => void;
2021-03-03 20:09:58 +00:00
closeMaximumGroupSizeModal: () => void;
closeRecommendedGroupSizeModal: () => void;
2022-06-16 19:12:50 +00:00
composeDeleteAvatarFromDisk: DeleteAvatarFromDiskActionType;
composeReplaceAvatar: ReplaceAvatarActionType;
composeSaveAvatarToDisk: SaveAvatarToDiskActionType;
2021-03-03 20:09:58 +00:00
createGroup: () => void;
endConversationSearch: () => void;
endSearch: () => void;
2023-08-09 00:53:06 +00:00
navTabsCollapsed: boolean;
2024-02-06 18:35:59 +00:00
openUsernameReservationModal: () => void;
2023-04-05 20:48:00 +00:00
onOutgoingAudioCallInConversation: (conversationId: string) => void;
onOutgoingVideoCallInConversation: (conversationId: string) => void;
removeConversation: (conversationId: string) => void;
savePreferredLeftPaneWidth: (_: number) => void;
2021-11-01 18:43:02 +00:00
searchInConversation: (conversationId: string) => unknown;
2021-09-24 00:49:05 +00:00
setComposeGroupAvatar: (_: undefined | Uint8Array) => void;
2022-11-16 20:18:02 +00:00
setComposeGroupExpireTimer: (_: DurationInSeconds) => void;
2022-06-16 19:12:50 +00:00
setComposeGroupName: (_: string) => void;
setComposeSearchTerm: (composeSearchTerm: string) => void;
setComposeSelectedRegion: (newRegion: string) => void;
2019-03-12 00:20:16 +00:00
showArchivedConversations: () => void;
2022-06-16 19:12:50 +00:00
showChooseGroupMembers: () => void;
showFindByUsername: () => void;
showFindByPhoneNumber: () => void;
2022-06-16 19:12:50 +00:00
showConversation: ShowConversationType;
2019-03-12 00:20:16 +00:00
showInbox: () => void;
startComposing: () => void;
2021-11-01 18:43:02 +00:00
startSearch: () => unknown;
2021-03-03 20:09:58 +00:00
startSettingGroupMetadata: () => void;
2021-08-06 00:17:05 +00:00
toggleComposeEditingAvatar: () => unknown;
2022-06-16 19:12:50 +00:00
toggleConversationInChooseMembers: (conversationId: string) => void;
2023-08-09 00:53:06 +00:00
toggleNavTabsCollapse: (navTabsCollapsed: boolean) => void;
2024-02-06 18:35:59 +00:00
toggleProfileEditor: (initialEditState?: ProfileEditorEditState) => void;
2021-11-01 18:43:02 +00:00
updateSearchTerm: (_: string) => void;
2019-01-14 21:49:58 +00:00
// Render Props
2021-08-11 16:23:21 +00:00
renderMessageSearchResult: (id: string) => JSX.Element;
renderNetworkStatus: (
_: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }>
) => JSX.Element;
2023-01-18 23:31:10 +00:00
renderUnsupportedOSDialog: (
_: Readonly<UnsupportedOSDialogPropsType>
) => JSX.Element;
renderRelinkDialog: (
_: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }>
) => JSX.Element;
renderUpdateDialog: (
_: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }>
) => JSX.Element;
renderCaptchaDialog: (props: { onSkip(): void }) => JSX.Element;
2022-01-11 20:02:46 +00:00
renderCrashReportDialog: () => JSX.Element;
2023-01-18 23:31:10 +00:00
renderExpiredBuildDialog: (_: DialogExpiredBuildPropsType) => JSX.Element;
renderToastManager: (_: {
containerWidthBreakpoint: WidthBreakpoint;
}) => JSX.Element;
} & LookupConversationWithoutServiceIdActionsType;
2019-01-14 21:49:58 +00:00
2022-11-18 00:45:19 +00:00
export function LeftPane({
2023-08-21 20:12:27 +00:00
otherTabsUnreadStats,
2023-04-05 20:48:00 +00:00
blockConversation,
2021-08-06 00:17:05 +00:00
challengeStatus,
2022-01-27 22:12:26 +00:00
clearConversationSearch,
2021-03-03 20:09:58 +00:00
clearGroupCreationError,
2021-11-01 18:43:02 +00:00
clearSearch,
2021-03-03 20:09:58 +00:00
closeMaximumGroupSizeModal,
closeRecommendedGroupSizeModal,
2021-08-06 00:17:05 +00:00
composeDeleteAvatarFromDisk,
composeReplaceAvatar,
composeSaveAvatarToDisk,
crashReportCount,
2021-03-03 20:09:58 +00:00
createGroup,
endConversationSearch,
endSearch,
2021-11-17 21:11:21 +00:00
getPreferredBadge,
2023-01-18 23:31:10 +00:00
hasExpiredDialog,
hasFailedStorySends,
2023-01-18 23:31:10 +00:00
hasNetworkDialog,
hasPendingUpdate,
2023-01-18 23:31:10 +00:00
hasRelinkDialog,
hasUpdateDialog,
i18n,
lookupConversationWithoutServiceId,
2023-01-18 23:31:10 +00:00
isMacOS,
isUpdateDownloaded,
modeSpecificProps,
2023-08-09 00:53:06 +00:00
navTabsCollapsed,
2023-04-05 20:48:00 +00:00
onOutgoingAudioCallInConversation,
onOutgoingVideoCallInConversation,
2023-08-09 00:53:06 +00:00
2024-02-06 18:35:59 +00:00
openUsernameReservationModal,
preferredWidthFromStorage,
2023-04-05 20:48:00 +00:00
removeConversation,
2021-08-06 00:17:05 +00:00
renderCaptchaDialog,
2022-01-11 20:02:46 +00:00
renderCrashReportDialog,
renderExpiredBuildDialog,
renderMessageSearchResult,
renderNetworkStatus,
2023-01-18 23:31:10 +00:00
renderUnsupportedOSDialog,
renderRelinkDialog,
renderUpdateDialog,
renderToastManager,
savePreferredLeftPaneWidth,
2021-11-01 18:43:02 +00:00
searchInConversation,
selectedConversationId,
2023-03-20 22:23:53 +00:00
targetedMessageId,
2023-08-09 00:53:06 +00:00
toggleNavTabsCollapse,
toggleProfileEditor,
2021-08-06 00:17:05 +00:00
setChallengeStatus,
2021-03-03 20:09:58 +00:00
setComposeGroupAvatar,
setComposeGroupExpireTimer,
2021-08-06 00:17:05 +00:00
setComposeGroupName,
setComposeSearchTerm,
setComposeSelectedRegion,
setIsFetchingUUID,
showArchivedConversations,
2021-08-06 00:17:05 +00:00
showChooseGroupMembers,
showFindByUsername,
showFindByPhoneNumber,
showConversation,
showInbox,
showUserNotFoundModal,
startComposing,
2021-11-01 18:43:02 +00:00
startSearch,
2021-03-03 20:09:58 +00:00
startSettingGroupMetadata,
2021-11-02 23:01:13 +00:00
theme,
2021-08-06 00:17:05 +00:00
toggleComposeEditingAvatar,
toggleConversationInChooseMembers,
2023-01-18 23:31:10 +00:00
unsupportedOSDialogType,
usernameCorrupted,
usernameLinkCorrupted,
2021-11-01 18:43:02 +00:00
updateSearchTerm,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
const previousModeSpecificProps = usePrevious(
modeSpecificProps,
modeSpecificProps
);
// The left pane can be in various modes: the inbox, the archive, the composer, etc.
// Ideally, this would render subcomponents such as `<LeftPaneInbox>` or
// `<LeftPaneArchive>` (and if there's a way to do that cleanly, we should refactor
// this).
//
// But doing that presents two problems:
//
// 1. Different components render the same logical inputs (the main header's search),
// but React doesn't know that they're the same, so you can lose focus as you change
// modes.
// 2. These components render virtualized lists, which are somewhat slow to initialize.
2022-02-09 20:33:19 +00:00
// Switching between modes can cause noticeable hiccups.
//
// To get around those problems, we use "helpers" which all correspond to the same
// interface.
//
// Unfortunately, there's a little bit of repetition here because TypeScript isn't quite
// smart enough.
let helper: LeftPaneHelper<unknown>;
let shouldRecomputeRowHeights: boolean;
switch (modeSpecificProps.mode) {
case LeftPaneMode.Inbox: {
const inboxHelper = new LeftPaneInboxHelper(modeSpecificProps);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? inboxHelper.shouldRecomputeRowHeights(previousModeSpecificProps)
: true;
helper = inboxHelper;
break;
}
case LeftPaneMode.Search: {
const searchHelper = new LeftPaneSearchHelper(modeSpecificProps);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? searchHelper.shouldRecomputeRowHeights(previousModeSpecificProps)
: true;
helper = searchHelper;
break;
}
case LeftPaneMode.Archive: {
const archiveHelper = new LeftPaneArchiveHelper(modeSpecificProps);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? archiveHelper.shouldRecomputeRowHeights(previousModeSpecificProps)
: true;
helper = archiveHelper;
break;
2019-03-12 00:20:16 +00:00
}
case LeftPaneMode.Compose: {
const composeHelper = new LeftPaneComposeHelper(modeSpecificProps);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? composeHelper.shouldRecomputeRowHeights(previousModeSpecificProps)
: true;
helper = composeHelper;
break;
2019-03-12 00:20:16 +00:00
}
case LeftPaneMode.FindByUsername: {
const findByUsernameHelper = new LeftPaneFindByUsernameHelper(
modeSpecificProps
);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? findByUsernameHelper.shouldRecomputeRowHeights(
previousModeSpecificProps
)
: true;
helper = findByUsernameHelper;
break;
}
case LeftPaneMode.FindByPhoneNumber: {
const findByPhoneNumberHelper = new LeftPaneFindByPhoneNumberHelper(
modeSpecificProps
);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? findByPhoneNumberHelper.shouldRecomputeRowHeights(
previousModeSpecificProps
)
: true;
helper = findByPhoneNumberHelper;
break;
}
2021-03-03 20:09:58 +00:00
case LeftPaneMode.ChooseGroupMembers: {
const chooseGroupMembersHelper = new LeftPaneChooseGroupMembersHelper(
modeSpecificProps
);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? chooseGroupMembersHelper.shouldRecomputeRowHeights(
previousModeSpecificProps
)
: true;
helper = chooseGroupMembersHelper;
break;
}
case LeftPaneMode.SetGroupMetadata: {
const setGroupMetadataHelper = new LeftPaneSetGroupMetadataHelper(
modeSpecificProps
);
shouldRecomputeRowHeights =
previousModeSpecificProps.mode === modeSpecificProps.mode
? setGroupMetadataHelper.shouldRecomputeRowHeights(
previousModeSpecificProps
)
: true;
helper = setGroupMetadataHelper;
break;
}
default:
throw missingCaseError(modeSpecificProps);
}
useEffect(() => {
const onKeyDown = (event: KeyboardEvent) => {
const { ctrlKey, shiftKey, altKey, metaKey } = event;
2023-01-18 23:31:10 +00:00
const commandOrCtrl = isMacOS ? metaKey : ctrlKey;
const key = KeyboardLayout.lookup(event);
if (key === 'Escape') {
2021-05-19 18:32:12 +00:00
const backAction = helper.getBackAction({
showInbox,
startComposing,
showChooseGroupMembers,
});
if (backAction) {
event.preventDefault();
event.stopPropagation();
backAction();
return;
}
}
if (
commandOrCtrl &&
!shiftKey &&
!altKey &&
(key === 'n' || key === 'N')
) {
startComposing();
event.preventDefault();
event.stopPropagation();
return;
2019-11-07 21:36:16 +00:00
}
let conversationToOpen:
| undefined
| {
conversationId: string;
messageId?: string;
};
2019-11-07 21:36:16 +00:00
const numericIndex = keyboardKeyToNumericIndex(event.key);
const openedByNumber = commandOrCtrl && isNumber(numericIndex);
if (openedByNumber) {
2021-11-11 22:43:05 +00:00
conversationToOpen =
helper.getConversationAndMessageAtIndex(numericIndex);
} else {
let toFind: undefined | ToFindType;
if (
(altKey && !shiftKey && key === 'ArrowUp') ||
(commandOrCtrl && shiftKey && key === '[') ||
(ctrlKey && shiftKey && key === 'Tab')
) {
toFind = { direction: FindDirection.Up, unreadOnly: false };
} else if (
(altKey && !shiftKey && key === 'ArrowDown') ||
(commandOrCtrl && shiftKey && key === ']') ||
(ctrlKey && key === 'Tab')
) {
toFind = { direction: FindDirection.Down, unreadOnly: false };
} else if (altKey && shiftKey && key === 'ArrowUp') {
toFind = { direction: FindDirection.Up, unreadOnly: true };
} else if (altKey && shiftKey && key === 'ArrowDown') {
toFind = { direction: FindDirection.Down, unreadOnly: true };
2019-11-07 21:36:16 +00:00
}
if (toFind) {
conversationToOpen = helper.getConversationAndMessageInDirection(
toFind,
selectedConversationId,
2023-03-20 22:23:53 +00:00
targetedMessageId
);
2019-11-07 21:36:16 +00:00
}
}
if (conversationToOpen) {
const { conversationId, messageId } = conversationToOpen;
2022-06-16 19:12:50 +00:00
showConversation({ conversationId, messageId });
if (openedByNumber) {
clearSearch();
}
event.preventDefault();
event.stopPropagation();
}
2021-11-01 18:43:02 +00:00
helper.onKeyDown(event, {
searchInConversation,
selectedConversationId,
startSearch,
});
};
2019-03-12 00:20:16 +00:00
document.addEventListener('keydown', onKeyDown);
return () => {
document.removeEventListener('keydown', onKeyDown);
};
}, [
2022-06-16 19:12:50 +00:00
clearSearch,
helper,
2023-01-18 23:31:10 +00:00
isMacOS,
2021-11-01 18:43:02 +00:00
searchInConversation,
selectedConversationId,
2023-03-20 22:23:53 +00:00
targetedMessageId,
2021-05-19 18:32:12 +00:00
showChooseGroupMembers,
2022-06-16 19:12:50 +00:00
showConversation,
2021-05-19 18:32:12 +00:00
showInbox,
startComposing,
2021-11-01 18:43:02 +00:00
startSearch,
]);
2024-08-13 23:34:42 +00:00
const backgroundNode = helper.getBackgroundNode({
i18n,
});
const preRowsNode = helper.getPreRowsNode({
2022-01-27 22:12:26 +00:00
clearConversationSearch,
2021-03-03 20:09:58 +00:00
clearGroupCreationError,
2022-01-27 22:12:26 +00:00
clearSearch,
2021-03-03 20:09:58 +00:00
closeMaximumGroupSizeModal,
closeRecommendedGroupSizeModal,
2021-08-06 00:17:05 +00:00
composeDeleteAvatarFromDisk,
composeReplaceAvatar,
composeSaveAvatarToDisk,
2021-03-03 20:09:58 +00:00
createGroup,
i18n,
2022-01-27 22:12:26 +00:00
removeSelectedContact: toggleConversationInChooseMembers,
2021-03-03 20:09:58 +00:00
setComposeGroupAvatar,
setComposeGroupExpireTimer,
2022-01-27 22:12:26 +00:00
setComposeGroupName,
2021-08-06 00:17:05 +00:00
toggleComposeEditingAvatar,
2021-03-03 20:09:58 +00:00
});
const footerContents = helper.getFooterContents({
createGroup,
i18n,
startSettingGroupMetadata,
lookupConversationWithoutServiceId,
showUserNotFoundModal,
setIsFetchingUUID,
showInbox,
showConversation,
});
2021-03-03 20:09:58 +00:00
const getRow = useMemo(() => helper.getRow.bind(helper), [helper]);
2021-08-11 16:23:21 +00:00
const onSelectConversation = useCallback(
(conversationId: string, messageId?: string) => {
2022-06-16 19:12:50 +00:00
showConversation({
2021-08-11 16:23:21 +00:00
conversationId,
messageId,
switchToAssociatedView: true,
});
},
2022-06-16 19:12:50 +00:00
[showConversation]
2021-08-11 16:23:21 +00:00
);
const previousSelectedConversationId = usePrevious(
selectedConversationId,
selectedConversationId
);
const isScrollable = helper.isScrollable();
let rowIndexToScrollTo: undefined | number;
let scrollBehavior: ScrollBehavior;
if (isScrollable) {
rowIndexToScrollTo =
previousSelectedConversationId === selectedConversationId
? undefined
: helper.getRowIndexToScrollTo(selectedConversationId);
scrollBehavior = ScrollBehavior.Default;
} else {
rowIndexToScrollTo = 0;
scrollBehavior = ScrollBehavior.Hard;
}
// We ensure that the listKey differs between some modes (e.g. inbox/archived), ensuring
// that AutoSizer properly detects the new size of its slot in the flexbox. The
// archive explainer text at the top of the archive view causes problems otherwise.
// It also ensures that we scroll to the top when switching views.
const listKey = preRowsNode ? 1 : 0;
const measureRef = useRef<HTMLDivElement>(null);
const measureSize = useSizeObserver(measureRef);
const widthBreakpoint = getNavSidebarWidthBreakpoint(
measureSize?.width ?? preferredWidthFromStorage
);
2023-01-18 23:31:10 +00:00
const commonDialogProps = {
i18n,
containerWidthBreakpoint: widthBreakpoint,
};
// Yellow dialogs
let maybeYellowDialog: JSX.Element | undefined;
if (unsupportedOSDialogType === 'warning') {
maybeYellowDialog = renderUnsupportedOSDialog({
type: 'warning',
...commonDialogProps,
});
} else if (hasNetworkDialog) {
maybeYellowDialog = renderNetworkStatus(commonDialogProps);
} else if (hasRelinkDialog) {
maybeYellowDialog = renderRelinkDialog(commonDialogProps);
}
// Update dialog
let maybeUpdateDialog: JSX.Element | undefined;
if (hasUpdateDialog && (!hasNetworkDialog || isUpdateDownloaded)) {
maybeUpdateDialog = renderUpdateDialog(commonDialogProps);
}
// Red dialogs
let maybeRedDialog: JSX.Element | undefined;
if (unsupportedOSDialogType === 'error') {
maybeRedDialog = renderUnsupportedOSDialog({
type: 'error',
...commonDialogProps,
});
} else if (hasExpiredDialog) {
maybeRedDialog = renderExpiredBuildDialog(commonDialogProps);
}
const dialogs = new Array<{ key: string; dialog: JSX.Element }>();
if (maybeRedDialog) {
dialogs.push({ key: 'red', dialog: maybeRedDialog });
if (maybeUpdateDialog) {
dialogs.push({ key: 'update', dialog: maybeUpdateDialog });
} else if (maybeYellowDialog) {
dialogs.push({ key: 'yellow', dialog: maybeYellowDialog });
}
} else {
if (maybeUpdateDialog) {
dialogs.push({ key: 'update', dialog: maybeUpdateDialog });
}
if (maybeYellowDialog) {
dialogs.push({ key: 'yellow', dialog: maybeYellowDialog });
}
}
let maybeBanner: JSX.Element | undefined;
if (usernameCorrupted) {
maybeBanner = (
<LeftPaneBanner
actionText={i18n('icu:LeftPane--corrupted-username--action-text')}
2024-02-06 18:35:59 +00:00
onClick={() => {
openUsernameReservationModal();
toggleProfileEditor(ProfileEditorEditState.Username);
}}
>
{i18n('icu:LeftPane--corrupted-username--text')}
</LeftPaneBanner>
);
} else if (usernameLinkCorrupted) {
maybeBanner = (
<LeftPaneBanner
actionText={i18n('icu:LeftPane--corrupted-username-link--action-text')}
2024-02-06 18:35:59 +00:00
onClick={() => toggleProfileEditor(ProfileEditorEditState.UsernameLink)}
>
{i18n('icu:LeftPane--corrupted-username-link--text')}
</LeftPaneBanner>
);
}
if (maybeBanner) {
dialogs.push({ key: 'banner', dialog: maybeBanner });
}
2024-02-16 22:01:06 +00:00
const hideHeader =
modeSpecificProps.mode === LeftPaneMode.Archive ||
modeSpecificProps.mode === LeftPaneMode.Compose ||
modeSpecificProps.mode === LeftPaneMode.FindByUsername ||
modeSpecificProps.mode === LeftPaneMode.FindByPhoneNumber ||
modeSpecificProps.mode === LeftPaneMode.ChooseGroupMembers ||
modeSpecificProps.mode === LeftPaneMode.SetGroupMetadata;
return (
2023-08-09 00:53:06 +00:00
<NavSidebar
2024-03-19 23:31:50 +00:00
title={i18n('icu:LeftPane--chats')}
2024-02-16 22:01:06 +00:00
hideHeader={hideHeader}
2023-08-09 00:53:06 +00:00
i18n={i18n}
2023-08-21 20:12:27 +00:00
otherTabsUnreadStats={otherTabsUnreadStats}
hasFailedStorySends={hasFailedStorySends}
hasPendingUpdate={hasPendingUpdate}
2023-08-09 00:53:06 +00:00
navTabsCollapsed={navTabsCollapsed}
onToggleNavTabsCollapse={toggleNavTabsCollapse}
preferredLeftPaneWidth={preferredWidthFromStorage}
requiresFullWidth={helper.requiresFullWidth()}
2023-08-09 00:53:06 +00:00
savePreferredLeftPaneWidth={savePreferredLeftPaneWidth}
renderToastManager={renderToastManager}
2023-08-09 00:53:06 +00:00
actions={
<>
<NavSidebarActionButton
label={i18n('icu:newConversation')}
icon={<span className="module-left-pane__startComposingIcon" />}
onClick={startComposing}
/>
<ContextMenu
i18n={i18n}
menuOptions={[
{
label: i18n('icu:avatarMenuViewArchive'),
onClick: showArchivedConversations,
},
]}
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="module-left-pane__moreActionsIcon" />}
label="More Actions"
/>
);
}}
</ContextMenu>
</>
}
>
2024-08-13 23:34:42 +00:00
{backgroundNode}
2023-08-09 00:53:06 +00:00
<nav
className={classNames(
'module-left-pane',
modeSpecificProps.mode === LeftPaneMode.ChooseGroupMembers &&
'module-left-pane--mode-choose-group-members',
modeSpecificProps.mode === LeftPaneMode.Compose &&
'module-left-pane--mode-compose'
)}
>
<div className="module-left-pane__header">
{helper.getHeaderContents({
i18n,
showInbox,
startComposing,
showChooseGroupMembers,
})}
</div>
{(widthBreakpoint === WidthBreakpoint.Wide ||
modeSpecificProps.mode !== LeftPaneMode.Inbox) && (
<NavSidebarSearchHeader>
{helper.getSearchInput({
clearConversationSearch,
clearSearch,
endConversationSearch,
endSearch,
i18n,
onChangeComposeSearchTerm: event => {
setComposeSearchTerm(event.target.value);
},
updateSearchTerm,
onChangeComposeSelectedRegion: setComposeSelectedRegion,
showConversation,
lookupConversationWithoutServiceId,
showUserNotFoundModal,
setIsFetchingUUID,
showInbox,
})}
</NavSidebarSearchHeader>
)}
2023-08-09 00:53:06 +00:00
<div className="module-left-pane__dialogs">
2024-02-16 22:01:06 +00:00
{!hideHeader &&
dialogs.map(({ key, dialog }) => (
<React.Fragment key={key}>{dialog}</React.Fragment>
))}
2023-08-09 00:53:06 +00:00
</div>
{preRowsNode && <React.Fragment key={0}>{preRowsNode}</React.Fragment>}
<div className="module-left-pane__list--measure" ref={measureRef}>
<div className="module-left-pane__list--wrapper">
<div
aria-live="polite"
className="module-left-pane__list"
data-supertab
key={listKey}
role="presentation"
tabIndex={-1}
>
<ConversationList
dimensions={measureSize ?? undefined}
getPreferredBadge={getPreferredBadge}
getRow={getRow}
i18n={i18n}
onClickArchiveButton={showArchivedConversations}
onClickContactCheckbox={(
conversationId: string,
disabledReason: undefined | ContactCheckboxDisabledReason
) => {
switch (disabledReason) {
case undefined:
toggleConversationInChooseMembers(conversationId);
break;
case ContactCheckboxDisabledReason.AlreadyAdded:
case ContactCheckboxDisabledReason.MaximumContactsSelected:
// These are no-ops.
break;
default:
throw missingCaseError(disabledReason);
}
}}
showUserNotFoundModal={showUserNotFoundModal}
setIsFetchingUUID={setIsFetchingUUID}
lookupConversationWithoutServiceId={
lookupConversationWithoutServiceId
}
showConversation={showConversation}
blockConversation={blockConversation}
onSelectConversation={onSelectConversation}
onOutgoingAudioCallInConversation={
onOutgoingAudioCallInConversation
}
onOutgoingVideoCallInConversation={
onOutgoingVideoCallInConversation
}
2023-10-25 23:01:16 +00:00
removeConversation={removeConversation}
renderMessageSearchResult={renderMessageSearchResult}
rowCount={helper.getRowCount()}
scrollBehavior={scrollBehavior}
scrollToRowIndex={rowIndexToScrollTo}
scrollable={isScrollable}
shouldRecomputeRowHeights={shouldRecomputeRowHeights}
showChooseGroupMembers={showChooseGroupMembers}
showFindByUsername={showFindByUsername}
showFindByPhoneNumber={showFindByPhoneNumber}
theme={theme}
/>
</div>
</div>
</div>
2023-08-09 00:53:06 +00:00
{footerContents && (
<div className="module-left-pane__footer">{footerContents}</div>
)}
2023-08-09 00:53:06 +00:00
{challengeStatus !== 'idle' &&
renderCaptchaDialog({
onSkip() {
setChallengeStatus('idle');
},
})}
{crashReportCount > 0 && renderCrashReportDialog()}
</nav>
</NavSidebar>
);
2022-11-18 00:45:19 +00:00
}
function keyboardKeyToNumericIndex(key: string): undefined | number {
if (key.length !== 1) {
return undefined;
}
const result = parseInt(key, 10) - 1;
const isValidIndex = Number.isInteger(result) && result >= 0 && result <= 8;
return isValidIndex ? result : undefined;
2019-01-14 21:49:58 +00:00
}