| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | // Copyright 2021 Signal Messenger, LLC
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ReactChild, ReactNode } from 'react'; | 
					
						
							|  |  |  | import React from 'react'; | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-20 09:41:21 -06:00
										 |  |  | import type { LocalizerType, ThemeType } from '../types/Util'; | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { ConversationType } from '../state/ducks/conversations'; | 
					
						
							| 
									
										
										
										
											2021-11-20 09:41:21 -06:00
										 |  |  | import type { PreferredBadgeSelectorType } from '../state/selectors/badges'; | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | import { ModalHost } from './ModalHost'; | 
					
						
							|  |  |  | import { Button, ButtonVariant } from './Button'; | 
					
						
							|  |  |  | import { Avatar, AvatarSize } from './Avatar'; | 
					
						
							|  |  |  | import { ContactName } from './conversation/ContactName'; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type PropsType = { | 
					
						
							|  |  |  |   children: ReactNode; | 
					
						
							|  |  |  |   i18n: LocalizerType; | 
					
						
							|  |  |  |   onClickPrimaryButton: () => void; | 
					
						
							|  |  |  |   onClose: () => void; | 
					
						
							|  |  |  |   primaryButtonText: string; | 
					
						
							|  |  |  |   title: string; | 
					
						
							|  |  |  | } & ( | 
					
						
							|  |  |  |   | // We use this empty type for an "all or nothing" setup.
 | 
					
						
							|  |  |  |   // eslint-disable-next-line @typescript-eslint/ban-types
 | 
					
						
							|  |  |  |   {} | 
					
						
							|  |  |  |   | { | 
					
						
							|  |  |  |       onClickSecondaryButton: () => void; | 
					
						
							|  |  |  |       secondaryButtonText: string; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-13 09:20:02 -05:00
										 |  |  | // TODO: This should use <Modal>. See DESKTOP-1038.
 | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | export function GroupDialog(props: Readonly<PropsType>): JSX.Element { | 
					
						
							|  |  |  |   const { | 
					
						
							|  |  |  |     children, | 
					
						
							|  |  |  |     i18n, | 
					
						
							|  |  |  |     onClickPrimaryButton, | 
					
						
							|  |  |  |     onClose, | 
					
						
							|  |  |  |     primaryButtonText, | 
					
						
							|  |  |  |     title, | 
					
						
							|  |  |  |   } = props; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let secondaryButton: undefined | ReactChild; | 
					
						
							|  |  |  |   if ('secondaryButtonText' in props) { | 
					
						
							|  |  |  |     const { onClickSecondaryButton, secondaryButtonText } = props; | 
					
						
							|  |  |  |     secondaryButton = ( | 
					
						
							|  |  |  |       <Button | 
					
						
							|  |  |  |         onClick={onClickSecondaryButton} | 
					
						
							|  |  |  |         variant={ButtonVariant.Secondary} | 
					
						
							|  |  |  |       > | 
					
						
							|  |  |  |         {secondaryButtonText} | 
					
						
							|  |  |  |       </Button> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							| 
									
										
										
										
											2022-09-27 13:24:21 -07:00
										 |  |  |     <ModalHost modalName="GroupDialog" onClose={onClose}> | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  |       <div className="module-GroupDialog"> | 
					
						
							|  |  |  |         <button | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |           aria-label={i18n('icu:close')} | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  |           type="button" | 
					
						
							|  |  |  |           className="module-GroupDialog__close-button" | 
					
						
							|  |  |  |           onClick={() => { | 
					
						
							|  |  |  |             onClose(); | 
					
						
							|  |  |  |           }} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |         <h1 className="module-GroupDialog__title">{title}</h1> | 
					
						
							|  |  |  |         <div className="module-GroupDialog__body">{children}</div> | 
					
						
							|  |  |  |         <div className="module-GroupDialog__button-container"> | 
					
						
							|  |  |  |           {secondaryButton} | 
					
						
							|  |  |  |           <Button | 
					
						
							|  |  |  |             onClick={onClickPrimaryButton} | 
					
						
							|  |  |  |             ref={focusRef} | 
					
						
							|  |  |  |             variant={ButtonVariant.Primary} | 
					
						
							|  |  |  |           > | 
					
						
							|  |  |  |             {primaryButtonText} | 
					
						
							|  |  |  |           </Button> | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     </ModalHost> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type ParagraphPropsType = { | 
					
						
							|  |  |  |   children: ReactNode; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | function Paragraph({ children }: Readonly<ParagraphPropsType>): JSX.Element { | 
					
						
							|  |  |  |   return <p className="module-GroupDialog__paragraph">{children}</p>; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GroupDialog.Paragraph = Paragraph; | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | type ContactsPropsType = { | 
					
						
							|  |  |  |   contacts: Array<ConversationType>; | 
					
						
							| 
									
										
										
										
											2021-11-20 09:41:21 -06:00
										 |  |  |   getPreferredBadge: PreferredBadgeSelectorType; | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  |   i18n: LocalizerType; | 
					
						
							| 
									
										
										
										
											2021-11-20 09:41:21 -06:00
										 |  |  |   theme: ThemeType; | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | function Contacts({ | 
					
						
							| 
									
										
										
										
											2021-11-20 09:41:21 -06:00
										 |  |  |   contacts, | 
					
						
							|  |  |  |   getPreferredBadge, | 
					
						
							|  |  |  |   i18n, | 
					
						
							|  |  |  |   theme, | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | }: Readonly<ContactsPropsType>): JSX.Element { | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <ul className="module-GroupDialog__contacts"> | 
					
						
							|  |  |  |       {contacts.map(contact => ( | 
					
						
							|  |  |  |         <li key={contact.id} className="module-GroupDialog__contacts__contact"> | 
					
						
							|  |  |  |           <Avatar | 
					
						
							| 
									
										
										
										
											2025-03-26 13:27:04 -07:00
										 |  |  |             avatarPlaceholderGradient={contact.avatarPlaceholderGradient} | 
					
						
							| 
									
										
										
										
											2024-07-11 12:44:09 -07:00
										 |  |  |             avatarUrl={contact.avatarUrl} | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |             badge={getPreferredBadge(contact.badges)} | 
					
						
							|  |  |  |             color={contact.color} | 
					
						
							|  |  |  |             conversationType={contact.type} | 
					
						
							| 
									
										
										
										
											2025-03-26 13:27:04 -07:00
										 |  |  |             hasAvatar={contact.hasAvatar} | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |             noteToSelf={contact.isMe} | 
					
						
							|  |  |  |             theme={theme} | 
					
						
							|  |  |  |             title={contact.title} | 
					
						
							|  |  |  |             sharedGroupNames={contact.sharedGroupNames} | 
					
						
							|  |  |  |             size={AvatarSize.TWENTY_EIGHT} | 
					
						
							|  |  |  |             i18n={i18n} | 
					
						
							|  |  |  |           /> | 
					
						
							|  |  |  |           <ContactName | 
					
						
							|  |  |  |             module="module-GroupDialog__contacts__contact__name" | 
					
						
							|  |  |  |             title={contact.title} | 
					
						
							|  |  |  |           /> | 
					
						
							|  |  |  |         </li> | 
					
						
							|  |  |  |       ))} | 
					
						
							|  |  |  |     </ul> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | GroupDialog.Contacts = Contacts; | 
					
						
							| 
									
										
										
										
											2021-03-03 14:09:58 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | function focusRef(el: HTMLElement | null) { | 
					
						
							|  |  |  |   if (el) { | 
					
						
							|  |  |  |     el.focus(); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |