// Copyright 2018 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ReactNode } from 'react'; import React from 'react'; import { compact, flatten } from 'lodash'; import { ContactName } from './ContactName'; import { SystemMessage } from './SystemMessage'; import { Intl } from '../Intl'; import type { LocalizerType } from '../../types/Util'; import { missingCaseError } from '../../util/missingCaseError'; import type { ConversationType } from '../../state/ducks/conversations'; export type ChangeType = 'add' | 'remove' | 'name' | 'avatar' | 'general'; type Change = { type: ChangeType; newName?: string; contacts?: Array; }; export type PropsData = { from: ConversationType; changes: Array; }; type PropsHousekeeping = { i18n: LocalizerType; }; export type Props = PropsData & PropsHousekeeping; function GroupNotificationChange({ change, from, i18n, }: { change: Change; from: ConversationType; i18n: LocalizerType; }): JSX.Element | null { const { contacts, type, newName } = change; const otherPeople: Array = compact( (contacts || []).map(contact => { if (contact.isMe) { return null; } return ( ); }) ); const otherPeopleWithCommas: Array = compact( flatten( otherPeople.map((person, index) => [index > 0 ? ', ' : null, person]) ) ); const contactsIncludesMe = (contacts || []).length !== otherPeople.length; switch (type) { case 'name': return ( ); case 'avatar': return ; case 'add': if (!contacts || !contacts.length) { throw new Error('Group update is missing contacts'); } return ( <> {otherPeople.length > 0 && ( <> {otherPeople.length === 1 ? ( ) : ( )} )} {contactsIncludesMe && (
)} ); case 'remove': if (from && from.isMe) { return <>{i18n('icu:youLeftTheGroup')}; } if (!contacts || !contacts.length) { throw new Error('Group update is missing contacts'); } return contacts.length > 1 ? ( ) : ( ); case 'general': return null; default: throw missingCaseError(type); } } export function GroupNotification({ changes: rawChanges, i18n, from, }: Props): JSX.Element { // This check is just to be extra careful, and can probably be removed. const changes: Array = Array.isArray(rawChanges) ? rawChanges : []; // Leave messages are always from the person leaving, so we omit the fromLabel if // the change is a 'leave.' const firstChange: undefined | Change = changes[0]; const isLeftOnly = changes.length === 1 && firstChange?.type === 'remove'; const fromLabel = from.isMe ? ( ) : ( }} /> ); let contents: ReactNode; if (isLeftOnly) { contents = ( ); } else { contents = ( <>

{fromLabel}

{changes.map((change, i) => ( // eslint-disable-next-line react/no-array-index-key

))} ); } return ; }