signal-desktop/ts/components/leftPane/LeftPaneArchiveHelper.tsx

252 lines
7.2 KiB
TypeScript
Raw Normal View History

2022-02-14 17:57:11 +00:00
// Copyright 2021-2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ReactChild } from 'react';
import React from 'react';
import { last } from 'lodash';
import type { ToFindType } from './LeftPaneHelper';
import { LeftPaneHelper } from './LeftPaneHelper';
import { getConversationInDirection } from './getConversationInDirection';
import type { Row } from '../ConversationList';
import { RowType } from '../ConversationList';
import type { PropsData as ConversationListItemPropsType } from '../conversationList/ConversationListItem';
import type { LocalizerType } from '../../types/Util';
import type {
ConversationType,
2022-06-16 19:12:50 +00:00
ShowConversationType,
} from '../../state/ducks/conversations';
2022-02-14 17:57:11 +00:00
import { LeftPaneSearchInput } from '../LeftPaneSearchInput';
2021-11-01 18:43:02 +00:00
import type { LeftPaneSearchPropsType } from './LeftPaneSearchHelper';
import { LeftPaneSearchHelper } from './LeftPaneSearchHelper';
import * as KeyboardLayout from '../../services/keyboardLayout';
2021-11-01 18:43:02 +00:00
type LeftPaneArchiveBasePropsType = {
archivedConversations: ReadonlyArray<ConversationListItemPropsType>;
2021-11-01 18:43:02 +00:00
searchConversation: undefined | ConversationType;
searchTerm: string;
startSearchCounter: number;
};
2021-11-01 18:43:02 +00:00
export type LeftPaneArchivePropsType =
| LeftPaneArchiveBasePropsType
| (LeftPaneArchiveBasePropsType & LeftPaneSearchPropsType);
2021-04-26 16:38:50 +00:00
export class LeftPaneArchiveHelper extends LeftPaneHelper<LeftPaneArchivePropsType> {
private readonly archivedConversations: ReadonlyArray<ConversationListItemPropsType>;
2021-11-01 18:43:02 +00:00
private readonly searchConversation: undefined | ConversationType;
private readonly searchTerm: string;
private readonly searchHelper: undefined | LeftPaneSearchHelper;
private readonly startSearchCounter: number;
2021-11-01 18:43:02 +00:00
constructor(props: Readonly<LeftPaneArchivePropsType>) {
super();
2021-11-01 18:43:02 +00:00
this.archivedConversations = props.archivedConversations;
this.searchConversation = props.searchConversation;
this.searchTerm = props.searchTerm;
this.startSearchCounter = props.startSearchCounter;
2021-11-01 18:43:02 +00:00
if ('conversationResults' in props) {
this.searchHelper = new LeftPaneSearchHelper(props);
}
}
override getHeaderContents({
i18n,
showInbox,
}: Readonly<{
i18n: LocalizerType;
showInbox: () => void;
}>): ReactChild {
return (
<div className="module-left-pane__header__contents">
<button
onClick={this.getBackAction({ showInbox })}
className="module-left-pane__header__contents__back-button"
title={i18n('backToInbox')}
aria-label={i18n('backToInbox')}
type="button"
/>
<div className="module-left-pane__header__contents__text">
2022-01-27 22:12:26 +00:00
{i18n('archivedConversations')}
</div>
</div>
);
}
2022-01-27 22:12:26 +00:00
override getSearchInput({
clearConversationSearch,
2022-01-27 22:12:26 +00:00
clearSearch,
i18n,
updateSearchTerm,
2022-06-16 19:12:50 +00:00
showConversation,
2022-01-27 22:12:26 +00:00
}: Readonly<{
clearConversationSearch: () => unknown;
clearSearch: () => unknown;
i18n: LocalizerType;
updateSearchTerm: (searchTerm: string) => unknown;
2022-06-16 19:12:50 +00:00
showConversation: ShowConversationType;
2022-01-27 22:12:26 +00:00
}>): ReactChild | null {
if (!this.searchConversation) {
return null;
}
return (
2022-02-14 17:57:11 +00:00
<LeftPaneSearchInput
clearConversationSearch={clearConversationSearch}
clearSearch={clearSearch}
2022-01-27 22:12:26 +00:00
i18n={i18n}
searchConversation={this.searchConversation}
searchTerm={this.searchTerm}
2022-06-16 19:12:50 +00:00
showConversation={showConversation}
startSearchCounter={this.startSearchCounter}
updateSearchTerm={updateSearchTerm}
2022-01-27 22:12:26 +00:00
/>
);
}
override getBackAction({ showInbox }: { showInbox: () => void }): () => void {
return showInbox;
}
override getPreRowsNode({
2021-11-01 18:43:02 +00:00
i18n,
}: Readonly<{ i18n: LocalizerType }>): ReactChild | null {
if (this.searchHelper) {
return this.searchHelper.getPreRowsNode({ i18n });
}
return (
<div className="module-left-pane__archive-helper-text">
2022-01-06 21:06:33 +00:00
{this.getRowCount() > 0
? i18n('archiveHelperText')
: i18n('noArchivedConversations')}
</div>
);
}
getRowCount(): number {
2021-11-01 18:43:02 +00:00
return (
this.searchHelper?.getRowCount() ?? this.archivedConversations.length
);
}
getRow(rowIndex: number): undefined | Row {
2021-11-01 18:43:02 +00:00
if (this.searchHelper) {
return this.searchHelper.getRow(rowIndex);
}
const conversation = this.archivedConversations[rowIndex];
return conversation
? {
type: RowType.Conversation,
conversation,
}
: undefined;
}
override getRowIndexToScrollTo(
selectedConversationId: undefined | string
): undefined | number {
2021-11-01 18:43:02 +00:00
if (this.searchHelper) {
return this.searchHelper.getRowIndexToScrollTo(selectedConversationId);
}
if (!selectedConversationId) {
return undefined;
}
const result = this.archivedConversations.findIndex(
conversation => conversation.id === selectedConversationId
);
return result === -1 ? undefined : result;
}
getConversationAndMessageAtIndex(
conversationIndex: number
): undefined | { conversationId: string } {
2021-11-01 18:43:02 +00:00
const { archivedConversations, searchHelper } = this;
if (searchHelper) {
return searchHelper.getConversationAndMessageAtIndex(conversationIndex);
}
const conversation =
archivedConversations[conversationIndex] || last(archivedConversations);
return conversation ? { conversationId: conversation.id } : undefined;
}
getConversationAndMessageInDirection(
toFind: Readonly<ToFindType>,
selectedConversationId: undefined | string,
2021-11-01 18:43:02 +00:00
selectedMessageId: unknown
): undefined | { conversationId: string } {
2021-11-01 18:43:02 +00:00
if (this.searchHelper) {
return this.searchHelper.getConversationAndMessageInDirection(
toFind,
selectedConversationId,
selectedMessageId
);
}
return getConversationInDirection(
this.archivedConversations,
toFind,
selectedConversationId
);
}
2021-11-01 18:43:02 +00:00
shouldRecomputeRowHeights(old: Readonly<LeftPaneArchivePropsType>): boolean {
const hasSearchingChanged =
'conversationResults' in old !== Boolean(this.searchHelper);
if (hasSearchingChanged) {
return true;
}
if ('conversationResults' in old && this.searchHelper) {
return this.searchHelper.shouldRecomputeRowHeights(old);
}
return false;
}
2021-11-01 18:43:02 +00:00
override onKeyDown(
2021-11-01 18:43:02 +00:00
event: KeyboardEvent,
{
searchInConversation,
selectedConversationId,
}: Readonly<{
searchInConversation: (conversationId: string) => unknown;
selectedConversationId: undefined | string;
}>
): void {
if (!selectedConversationId) {
return;
}
const { ctrlKey, metaKey, shiftKey } = event;
2021-11-01 18:43:02 +00:00
const commandKey = window.platform === 'darwin' && metaKey;
const controlKey = window.platform !== 'darwin' && ctrlKey;
const commandOrCtrl = commandKey || controlKey;
const commandAndCtrl = commandKey && ctrlKey;
const key = KeyboardLayout.lookup(event);
2021-11-01 18:43:02 +00:00
if (
commandOrCtrl &&
!commandAndCtrl &&
shiftKey &&
(key === 'f' || key === 'F') &&
2021-11-01 18:43:02 +00:00
this.archivedConversations.some(({ id }) => id === selectedConversationId)
) {
searchInConversation(selectedConversationId);
event.preventDefault();
event.stopPropagation();
}
}
}