2021-11-12 23:44:20 +00:00
|
|
|
// Copyright 2018-2021 Signal Messenger, LLC
|
2020-10-30 20:34:04 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2018-04-11 15:57:31 +00:00
|
|
|
import React from 'react';
|
2018-07-18 00:15:34 +00:00
|
|
|
import classNames from 'classnames';
|
2018-04-11 15:57:31 +00:00
|
|
|
|
2018-04-13 00:56:05 +00:00
|
|
|
import moment from 'moment';
|
|
|
|
|
2018-04-15 01:11:40 +00:00
|
|
|
import { AttachmentSection } from './AttachmentSection';
|
2018-04-26 23:06:48 +00:00
|
|
|
import { EmptyState } from './EmptyState';
|
2018-11-14 18:47:19 +00:00
|
|
|
import { groupMediaItemsByDate } from './groupMediaItemsByDate';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { ItemClickEvent } from './types/ItemClickEvent';
|
2018-04-26 23:06:48 +00:00
|
|
|
import { missingCaseError } from '../../../util/missingCaseError';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { LocalizerType } from '../../../types/Util';
|
2021-03-04 21:44:57 +00:00
|
|
|
import { getMessageTimestamp } from '../../../util/getMessageTimestamp';
|
2018-04-12 20:23:26 +00:00
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { MediaItemType } from '../../../types/MediaItem';
|
2018-11-14 18:47:19 +00:00
|
|
|
|
2021-01-14 18:07:05 +00:00
|
|
|
export type Props = {
|
2018-11-14 18:47:19 +00:00
|
|
|
documents: Array<MediaItemType>;
|
2019-01-14 21:49:58 +00:00
|
|
|
i18n: LocalizerType;
|
2018-11-14 18:47:19 +00:00
|
|
|
media: Array<MediaItemType>;
|
2019-11-07 21:36:16 +00:00
|
|
|
|
2018-04-15 06:16:39 +00:00
|
|
|
onItemClick?: (event: ItemClickEvent) => void;
|
2021-01-14 18:07:05 +00:00
|
|
|
};
|
2018-04-12 20:23:26 +00:00
|
|
|
|
2021-01-14 18:07:05 +00:00
|
|
|
type State = {
|
2018-11-14 18:47:19 +00:00
|
|
|
selectedTab: 'media' | 'documents';
|
2021-01-14 18:07:05 +00:00
|
|
|
};
|
2018-04-11 15:57:31 +00:00
|
|
|
|
2018-04-13 00:56:05 +00:00
|
|
|
const MONTH_FORMAT = 'MMMM YYYY';
|
2018-04-12 20:23:26 +00:00
|
|
|
|
2021-01-14 18:07:05 +00:00
|
|
|
type TabSelectEvent = {
|
2018-11-14 18:47:19 +00:00
|
|
|
type: 'media' | 'documents';
|
2021-01-14 18:07:05 +00:00
|
|
|
};
|
2018-04-12 20:23:26 +00:00
|
|
|
|
|
|
|
const Tab = ({
|
|
|
|
isSelected,
|
|
|
|
label,
|
|
|
|
onSelect,
|
|
|
|
type,
|
|
|
|
}: {
|
2018-04-13 20:25:52 +00:00
|
|
|
isSelected: boolean;
|
|
|
|
label: string;
|
|
|
|
onSelect?: (event: TabSelectEvent) => void;
|
2018-11-14 18:47:19 +00:00
|
|
|
type: 'media' | 'documents';
|
2018-04-12 20:23:26 +00:00
|
|
|
}) => {
|
2018-05-22 19:31:43 +00:00
|
|
|
const handleClick = onSelect
|
|
|
|
? () => {
|
|
|
|
onSelect({ type });
|
|
|
|
}
|
|
|
|
: undefined;
|
2018-04-12 20:23:26 +00:00
|
|
|
|
|
|
|
return (
|
2020-09-14 19:51:27 +00:00
|
|
|
// Has key events handled elsewhere
|
|
|
|
// eslint-disable-next-line jsx-a11y/click-events-have-key-events
|
2018-04-12 20:23:26 +00:00
|
|
|
<div
|
2018-07-18 00:15:34 +00:00
|
|
|
className={classNames(
|
|
|
|
'module-media-gallery__tab',
|
|
|
|
isSelected ? 'module-media-gallery__tab--active' : null
|
|
|
|
)}
|
2018-04-12 20:23:26 +00:00
|
|
|
onClick={handleClick}
|
2018-05-22 19:31:43 +00:00
|
|
|
role="tab"
|
2019-11-21 19:16:06 +00:00
|
|
|
tabIndex={0}
|
2018-04-12 20:23:26 +00:00
|
|
|
>
|
|
|
|
{label}
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export class MediaGallery extends React.Component<Props, State> {
|
2019-11-07 21:36:16 +00:00
|
|
|
public readonly focusRef: React.RefObject<HTMLDivElement> = React.createRef();
|
2018-04-12 20:23:26 +00:00
|
|
|
|
2020-09-14 19:51:27 +00:00
|
|
|
constructor(props: Props) {
|
|
|
|
super(props);
|
|
|
|
this.state = {
|
|
|
|
selectedTab: 'media',
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
public override componentDidMount(): void {
|
2019-11-07 21:36:16 +00:00
|
|
|
// When this component is created, it's initially not part of the DOM, and then it's
|
|
|
|
// added off-screen and animated in. This ensures that the focus takes.
|
|
|
|
setTimeout(() => {
|
|
|
|
if (this.focusRef.current) {
|
|
|
|
this.focusRef.current.focus();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-11-12 23:44:20 +00:00
|
|
|
public override render(): JSX.Element {
|
2022-05-02 23:42:07 +00:00
|
|
|
const { i18n } = this.props;
|
2018-04-12 20:23:26 +00:00
|
|
|
const { selectedTab } = this.state;
|
|
|
|
|
2018-04-11 15:57:31 +00:00
|
|
|
return (
|
2019-11-21 19:16:06 +00:00
|
|
|
<div className="module-media-gallery" tabIndex={-1} ref={this.focusRef}>
|
2018-07-18 00:15:34 +00:00
|
|
|
<div className="module-media-gallery__tab-container">
|
2018-04-12 20:23:26 +00:00
|
|
|
<Tab
|
2022-05-02 23:42:07 +00:00
|
|
|
label={i18n('media')}
|
2018-04-12 20:23:26 +00:00
|
|
|
type="media"
|
|
|
|
isSelected={selectedTab === 'media'}
|
|
|
|
onSelect={this.handleTabSelect}
|
|
|
|
/>
|
|
|
|
<Tab
|
2022-05-02 23:42:07 +00:00
|
|
|
label={i18n('documents')}
|
2018-04-12 20:23:26 +00:00
|
|
|
type="documents"
|
|
|
|
isSelected={selectedTab === 'documents'}
|
|
|
|
onSelect={this.handleTabSelect}
|
|
|
|
/>
|
|
|
|
</div>
|
2018-07-18 00:15:34 +00:00
|
|
|
<div className="module-media-gallery__content">
|
|
|
|
{this.renderSections()}
|
|
|
|
</div>
|
2018-04-12 20:23:26 +00:00
|
|
|
</div>
|
2018-04-11 15:57:31 +00:00
|
|
|
);
|
|
|
|
}
|
2018-04-12 20:23:26 +00:00
|
|
|
|
2019-01-14 21:49:58 +00:00
|
|
|
private readonly handleTabSelect = (event: TabSelectEvent): void => {
|
2018-04-13 20:25:52 +00:00
|
|
|
this.setState({ selectedTab: event.type });
|
|
|
|
};
|
2018-04-13 14:55:34 +00:00
|
|
|
|
2018-04-13 00:56:05 +00:00
|
|
|
private renderSections() {
|
2018-04-15 06:16:39 +00:00
|
|
|
const { i18n, media, documents, onItemClick } = this.props;
|
2018-04-13 00:56:05 +00:00
|
|
|
const { selectedTab } = this.state;
|
|
|
|
|
2018-11-14 18:47:19 +00:00
|
|
|
const mediaItems = selectedTab === 'media' ? media : documents;
|
2018-04-13 00:56:05 +00:00
|
|
|
const type = selectedTab;
|
|
|
|
|
2018-11-14 18:47:19 +00:00
|
|
|
if (!mediaItems || mediaItems.length === 0) {
|
2018-04-26 23:06:48 +00:00
|
|
|
const label = (() => {
|
|
|
|
switch (type) {
|
|
|
|
case 'media':
|
|
|
|
return i18n('mediaEmptyState');
|
|
|
|
|
|
|
|
case 'documents':
|
|
|
|
return i18n('documentsEmptyState');
|
|
|
|
|
|
|
|
default:
|
|
|
|
throw missingCaseError(type);
|
|
|
|
}
|
|
|
|
})();
|
2018-05-22 19:31:43 +00:00
|
|
|
|
2018-04-26 23:06:48 +00:00
|
|
|
return <EmptyState data-test="EmptyState" label={label} />;
|
2018-04-13 00:56:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
const now = Date.now();
|
2018-11-14 18:47:19 +00:00
|
|
|
const sections = groupMediaItemsByDate(now, mediaItems).map(section => {
|
|
|
|
const first = section.mediaItems[0];
|
|
|
|
const { message } = first;
|
2021-03-04 21:44:57 +00:00
|
|
|
const date = moment(getMessageTimestamp(message));
|
2018-04-13 20:27:06 +00:00
|
|
|
const header =
|
|
|
|
section.type === 'yearMonth'
|
|
|
|
? date.format(MONTH_FORMAT)
|
|
|
|
: i18n(section.type);
|
2018-05-22 19:31:43 +00:00
|
|
|
|
2018-04-13 00:56:05 +00:00
|
|
|
return (
|
2018-04-15 01:11:40 +00:00
|
|
|
<AttachmentSection
|
2018-04-13 00:56:05 +00:00
|
|
|
key={header}
|
|
|
|
header={header}
|
|
|
|
i18n={i18n}
|
|
|
|
type={type}
|
2018-11-14 18:47:19 +00:00
|
|
|
mediaItems={section.mediaItems}
|
2018-04-15 06:16:39 +00:00
|
|
|
onItemClick={onItemClick}
|
2018-04-13 00:56:05 +00:00
|
|
|
/>
|
|
|
|
);
|
|
|
|
});
|
2018-04-26 23:06:13 +00:00
|
|
|
|
2018-07-18 00:15:34 +00:00
|
|
|
return <div className="module-media-gallery__sections">{sections}</div>;
|
2018-04-12 20:23:26 +00:00
|
|
|
}
|
2018-04-11 15:57:31 +00:00
|
|
|
}
|