signal-desktop/ts/components/conversation/media-gallery/MediaGallery.tsx

292 lines
7.5 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2018 Signal Messenger, LLC
2020-10-30 20:34:04 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
2022-12-20 17:50:23 +00:00
import React, { useEffect, useRef } from 'react';
2018-04-11 15:57:31 +00:00
2018-04-13 00:56:05 +00:00
import moment from 'moment';
2022-12-20 17:50:23 +00:00
import type { ItemClickEvent } from './types/ItemClickEvent';
import type { LocalizerType } from '../../../types/Util';
import type { MediaItemType } from '../../../types/MediaItem';
import type { SaveAttachmentActionCreatorType } from '../../../state/ducks/conversations';
2018-04-15 01:11:40 +00:00
import { AttachmentSection } from './AttachmentSection';
2018-04-26 23:06:48 +00:00
import { EmptyState } from './EmptyState';
2022-12-20 17:50:23 +00:00
import { Tabs } from '../../Tabs';
import { groupMediaItemsByDate } from './groupMediaItemsByDate';
2018-04-26 23:06:48 +00:00
import { missingCaseError } from '../../../util/missingCaseError';
import { usePrevious } from '../../../hooks/usePrevious';
import type { AttachmentType } from '../../../types/Attachment';
2018-04-12 20:23:26 +00:00
2022-12-20 17:50:23 +00:00
enum TabViews {
Media = 'Media',
Documents = 'Documents',
}
export type Props = {
2022-12-20 17:50:23 +00:00
conversationId: string;
documents: ReadonlyArray<MediaItemType>;
2019-01-14 21:49:58 +00:00
i18n: LocalizerType;
haveOldestMedia: boolean;
haveOldestDocument: boolean;
loading: boolean;
initialLoad: (id: string) => unknown;
loadMoreMedia: (id: string) => unknown;
loadMoreDocuments: (id: string) => unknown;
media: ReadonlyArray<MediaItemType>;
2022-12-20 17:50:23 +00:00
saveAttachment: SaveAttachmentActionCreatorType;
showLightbox: (options: {
attachment: AttachmentType;
messageId: string;
}) => void;
};
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
2022-12-20 17:50:23 +00:00
function MediaSection({
documents,
2022-12-20 17:50:23 +00:00
i18n,
loading,
2022-12-20 17:50:23 +00:00
media,
saveAttachment,
showLightbox,
type,
2022-12-20 17:50:23 +00:00
}: Pick<
Props,
'documents' | 'i18n' | 'loading' | 'media' | 'saveAttachment' | 'showLightbox'
2022-12-20 17:50:23 +00:00
> & { type: 'media' | 'documents' }): JSX.Element {
const mediaItems = type === 'media' ? media : documents;
if (!mediaItems || mediaItems.length === 0) {
if (loading) {
return <div />;
}
2022-12-20 17:50:23 +00:00
const label = (() => {
switch (type) {
case 'media':
2023-03-30 00:03:25 +00:00
return i18n('icu:mediaEmptyState');
2022-12-20 17:50:23 +00:00
case 'documents':
2023-03-30 00:03:25 +00:00
return i18n('icu:documentsEmptyState');
2022-12-20 17:50:23 +00:00
default:
throw missingCaseError(type);
}
2022-12-20 17:50:23 +00:00
})();
2018-04-12 20:23:26 +00:00
2022-12-20 17:50:23 +00:00
return <EmptyState data-test="EmptyState" label={label} />;
2020-09-14 19:51:27 +00:00
}
2022-12-20 17:50:23 +00:00
const now = Date.now();
const sections = groupMediaItemsByDate(now, mediaItems).map(section => {
const first = section.mediaItems[0];
const { message } = first;
const date = moment(message.receivedAtMs || message.receivedAt);
function getHeader(): string {
switch (section.type) {
case 'yearMonth':
return date.format(MONTH_FORMAT);
case 'today':
2023-03-30 00:03:25 +00:00
return i18n('icu:today');
case 'yesterday':
2023-03-30 00:03:25 +00:00
return i18n('icu:yesterday');
case 'thisWeek':
2023-03-30 00:03:25 +00:00
return i18n('icu:thisWeek');
case 'thisMonth':
2023-03-30 00:03:25 +00:00
return i18n('icu:thisMonth');
default:
throw missingCaseError(section);
}
}
const header = getHeader();
2018-04-12 20:23:26 +00:00
2018-04-11 15:57:31 +00:00
return (
2022-12-20 17:50:23 +00:00
<AttachmentSection
key={header}
header={header}
i18n={i18n}
type={type}
mediaItems={section.mediaItems}
onItemClick={(event: ItemClickEvent) => {
switch (event.type) {
case 'documents': {
saveAttachment(event.attachment, event.message.sentAt);
2022-12-20 17:50:23 +00:00
break;
}
case 'media': {
showLightbox({
attachment: event.attachment,
messageId: event.message.id,
});
2022-12-20 17:50:23 +00:00
break;
}
default:
throw new TypeError(`Unknown attachment type: '${event.type}'`);
}
}}
/>
2018-04-11 15:57:31 +00:00
);
2022-12-20 17:50:23 +00:00
});
2018-04-12 20:23:26 +00:00
2022-12-20 17:50:23 +00:00
return <div className="module-media-gallery__sections">{sections}</div>;
}
export function MediaGallery({
conversationId,
documents,
haveOldestDocument,
haveOldestMedia,
2022-12-20 17:50:23 +00:00
i18n,
initialLoad,
loading,
loadMoreDocuments,
loadMoreMedia,
2022-12-20 17:50:23 +00:00
media,
saveAttachment,
showLightbox,
2022-12-20 17:50:23 +00:00
}: Props): JSX.Element {
const focusRef = useRef<HTMLDivElement | null>(null);
const scrollObserverRef = useRef<HTMLDivElement | null>(null);
const intersectionObserver = useRef<IntersectionObserver | null>(null);
const loadingRef = useRef<boolean>(false);
const tabViewRef = useRef<TabViews>(TabViews.Media);
2022-12-20 17:50:23 +00:00
useEffect(() => {
focusRef.current?.focus();
}, []);
useEffect(() => {
if (
media.length > 0 ||
documents.length > 0 ||
haveOldestDocument ||
haveOldestMedia
) {
return;
}
initialLoad(conversationId);
loadingRef.current = true;
}, [
conversationId,
haveOldestDocument,
haveOldestMedia,
initialLoad,
media,
documents,
]);
const previousLoading = usePrevious(loading, loading);
if (previousLoading && !loading) {
loadingRef.current = false;
}
useEffect(() => {
if (loading || !scrollObserverRef.current) {
return;
}
intersectionObserver.current?.disconnect();
intersectionObserver.current = null;
intersectionObserver.current = new IntersectionObserver(
(entries: ReadonlyArray<IntersectionObserverEntry>) => {
if (loadingRef.current) {
return;
}
const entry = entries.find(
item => item.target === scrollObserverRef.current
);
if (entry && entry.intersectionRatio > 0) {
if (tabViewRef.current === TabViews.Media) {
if (!haveOldestMedia) {
loadMoreMedia(conversationId);
loadingRef.current = true;
}
} else {
// eslint-disable-next-line no-lonely-if
if (!haveOldestDocument) {
loadMoreDocuments(conversationId);
loadingRef.current = true;
}
}
}
}
);
intersectionObserver.current.observe(scrollObserverRef.current);
return () => {
intersectionObserver.current?.disconnect();
intersectionObserver.current = null;
};
}, [
conversationId,
haveOldestDocument,
haveOldestMedia,
loading,
loadMoreDocuments,
loadMoreMedia,
]);
2022-12-20 17:50:23 +00:00
return (
<div className="module-media-gallery" tabIndex={-1} ref={focusRef}>
<Tabs
initialSelectedTab={TabViews.Media}
tabs={[
{
id: TabViews.Media,
2023-03-30 00:03:25 +00:00
label: i18n('icu:media'),
2022-12-20 17:50:23 +00:00
},
{
id: TabViews.Documents,
2023-03-30 00:03:25 +00:00
label: i18n('icu:documents'),
2022-12-20 17:50:23 +00:00
},
]}
>
{({ selectedTab }) => {
tabViewRef.current =
selectedTab === TabViews.Media
? TabViews.Media
: TabViews.Documents;
return (
<div className="module-media-gallery__content">
{selectedTab === TabViews.Media && (
<MediaSection
documents={documents}
i18n={i18n}
loading={loading}
media={media}
saveAttachment={saveAttachment}
showLightbox={showLightbox}
type="media"
/>
)}
{selectedTab === TabViews.Documents && (
<MediaSection
documents={documents}
i18n={i18n}
loading={loading}
media={media}
saveAttachment={saveAttachment}
showLightbox={showLightbox}
type="documents"
/>
)}
</div>
);
}}
2022-12-20 17:50:23 +00:00
</Tabs>
<div
ref={scrollObserverRef}
className="module-media-gallery__scroll-observer"
/>
2022-12-20 17:50:23 +00:00
</div>
);
2018-04-11 15:57:31 +00:00
}