2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
2021-02-23 20:34:28 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import { last } from 'lodash';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ReactChild } from 'react';
|
|
|
|
import React from 'react';
|
2021-02-23 20:34:28 +00:00
|
|
|
|
2024-05-15 21:48:02 +00:00
|
|
|
import { I18n } from '../I18n';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ToFindType } from './LeftPaneHelper';
|
2022-04-30 05:24:20 +00:00
|
|
|
import type {
|
|
|
|
ConversationType,
|
2022-06-16 19:12:50 +00:00
|
|
|
ShowConversationType,
|
2022-04-30 05:24:20 +00:00
|
|
|
} from '../../state/ducks/conversations';
|
2021-10-26 19:15:33 +00:00
|
|
|
import { LeftPaneHelper } from './LeftPaneHelper';
|
2021-02-23 20:34:28 +00:00
|
|
|
import { getConversationInDirection } from './getConversationInDirection';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { Row } from '../ConversationList';
|
|
|
|
import { RowType } from '../ConversationList';
|
|
|
|
import type { PropsData as ConversationListItemPropsType } from '../conversationList/ConversationListItem';
|
|
|
|
import type { LocalizerType } from '../../types/Util';
|
2021-11-01 18:43:02 +00:00
|
|
|
import { handleKeydownForSearch } from './handleKeydownForSearch';
|
2022-02-14 17:57:11 +00:00
|
|
|
import { LeftPaneSearchInput } from '../LeftPaneSearchInput';
|
2021-02-23 20:34:28 +00:00
|
|
|
|
|
|
|
export type LeftPaneInboxPropsType = {
|
|
|
|
conversations: ReadonlyArray<ConversationListItemPropsType>;
|
|
|
|
archivedConversations: ReadonlyArray<ConversationListItemPropsType>;
|
|
|
|
pinnedConversations: ReadonlyArray<ConversationListItemPropsType>;
|
2023-02-22 19:21:59 +00:00
|
|
|
isAboutToSearch: boolean;
|
2021-10-12 23:59:08 +00:00
|
|
|
startSearchCounter: number;
|
2022-01-27 22:12:26 +00:00
|
|
|
searchDisabled: boolean;
|
|
|
|
searchTerm: string;
|
|
|
|
searchConversation: undefined | ConversationType;
|
2021-02-23 20:34:28 +00:00
|
|
|
};
|
|
|
|
|
2021-04-26 16:38:50 +00:00
|
|
|
export class LeftPaneInboxHelper extends LeftPaneHelper<LeftPaneInboxPropsType> {
|
2021-02-23 20:34:28 +00:00
|
|
|
private readonly conversations: ReadonlyArray<ConversationListItemPropsType>;
|
|
|
|
|
2021-04-26 16:38:50 +00:00
|
|
|
private readonly archivedConversations: ReadonlyArray<ConversationListItemPropsType>;
|
2021-02-23 20:34:28 +00:00
|
|
|
|
2021-04-26 16:38:50 +00:00
|
|
|
private readonly pinnedConversations: ReadonlyArray<ConversationListItemPropsType>;
|
2021-02-23 20:34:28 +00:00
|
|
|
|
2023-02-22 19:21:59 +00:00
|
|
|
private readonly isAboutToSearch: boolean;
|
2021-10-12 23:59:08 +00:00
|
|
|
|
|
|
|
private readonly startSearchCounter: number;
|
|
|
|
|
2022-01-27 22:12:26 +00:00
|
|
|
private readonly searchDisabled: boolean;
|
|
|
|
|
|
|
|
private readonly searchTerm: string;
|
|
|
|
|
|
|
|
private readonly searchConversation: undefined | ConversationType;
|
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
constructor({
|
|
|
|
conversations,
|
|
|
|
archivedConversations,
|
|
|
|
pinnedConversations,
|
2023-02-22 19:21:59 +00:00
|
|
|
isAboutToSearch,
|
2021-10-12 23:59:08 +00:00
|
|
|
startSearchCounter,
|
2022-01-27 22:12:26 +00:00
|
|
|
searchDisabled,
|
|
|
|
searchTerm,
|
|
|
|
searchConversation,
|
2021-02-23 20:34:28 +00:00
|
|
|
}: Readonly<LeftPaneInboxPropsType>) {
|
|
|
|
super();
|
|
|
|
|
|
|
|
this.conversations = conversations;
|
|
|
|
this.archivedConversations = archivedConversations;
|
|
|
|
this.pinnedConversations = pinnedConversations;
|
2023-02-22 19:21:59 +00:00
|
|
|
this.isAboutToSearch = isAboutToSearch;
|
2021-10-12 23:59:08 +00:00
|
|
|
this.startSearchCounter = startSearchCounter;
|
2022-01-27 22:12:26 +00:00
|
|
|
this.searchDisabled = searchDisabled;
|
|
|
|
this.searchTerm = searchTerm;
|
|
|
|
this.searchConversation = searchConversation;
|
2021-02-23 20:34:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
getRowCount(): number {
|
|
|
|
const headerCount = this.hasPinnedAndNonpinned() ? 2 : 0;
|
|
|
|
const buttonCount = this.archivedConversations.length ? 1 : 0;
|
|
|
|
return (
|
|
|
|
headerCount +
|
|
|
|
this.pinnedConversations.length +
|
|
|
|
this.conversations.length +
|
|
|
|
buttonCount
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-01-27 22:12:26 +00:00
|
|
|
override getSearchInput({
|
|
|
|
clearConversationSearch,
|
|
|
|
clearSearch,
|
|
|
|
i18n,
|
2022-06-16 19:12:50 +00:00
|
|
|
showConversation,
|
2022-01-27 22:12:26 +00:00
|
|
|
updateSearchTerm,
|
|
|
|
}: Readonly<{
|
|
|
|
clearConversationSearch: () => unknown;
|
|
|
|
clearSearch: () => unknown;
|
|
|
|
i18n: LocalizerType;
|
2022-06-16 19:12:50 +00:00
|
|
|
showConversation: ShowConversationType;
|
2022-01-27 22:12:26 +00:00
|
|
|
updateSearchTerm: (searchTerm: string) => unknown;
|
|
|
|
}>): ReactChild {
|
|
|
|
return (
|
2022-02-14 17:57:11 +00:00
|
|
|
<LeftPaneSearchInput
|
2022-01-27 22:12:26 +00:00
|
|
|
clearConversationSearch={clearConversationSearch}
|
|
|
|
clearSearch={clearSearch}
|
|
|
|
disabled={this.searchDisabled}
|
|
|
|
i18n={i18n}
|
|
|
|
searchConversation={this.searchConversation}
|
|
|
|
searchTerm={this.searchTerm}
|
2022-06-16 19:12:50 +00:00
|
|
|
showConversation={showConversation}
|
2022-01-27 22:12:26 +00:00
|
|
|
startSearchCounter={this.startSearchCounter}
|
|
|
|
updateSearchTerm={updateSearchTerm}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
override getPreRowsNode({
|
2021-04-20 23:16:49 +00:00
|
|
|
i18n,
|
2022-01-27 22:12:26 +00:00
|
|
|
}: Readonly<{
|
|
|
|
i18n: LocalizerType;
|
|
|
|
}>): ReactChild | null {
|
2021-04-20 23:16:49 +00:00
|
|
|
if (this.getRowCount() === 0) {
|
|
|
|
return (
|
|
|
|
<div className="module-left-pane__empty">
|
|
|
|
<div>
|
2024-05-15 21:48:02 +00:00
|
|
|
<I18n
|
2021-04-20 23:16:49 +00:00
|
|
|
i18n={i18n}
|
2022-10-03 21:19:54 +00:00
|
|
|
id="icu:emptyInboxMessage"
|
|
|
|
components={{
|
|
|
|
composeIcon: (
|
|
|
|
<span>
|
2023-03-30 00:03:25 +00:00
|
|
|
<strong>{i18n('icu:composeIcon')}</strong>
|
2022-10-03 21:19:54 +00:00
|
|
|
<span className="module-left-pane__empty--composer_icon">
|
|
|
|
<i className="module-left-pane__empty--composer_icon--icon" />
|
|
|
|
</span>
|
2021-04-20 23:16:49 +00:00
|
|
|
</span>
|
2022-10-03 21:19:54 +00:00
|
|
|
),
|
|
|
|
}}
|
2021-04-20 23:16:49 +00:00
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
getRow(rowIndex: number): undefined | Row {
|
|
|
|
const { conversations, archivedConversations, pinnedConversations } = this;
|
|
|
|
|
|
|
|
const archivedConversationsCount = archivedConversations.length;
|
|
|
|
|
|
|
|
if (this.hasPinnedAndNonpinned()) {
|
|
|
|
switch (rowIndex) {
|
|
|
|
case 0:
|
|
|
|
return {
|
|
|
|
type: RowType.Header,
|
2023-03-30 00:03:25 +00:00
|
|
|
getHeaderText: i18n => i18n('icu:LeftPane--pinned'),
|
2021-02-23 20:34:28 +00:00
|
|
|
};
|
|
|
|
case pinnedConversations.length + 1:
|
|
|
|
return {
|
|
|
|
type: RowType.Header,
|
2023-03-30 00:03:25 +00:00
|
|
|
getHeaderText: i18n => i18n('icu:LeftPane--chats'),
|
2021-02-23 20:34:28 +00:00
|
|
|
};
|
|
|
|
case pinnedConversations.length + conversations.length + 2:
|
|
|
|
if (archivedConversationsCount) {
|
|
|
|
return {
|
|
|
|
type: RowType.ArchiveButton,
|
|
|
|
archivedConversationsCount,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return undefined;
|
|
|
|
default: {
|
|
|
|
const pinnedConversation = pinnedConversations[rowIndex - 1];
|
|
|
|
if (pinnedConversation) {
|
|
|
|
return {
|
|
|
|
type: RowType.Conversation,
|
|
|
|
conversation: pinnedConversation,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
const conversation =
|
|
|
|
conversations[rowIndex - pinnedConversations.length - 2];
|
|
|
|
return conversation
|
|
|
|
? {
|
|
|
|
type: RowType.Conversation,
|
|
|
|
conversation,
|
|
|
|
}
|
|
|
|
: undefined;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const onlyConversations = pinnedConversations.length
|
|
|
|
? pinnedConversations
|
|
|
|
: conversations;
|
|
|
|
if (rowIndex < onlyConversations.length) {
|
|
|
|
const conversation = onlyConversations[rowIndex];
|
|
|
|
return conversation
|
|
|
|
? {
|
|
|
|
type: RowType.Conversation,
|
|
|
|
conversation,
|
|
|
|
}
|
|
|
|
: undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (rowIndex === onlyConversations.length && archivedConversationsCount) {
|
|
|
|
return {
|
|
|
|
type: RowType.ArchiveButton,
|
|
|
|
archivedConversationsCount,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
override getRowIndexToScrollTo(
|
2021-02-23 20:34:28 +00:00
|
|
|
selectedConversationId: undefined | string
|
|
|
|
): undefined | number {
|
|
|
|
if (!selectedConversationId) {
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
const isConversationSelected = (
|
|
|
|
conversation: Readonly<ConversationListItemPropsType>
|
|
|
|
) => conversation.id === selectedConversationId;
|
|
|
|
const hasHeaders = this.hasPinnedAndNonpinned();
|
|
|
|
|
|
|
|
const pinnedConversationIndex = this.pinnedConversations.findIndex(
|
|
|
|
isConversationSelected
|
|
|
|
);
|
|
|
|
if (pinnedConversationIndex !== -1) {
|
|
|
|
const headerOffset = hasHeaders ? 1 : 0;
|
|
|
|
return pinnedConversationIndex + headerOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
const conversationIndex = this.conversations.findIndex(
|
|
|
|
isConversationSelected
|
|
|
|
);
|
|
|
|
if (conversationIndex !== -1) {
|
|
|
|
const pinnedOffset = this.pinnedConversations.length;
|
|
|
|
const headerOffset = hasHeaders ? 2 : 0;
|
|
|
|
return conversationIndex + pinnedOffset + headerOffset;
|
|
|
|
}
|
|
|
|
|
|
|
|
return undefined;
|
|
|
|
}
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
override requiresFullWidth(): boolean {
|
2021-10-12 23:59:08 +00:00
|
|
|
const hasNoConversations =
|
|
|
|
!this.conversations.length &&
|
|
|
|
!this.pinnedConversations.length &&
|
|
|
|
!this.archivedConversations.length;
|
2023-02-22 19:21:59 +00:00
|
|
|
return hasNoConversations || this.isAboutToSearch;
|
2021-10-12 23:59:08 +00:00
|
|
|
}
|
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
shouldRecomputeRowHeights(old: Readonly<LeftPaneInboxPropsType>): boolean {
|
|
|
|
return old.pinnedConversations.length !== this.pinnedConversations.length;
|
|
|
|
}
|
|
|
|
|
|
|
|
getConversationAndMessageAtIndex(
|
|
|
|
conversationIndex: number
|
|
|
|
): undefined | { conversationId: string } {
|
|
|
|
const { conversations, pinnedConversations } = this;
|
|
|
|
const conversation =
|
|
|
|
pinnedConversations[conversationIndex] ||
|
|
|
|
conversations[conversationIndex - pinnedConversations.length] ||
|
|
|
|
last(conversations) ||
|
|
|
|
last(pinnedConversations);
|
|
|
|
return conversation ? { conversationId: conversation.id } : undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
getConversationAndMessageInDirection(
|
|
|
|
toFind: Readonly<ToFindType>,
|
|
|
|
selectedConversationId: undefined | string,
|
2023-03-20 22:23:53 +00:00
|
|
|
_targetedMessageId: unknown
|
2021-02-23 20:34:28 +00:00
|
|
|
): undefined | { conversationId: string } {
|
|
|
|
return getConversationInDirection(
|
|
|
|
[...this.pinnedConversations, ...this.conversations],
|
|
|
|
toFind,
|
|
|
|
selectedConversationId
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
override onKeyDown(
|
2021-11-01 18:43:02 +00:00
|
|
|
event: KeyboardEvent,
|
|
|
|
options: Readonly<{
|
|
|
|
searchInConversation: (conversationId: string) => unknown;
|
|
|
|
selectedConversationId: undefined | string;
|
|
|
|
startSearch: () => unknown;
|
|
|
|
}>
|
|
|
|
): void {
|
|
|
|
handleKeydownForSearch(event, options);
|
|
|
|
}
|
|
|
|
|
2021-02-23 20:34:28 +00:00
|
|
|
private hasPinnedAndNonpinned(): boolean {
|
|
|
|
return Boolean(
|
|
|
|
this.pinnedConversations.length && this.conversations.length
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|