// Copyright 2018-2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React from 'react'; import classNames from 'classnames'; import moment from 'moment'; import { AttachmentSection } from './AttachmentSection'; import { EmptyState } from './EmptyState'; import { groupMediaItemsByDate } from './groupMediaItemsByDate'; import type { ItemClickEvent } from './types/ItemClickEvent'; import { missingCaseError } from '../../../util/missingCaseError'; import type { LocalizerType } from '../../../types/Util'; import { getMessageTimestamp } from '../../../util/getMessageTimestamp'; import type { MediaItemType } from '../../../types/MediaItem'; export type Props = { documents: Array; i18n: LocalizerType; media: Array; onItemClick?: (event: ItemClickEvent) => void; }; type State = { selectedTab: 'media' | 'documents'; }; const MONTH_FORMAT = 'MMMM YYYY'; type TabSelectEvent = { type: 'media' | 'documents'; }; function Tab({ isSelected, label, onSelect, type, }: { isSelected: boolean; label: string; onSelect?: (event: TabSelectEvent) => void; type: 'media' | 'documents'; }) { const handleClick = onSelect ? () => { onSelect({ type }); } : undefined; return ( // Has key events handled elsewhere // eslint-disable-next-line jsx-a11y/click-events-have-key-events
{label}
); } export class MediaGallery extends React.Component { public readonly focusRef: React.RefObject = React.createRef(); constructor(props: Props) { super(props); this.state = { selectedTab: 'media', }; } public override componentDidMount(): void { // 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(); } }); } public override render(): JSX.Element { const { i18n } = this.props; const { selectedTab } = this.state; return (
{this.renderSections()}
); } private readonly handleTabSelect = (event: TabSelectEvent): void => { this.setState({ selectedTab: event.type }); }; private renderSections() { const { i18n, media, documents, onItemClick } = this.props; const { selectedTab } = this.state; const mediaItems = selectedTab === 'media' ? media : documents; const type = selectedTab; if (!mediaItems || mediaItems.length === 0) { const label = (() => { switch (type) { case 'media': return i18n('mediaEmptyState'); case 'documents': return i18n('documentsEmptyState'); default: throw missingCaseError(type); } })(); return ; } const now = Date.now(); const sections = groupMediaItemsByDate(now, mediaItems).map(section => { const first = section.mediaItems[0]; const { message } = first; const date = moment(getMessageTimestamp(message)); const header = section.type === 'yearMonth' ? date.format(MONTH_FORMAT) : i18n(section.type); return ( ); }); return
{sections}
; } }