| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2021 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { KeyboardEvent, MouseEvent, ReactNode } from 'react'; | 
					
						
							|  |  |  | import React, { memo, useEffect } from 'react'; | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | import classNames from 'classnames'; | 
					
						
							| 
									
										
										
										
											2021-10-06 17:00:51 -04:00
										 |  |  | import { useRestoreFocus } from '../hooks/useRestoreFocus'; | 
					
						
							| 
									
										
										
										
											2022-02-25 12:37:15 -06:00
										 |  |  | import { clearTimeoutIfNecessary } from '../util/clearTimeoutIfNecessary'; | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | export type PropsType = { | 
					
						
							|  |  |  |   autoDismissDisabled?: boolean; | 
					
						
							|  |  |  |   children: ReactNode; | 
					
						
							|  |  |  |   className?: string; | 
					
						
							| 
									
										
										
										
											2021-09-23 22:08:01 -04:00
										 |  |  |   disableCloseOnClick?: boolean; | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  |   onClose: () => unknown; | 
					
						
							|  |  |  |   timeout?: number; | 
					
						
							| 
									
										
										
										
											2021-10-06 17:00:51 -04:00
										 |  |  |   toastAction?: { | 
					
						
							|  |  |  |     label: string; | 
					
						
							|  |  |  |     onClick: () => unknown; | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2021-11-01 12:13:35 -07:00
										 |  |  |   style?: React.CSSProperties; | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export const Toast = memo(function ToastInner({ | 
					
						
							|  |  |  |   autoDismissDisabled = false, | 
					
						
							|  |  |  |   children, | 
					
						
							|  |  |  |   className, | 
					
						
							|  |  |  |   disableCloseOnClick = false, | 
					
						
							|  |  |  |   onClose, | 
					
						
							|  |  |  |   style, | 
					
						
							|  |  |  |   timeout = 8000, | 
					
						
							|  |  |  |   toastAction, | 
					
						
							|  |  |  | }: PropsType): JSX.Element | null { | 
					
						
							|  |  |  |   const [focusRef] = useRestoreFocus(); | 
					
						
							| 
									
										
										
										
											2023-06-13 09:40:48 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |     if (autoDismissDisabled) { | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |     const timeoutId = setTimeout(onClose, timeout); | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |     return () => { | 
					
						
							|  |  |  |       clearTimeoutIfNecessary(timeoutId); | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |   }, [autoDismissDisabled, onClose, timeout]); | 
					
						
							| 
									
										
										
										
											2021-09-22 16:59:54 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |   return ( | 
					
						
							|  |  |  |     <div | 
					
						
							|  |  |  |       aria-live="assertive" | 
					
						
							|  |  |  |       className={classNames('Toast', className)} | 
					
						
							|  |  |  |       onClick={() => { | 
					
						
							|  |  |  |         if (!disableCloseOnClick) { | 
					
						
							|  |  |  |           onClose(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }} | 
					
						
							|  |  |  |       onKeyDown={(ev: KeyboardEvent<HTMLDivElement>) => { | 
					
						
							|  |  |  |         if (ev.key === 'Enter' || ev.key === ' ') { | 
					
						
							|  |  |  |           if (!disableCloseOnClick) { | 
					
						
							|  |  |  |             onClose(); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }} | 
					
						
							|  |  |  |       role="button" | 
					
						
							|  |  |  |       tabIndex={0} | 
					
						
							|  |  |  |       style={style} | 
					
						
							|  |  |  |     > | 
					
						
							|  |  |  |       <div className="Toast__content">{children}</div> | 
					
						
							|  |  |  |       {toastAction && ( | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |         <div | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |           className="Toast__button" | 
					
						
							|  |  |  |           onClick={(ev: MouseEvent<HTMLDivElement>) => { | 
					
						
							|  |  |  |             ev.stopPropagation(); | 
					
						
							|  |  |  |             ev.preventDefault(); | 
					
						
							|  |  |  |             toastAction.onClick(); | 
					
						
							|  |  |  |             onClose(); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |           }} | 
					
						
							|  |  |  |           onKeyDown={(ev: KeyboardEvent<HTMLDivElement>) => { | 
					
						
							|  |  |  |             if (ev.key === 'Enter' || ev.key === ' ') { | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |               ev.stopPropagation(); | 
					
						
							|  |  |  |               ev.preventDefault(); | 
					
						
							|  |  |  |               toastAction.onClick(); | 
					
						
							|  |  |  |               onClose(); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |             } | 
					
						
							|  |  |  |           }} | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |           ref={focusRef} | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  |           role="button" | 
					
						
							|  |  |  |           tabIndex={0} | 
					
						
							|  |  |  |         > | 
					
						
							| 
									
										
										
										
											2024-01-29 12:09:54 -08:00
										 |  |  |           {toastAction.label} | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       )} | 
					
						
							|  |  |  |     </div> | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | }); |