signal-desktop/ts/components/conversationList/BaseConversationListItem.tsx

394 lines
11 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2021 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ReactNode, FunctionComponent } from 'react';
2022-01-31 22:45:56 +00:00
import React, { useCallback, useEffect, useMemo, useState } from 'react';
import classNames from 'classnames';
import { isBoolean, isNumber } from 'lodash';
import { v4 as generateUuid } from 'uuid';
import { Avatar, AvatarSize } from '../Avatar';
2021-11-02 23:01:13 +00:00
import type { BadgeType } from '../../badges/types';
import { isConversationUnread } from '../../util/isConversationUnread';
import { cleanId } from '../_util';
2021-11-02 23:01:13 +00:00
import type { LocalizerType, ThemeType } from '../../types/Util';
import type { ConversationType } from '../../state/ducks/conversations';
2021-11-12 01:17:29 +00:00
import { Spinner } from '../Spinner';
2022-01-31 22:45:56 +00:00
import { Time } from '../Time';
2022-01-26 23:05:26 +00:00
import { formatDateTimeShort } from '../../util/timestamp';
2022-01-31 22:45:56 +00:00
import * as durations from '../../util/durations';
const BASE_CLASS_NAME =
'module-conversation-list__item--contact-or-conversation';
const AVATAR_CONTAINER_CLASS_NAME = `${BASE_CLASS_NAME}__avatar-container`;
const CONTENT_CLASS_NAME = `${BASE_CLASS_NAME}__content`;
const HEADER_CLASS_NAME = `${CONTENT_CLASS_NAME}__header`;
2021-10-14 15:48:48 +00:00
export const HEADER_NAME_CLASS_NAME = `${HEADER_CLASS_NAME}__name`;
export const HEADER_CONTACT_NAME_CLASS_NAME = `${HEADER_NAME_CLASS_NAME}__contact-name`;
export const DATE_CLASS_NAME = `${HEADER_CLASS_NAME}__date`;
const MESSAGE_CLASS_NAME = `${CONTENT_CLASS_NAME}__message`;
export const MESSAGE_TEXT_CLASS_NAME = `${MESSAGE_CLASS_NAME}__text`;
2022-08-10 18:37:19 +00:00
const CHECKBOX_CONTAINER_CLASS_NAME = `${BASE_CLASS_NAME}__checkbox--container`;
2021-03-03 20:09:58 +00:00
const CHECKBOX_CLASS_NAME = `${BASE_CLASS_NAME}__checkbox`;
export const SPINNER_CLASS_NAME = `${BASE_CLASS_NAME}__spinner`;
type PropsType = {
2023-04-21 21:23:30 +00:00
buttonAriaLabel?: string;
2021-03-03 20:09:58 +00:00
checked?: boolean;
conversationType: 'group' | 'direct';
2021-03-03 20:09:58 +00:00
disabled?: boolean;
headerDate?: number;
headerName: ReactNode;
id?: string;
i18n: LocalizerType;
isNoteToSelf?: boolean;
isSelected: boolean;
2021-11-12 01:17:29 +00:00
isUsernameSearchResult?: boolean;
markedUnread?: boolean;
messageId?: string;
messageStatusIcon?: ReactNode;
messageText?: ReactNode;
messageTextIsAlwaysFullSize?: boolean;
2021-03-03 20:09:58 +00:00
onClick?: () => void;
onMouseDown?: () => void;
2021-11-12 01:17:29 +00:00
shouldShowSpinner?: boolean;
unreadCount?: number;
unreadMentionsCount?: number;
avatarSize?: AvatarSize;
2023-01-13 00:24:59 +00:00
testId?: string;
} & Pick<
ConversationType,
| 'acceptedMessageRequest'
2024-07-11 19:44:09 +00:00
| 'avatarUrl'
| 'color'
2023-01-13 00:24:59 +00:00
| 'groupId'
| 'isMe'
| 'markedUnread'
| 'phoneNumber'
| 'profileName'
| 'sharedGroupNames'
| 'title'
2024-07-11 19:44:09 +00:00
| 'unblurredAvatarUrl'
2023-08-16 20:54:39 +00:00
| 'serviceId'
2021-11-17 21:11:21 +00:00
> &
(
| { badge?: undefined; theme?: ThemeType }
| { badge: BadgeType; theme: ThemeType }
);
2021-11-11 22:43:05 +00:00
export const BaseConversationListItem: FunctionComponent<PropsType> =
React.memo(function BaseConversationListItem(props) {
const {
acceptedMessageRequest,
2024-07-11 19:44:09 +00:00
avatarUrl,
avatarSize,
2023-04-21 21:23:30 +00:00
buttonAriaLabel,
checked,
color,
conversationType,
disabled,
2023-01-13 00:24:59 +00:00
groupId,
headerDate,
headerName,
i18n,
id,
isMe,
isNoteToSelf,
isUsernameSearchResult,
isSelected,
markedUnread,
messageStatusIcon,
messageText,
messageTextIsAlwaysFullSize,
onClick,
onMouseDown,
phoneNumber,
profileName,
sharedGroupNames,
shouldShowSpinner,
2023-01-13 00:24:59 +00:00
testId: overrideTestId,
title,
2024-07-11 19:44:09 +00:00
unblurredAvatarUrl,
unreadCount,
unreadMentionsCount,
2023-08-16 20:54:39 +00:00
serviceId,
} = props;
const identifier = id ? cleanId(id) : undefined;
const htmlId = useMemo(() => generateUuid(), []);
2023-08-16 20:54:39 +00:00
const testId = overrideTestId || groupId || serviceId;
const isUnread = isConversationUnread({ markedUnread, unreadCount });
const isAvatarNoteToSelf = isBoolean(isNoteToSelf)
? isNoteToSelf
: Boolean(isMe);
2021-03-03 20:09:58 +00:00
const isCheckbox = isBoolean(checked);
2021-11-12 01:17:29 +00:00
let actionNode: ReactNode;
if (shouldShowSpinner) {
actionNode = (
<Spinner
size="20px"
svgSize="small"
moduleClassName={SPINNER_CLASS_NAME}
direction="on-progress-dialog"
/>
2021-11-12 01:17:29 +00:00
);
} else if (isCheckbox) {
2021-03-03 20:09:58 +00:00
let ariaLabel: string;
if (disabled) {
2023-03-30 00:03:25 +00:00
ariaLabel = i18n('icu:cannotSelectContact', {
2023-03-27 23:37:39 +00:00
name: title,
});
2021-03-03 20:09:58 +00:00
} else if (checked) {
2023-03-30 00:03:25 +00:00
ariaLabel = i18n('icu:deselectContact', {
2023-03-27 23:37:39 +00:00
name: title,
});
2021-03-03 20:09:58 +00:00
} else {
2023-03-30 00:03:25 +00:00
ariaLabel = i18n('icu:selectContact', {
2023-03-27 23:37:39 +00:00
name: title,
});
2021-03-03 20:09:58 +00:00
}
2021-11-12 01:17:29 +00:00
actionNode = (
2022-08-10 18:37:19 +00:00
<div className={CHECKBOX_CONTAINER_CLASS_NAME}>
<input
aria-label={ariaLabel}
checked={checked}
className={CHECKBOX_CLASS_NAME}
disabled={disabled}
id={htmlId}
onChange={onClick}
onKeyDown={event => {
if (onClick && !disabled && event.key === 'Enter') {
onClick();
}
}}
type="checkbox"
/>
</div>
2021-03-03 20:09:58 +00:00
);
}
const unreadIndicators = (() => {
if (!isUnread) {
return null;
}
return (
<div className={`${CONTENT_CLASS_NAME}__unread-indicators`}>
{unreadMentionsCount ? (
<UnreadIndicator variant={UnreadIndicatorVariant.UNREAD_MENTIONS} />
) : null}
{unreadCount ? (
<UnreadIndicator
variant={UnreadIndicatorVariant.UNREAD_MESSAGES}
count={unreadCount}
/>
) : (
<UnreadIndicator variant={UnreadIndicatorVariant.MARKED_UNREAD} />
)}
</div>
);
})();
2021-03-03 20:09:58 +00:00
const contents = (
<>
<div className={AVATAR_CONTAINER_CLASS_NAME}>
<Avatar
acceptedMessageRequest={acceptedMessageRequest}
2024-07-11 19:44:09 +00:00
avatarUrl={avatarUrl}
color={color}
conversationType={conversationType}
noteToSelf={isAvatarNoteToSelf}
searchResult={isUsernameSearchResult}
i18n={i18n}
isMe={isMe}
phoneNumber={phoneNumber}
profileName={profileName}
title={title}
sharedGroupNames={sharedGroupNames}
size={avatarSize ?? AvatarSize.FORTY_EIGHT}
2024-07-11 19:44:09 +00:00
unblurredAvatarUrl={unblurredAvatarUrl}
// This is here to appease the type checker.
{...(props.badge
? { badge: props.badge, theme: props.theme }
: { badge: undefined })}
/>
{unreadIndicators}
</div>
2021-03-03 20:09:58 +00:00
<div
className={classNames(
CONTENT_CLASS_NAME,
disabled && `${CONTENT_CLASS_NAME}--disabled`
)}
>
<div className={HEADER_CLASS_NAME}>
<div className={`${HEADER_CLASS_NAME}__name`}>{headerName}</div>
2022-01-26 23:05:26 +00:00
<Timestamp timestamp={headerDate} i18n={i18n} />
</div>
{messageText || isUnread ? (
<div className={MESSAGE_CLASS_NAME}>
{Boolean(messageText) && (
<div
dir="auto"
className={classNames(
MESSAGE_TEXT_CLASS_NAME,
messageTextIsAlwaysFullSize &&
`${MESSAGE_TEXT_CLASS_NAME}--always-full-size`
)}
>
{messageText}
</div>
)}
{messageStatusIcon}
{unreadIndicators}
</div>
) : null}
</div>
2021-11-12 01:17:29 +00:00
{actionNode}
2021-03-03 20:09:58 +00:00
</>
);
const commonClassNames = classNames(BASE_CLASS_NAME, {
[`${BASE_CLASS_NAME}--is-selected`]: isSelected,
});
if (isCheckbox) {
return (
<label
className={classNames(
commonClassNames,
`${BASE_CLASS_NAME}--is-checkbox`,
{ [`${BASE_CLASS_NAME}--disabled`]: disabled }
2021-03-03 20:09:58 +00:00
)}
data-id={identifier}
2023-01-13 00:24:59 +00:00
data-testid={testId}
htmlFor={htmlId}
2021-03-03 20:09:58 +00:00
// `onClick` is will double-fire if we're enabled. We want it to fire when we're
// disabled so we can show any "can't add contact" modals, etc. This won't
// work for keyboard users, though, because labels are not tabbable.
{...(disabled ? { onClick } : {})}
>
{contents}
</label>
);
}
if (onClick) {
return (
<button
2023-04-21 21:23:30 +00:00
aria-label={
buttonAriaLabel ||
i18n('icu:BaseConversationListItem__aria-label', {
title,
})
}
2021-03-03 20:09:58 +00:00
className={classNames(
commonClassNames,
`${BASE_CLASS_NAME}--is-button`
)}
data-id={identifier}
2023-01-13 00:24:59 +00:00
data-testid={testId}
2021-03-03 20:09:58 +00:00
disabled={disabled}
onClick={onClick}
onMouseDown={onMouseDown}
2021-03-03 20:09:58 +00:00
type="button"
>
{contents}
</button>
);
}
return (
2023-01-13 00:24:59 +00:00
<div
className={commonClassNames}
data-id={identifier}
data-testid={testId}
>
2021-03-03 20:09:58 +00:00
{contents}
</div>
);
2021-11-11 22:43:05 +00:00
});
2022-01-26 23:05:26 +00:00
function Timestamp({
i18n,
timestamp,
}: Readonly<{ i18n: LocalizerType; timestamp?: number }>) {
2022-01-31 22:45:56 +00:00
const getText = useCallback(
() => (isNumber(timestamp) ? formatDateTimeShort(i18n, timestamp) : ''),
[i18n, timestamp]
);
const [text, setText] = useState(getText());
useEffect(() => {
const update = () => setText(getText());
update();
const interval = setInterval(update, durations.MINUTE);
return () => {
clearInterval(interval);
};
}, [getText]);
2022-01-26 23:05:26 +00:00
if (!isNumber(timestamp)) {
return null;
}
return (
2022-01-31 22:45:56 +00:00
<Time className={DATE_CLASS_NAME} timestamp={timestamp}>
{text}
</Time>
2022-01-26 23:05:26 +00:00
);
}
enum UnreadIndicatorVariant {
MARKED_UNREAD = 'marked-unread',
UNREAD_MESSAGES = 'unread-messages',
UNREAD_MENTIONS = 'unread-mentions',
}
type UnreadIndicatorPropsType =
| {
variant: UnreadIndicatorVariant.MARKED_UNREAD;
}
| {
variant: UnreadIndicatorVariant.UNREAD_MESSAGES;
count: number;
}
| { variant: UnreadIndicatorVariant.UNREAD_MENTIONS };
function UnreadIndicator(props: UnreadIndicatorPropsType) {
let content: React.ReactNode;
switch (props.variant) {
case UnreadIndicatorVariant.MARKED_UNREAD:
content = null;
break;
case UnreadIndicatorVariant.UNREAD_MESSAGES:
content = props.count > 0 && props.count;
break;
case UnreadIndicatorVariant.UNREAD_MENTIONS:
content = (
<div
className={classNames(
`${BASE_CLASS_NAME}__unread-indicator--${props.variant}__icon`
)}
/>
);
break;
default:
throw new Error('Unexpected variant');
}
2021-10-14 20:21:10 +00:00
return (
<div
className={classNames(
`${BASE_CLASS_NAME}__unread-indicator`,
`${BASE_CLASS_NAME}__unread-indicator--${props.variant}`
)}
>
{content}
2021-10-14 20:21:10 +00:00
</div>
);
}