274 lines
7.7 KiB
TypeScript
274 lines
7.7 KiB
TypeScript
// Copyright 2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import type { ReactNode, FunctionComponent } from 'react';
|
|
import React, { useMemo } from 'react';
|
|
import classNames from 'classnames';
|
|
import { isBoolean, isNumber } from 'lodash';
|
|
import { v4 as uuid } from 'uuid';
|
|
|
|
import { Avatar, AvatarSize } from '../Avatar';
|
|
import type { BadgeType } from '../../badges/types';
|
|
import { Timestamp } from '../conversation/Timestamp';
|
|
import { isConversationUnread } from '../../util/isConversationUnread';
|
|
import { cleanId } from '../_util';
|
|
import type { LocalizerType, ThemeType } from '../../types/Util';
|
|
import type { ConversationType } from '../../state/ducks/conversations';
|
|
import { Spinner } from '../Spinner';
|
|
|
|
const BASE_CLASS_NAME =
|
|
'module-conversation-list__item--contact-or-conversation';
|
|
const CONTENT_CLASS_NAME = `${BASE_CLASS_NAME}__content`;
|
|
const HEADER_CLASS_NAME = `${CONTENT_CLASS_NAME}__header`;
|
|
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 TIMESTAMP_CLASS_NAME = `${DATE_CLASS_NAME}__timestamp`;
|
|
const MESSAGE_CLASS_NAME = `${CONTENT_CLASS_NAME}__message`;
|
|
export const MESSAGE_TEXT_CLASS_NAME = `${MESSAGE_CLASS_NAME}__text`;
|
|
const CHECKBOX_CLASS_NAME = `${BASE_CLASS_NAME}__checkbox`;
|
|
|
|
type PropsType = {
|
|
checked?: boolean;
|
|
conversationType: 'group' | 'direct';
|
|
disabled?: boolean;
|
|
headerDate?: number;
|
|
headerName: ReactNode;
|
|
id?: string;
|
|
i18n: LocalizerType;
|
|
isNoteToSelf?: boolean;
|
|
isSelected: boolean;
|
|
isUsernameSearchResult?: boolean;
|
|
markedUnread?: boolean;
|
|
messageId?: string;
|
|
messageStatusIcon?: ReactNode;
|
|
messageText?: ReactNode;
|
|
messageTextIsAlwaysFullSize?: boolean;
|
|
onClick?: () => void;
|
|
shouldShowSpinner?: boolean;
|
|
unreadCount?: number;
|
|
} & Pick<
|
|
ConversationType,
|
|
| 'acceptedMessageRequest'
|
|
| 'avatarPath'
|
|
| 'color'
|
|
| 'isMe'
|
|
| 'markedUnread'
|
|
| 'name'
|
|
| 'phoneNumber'
|
|
| 'profileName'
|
|
| 'sharedGroupNames'
|
|
| 'title'
|
|
| 'unblurredAvatarPath'
|
|
> &
|
|
(
|
|
| { badge?: undefined; theme?: ThemeType }
|
|
| { badge: BadgeType; theme: ThemeType }
|
|
);
|
|
|
|
export const BaseConversationListItem: FunctionComponent<PropsType> =
|
|
React.memo(function BaseConversationListItem({
|
|
acceptedMessageRequest,
|
|
avatarPath,
|
|
badge,
|
|
checked,
|
|
color,
|
|
conversationType,
|
|
disabled,
|
|
headerDate,
|
|
headerName,
|
|
i18n,
|
|
id,
|
|
isMe,
|
|
isNoteToSelf,
|
|
isUsernameSearchResult,
|
|
isSelected,
|
|
markedUnread,
|
|
messageStatusIcon,
|
|
messageText,
|
|
messageTextIsAlwaysFullSize,
|
|
name,
|
|
onClick,
|
|
phoneNumber,
|
|
profileName,
|
|
sharedGroupNames,
|
|
shouldShowSpinner,
|
|
theme,
|
|
title,
|
|
unblurredAvatarPath,
|
|
unreadCount,
|
|
}) {
|
|
const identifier = id ? cleanId(id) : undefined;
|
|
const htmlId = useMemo(() => uuid(), []);
|
|
const isUnread = isConversationUnread({ markedUnread, unreadCount });
|
|
|
|
const isAvatarNoteToSelf = isBoolean(isNoteToSelf)
|
|
? isNoteToSelf
|
|
: Boolean(isMe);
|
|
|
|
const isCheckbox = isBoolean(checked);
|
|
|
|
let actionNode: ReactNode;
|
|
if (shouldShowSpinner) {
|
|
actionNode = (
|
|
<Spinner size="20px" svgSize="small" direction="on-progress-dialog" />
|
|
);
|
|
} else if (isCheckbox) {
|
|
let ariaLabel: string;
|
|
if (disabled) {
|
|
ariaLabel = i18n('cannotSelectContact', [title]);
|
|
} else if (checked) {
|
|
ariaLabel = i18n('deselectContact', [title]);
|
|
} else {
|
|
ariaLabel = i18n('selectContact', [title]);
|
|
}
|
|
actionNode = (
|
|
<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"
|
|
/>
|
|
);
|
|
}
|
|
|
|
const contents = (
|
|
<>
|
|
<Avatar
|
|
acceptedMessageRequest={acceptedMessageRequest}
|
|
avatarPath={avatarPath}
|
|
badge={badge}
|
|
color={color}
|
|
conversationType={conversationType}
|
|
noteToSelf={isAvatarNoteToSelf}
|
|
searchResult={isUsernameSearchResult}
|
|
i18n={i18n}
|
|
isMe={isMe}
|
|
name={name}
|
|
phoneNumber={phoneNumber}
|
|
profileName={profileName}
|
|
theme={theme}
|
|
title={title}
|
|
sharedGroupNames={sharedGroupNames}
|
|
size={AvatarSize.FORTY_EIGHT}
|
|
unblurredAvatarPath={unblurredAvatarPath}
|
|
/>
|
|
<div
|
|
className={classNames(
|
|
CONTENT_CLASS_NAME,
|
|
disabled && `${CONTENT_CLASS_NAME}--disabled`
|
|
)}
|
|
>
|
|
<div className={HEADER_CLASS_NAME}>
|
|
<div className={`${HEADER_CLASS_NAME}__name`}>{headerName}</div>
|
|
{isNumber(headerDate) && (
|
|
<div className={DATE_CLASS_NAME}>
|
|
<Timestamp
|
|
timestamp={headerDate}
|
|
extended={false}
|
|
module={TIMESTAMP_CLASS_NAME}
|
|
i18n={i18n}
|
|
/>
|
|
</div>
|
|
)}
|
|
</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}
|
|
{isUnread && <UnreadIndicator count={unreadCount} />}
|
|
</div>
|
|
) : null}
|
|
</div>
|
|
{actionNode}
|
|
</>
|
|
);
|
|
|
|
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}--is-checkbox--disabled`]: disabled }
|
|
)}
|
|
data-id={identifier}
|
|
htmlFor={htmlId}
|
|
// `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
|
|
aria-label={i18n('BaseConversationListItem__aria-label', { title })}
|
|
className={classNames(
|
|
commonClassNames,
|
|
`${BASE_CLASS_NAME}--is-button`
|
|
)}
|
|
data-id={identifier}
|
|
disabled={disabled}
|
|
onClick={onClick}
|
|
type="button"
|
|
>
|
|
{contents}
|
|
</button>
|
|
);
|
|
}
|
|
|
|
return (
|
|
<div className={commonClassNames} data-id={identifier}>
|
|
{contents}
|
|
</div>
|
|
);
|
|
});
|
|
|
|
function UnreadIndicator({ count = 0 }: Readonly<{ count?: number }>) {
|
|
let classModifier: undefined | string;
|
|
if (count > 99) {
|
|
classModifier = 'many';
|
|
} else if (count > 9) {
|
|
classModifier = 'two-digits';
|
|
}
|
|
|
|
return (
|
|
<div
|
|
className={classNames(
|
|
`${BASE_CLASS_NAME}__unread-indicator`,
|
|
classModifier &&
|
|
`${BASE_CLASS_NAME}__unread-indicator--${classModifier}`
|
|
)}
|
|
>
|
|
{Boolean(count) && Math.min(count, 99)}
|
|
</div>
|
|
);
|
|
}
|