| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2021 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  | import React, { useCallback } from 'react'; | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  | import type { RefObject } from 'react'; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | import classNames from 'classnames'; | 
					
						
							|  |  |  | import { noop } from 'lodash'; | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  | import { animated, useSpring } from '@react-spring/web'; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-10-26 14:15:33 -05:00
										 |  |  | import type { LocalizerType } from '../../types/Util'; | 
					
						
							|  |  |  | import type { AttachmentType } from '../../types/Attachment'; | 
					
						
							| 
									
										
										
										
											2022-12-21 15:44:23 -05:00
										 |  |  | import type { PushPanelForConversationActionType } from '../../state/ducks/conversations'; | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  | import { isDownloaded } from '../../types/Attachment'; | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  | import type { DirectionType, MessageStatusType } from './Message'; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  | import type { ComputePeaksResult } from '../VoiceNotesPlaybackContext'; | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  | import { MessageMetadata } from './MessageMetadata'; | 
					
						
							| 
									
										
										
										
											2025-06-16 11:59:31 -07:00
										 |  |  | import { createLogger } from '../../logging/log'; | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  | import type { ActiveAudioPlayerStateType } from '../../state/ducks/audioPlayer'; | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  | import { PlaybackRateButton } from '../PlaybackRateButton'; | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  | import { PlaybackButton } from '../PlaybackButton'; | 
					
						
							|  |  |  | import { WaveformScrubber } from './WaveformScrubber'; | 
					
						
							|  |  |  | import { useComputePeaks } from '../../hooks/useComputePeaks'; | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  | import { durationToPlaybackText } from '../../util/durationToPlaybackText'; | 
					
						
							| 
									
										
										
										
											2023-04-20 12:31:59 -04:00
										 |  |  | import { shouldNeverBeCalled } from '../../util/shouldNeverBeCalled'; | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  | import { formatFileSize } from '../../util/formatFileSize'; | 
					
						
							|  |  |  | import { roundFractionForProgressBar } from '../../util/numbers'; | 
					
						
							| 
									
										
										
										
											2021-04-15 14:02:24 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-06-16 11:59:31 -07:00
										 |  |  | const log = createLogger('MessageAudio'); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  | export type OwnProps = Readonly<{ | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |   active: | 
					
						
							|  |  |  |     | Pick< | 
					
						
							|  |  |  |         ActiveAudioPlayerStateType, | 
					
						
							|  |  |  |         'currentTime' | 'duration' | 'playing' | 'playbackRate' | 
					
						
							|  |  |  |       > | 
					
						
							|  |  |  |     | undefined; | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |   buttonRef: RefObject<HTMLButtonElement>; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |   i18n: LocalizerType; | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   attachment: AttachmentType; | 
					
						
							| 
									
										
										
										
											2022-03-08 08:32:42 -06:00
										 |  |  |   collapseMetadata: boolean; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |   withContentAbove: boolean; | 
					
						
							|  |  |  |   withContentBelow: boolean; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |   // Message properties. Many are needed for rendering metadata
 | 
					
						
							|  |  |  |   direction: DirectionType; | 
					
						
							|  |  |  |   expirationLength?: number; | 
					
						
							|  |  |  |   expirationTimestamp?: number; | 
					
						
							|  |  |  |   id: string; | 
					
						
							| 
									
										
										
										
											2021-07-27 10:42:25 -05:00
										 |  |  |   played: boolean; | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |   status?: MessageStatusType; | 
					
						
							|  |  |  |   textPending?: boolean; | 
					
						
							|  |  |  |   timestamp: number; | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |   cancelAttachmentDownload(): void; | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   kickOffAttachmentDownload(): void; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:51:53 -07:00
										 |  |  |   onCorrupted(): void; | 
					
						
							| 
									
										
										
										
											2021-04-15 14:02:24 -07:00
										 |  |  |   computePeaks(url: string, barCount: number): Promise<ComputePeaksResult>; | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |   onPlayMessage: (id: string, position: number) => void; | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  | }>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type DispatchProps = Readonly<{ | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  |   pushPanelForConversation: PushPanelForConversationActionType; | 
					
						
							| 
									
										
										
										
											2023-02-28 06:07:40 -07:00
										 |  |  |   setPosition: (positionAsRatio: number) => void; | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |   setPlaybackRate: (rate: number) => void; | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |   setIsPlaying: (value: boolean) => void; | 
					
						
							|  |  |  | }>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export type Props = OwnProps & DispatchProps; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  | enum State { | 
					
						
							|  |  |  |   NotDownloaded = 'NotDownloaded', | 
					
						
							|  |  |  |   Pending = 'Pending', | 
					
						
							| 
									
										
										
										
											2021-04-15 14:02:24 -07:00
										 |  |  |   Computing = 'Computing', | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   Normal = 'Normal', | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | // Constants
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const CSS_BASE = 'module-message__audio-attachment'; | 
					
						
							| 
									
										
										
										
											2021-03-24 16:08:57 -07:00
										 |  |  | const BAR_COUNT = 47; | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  | const BAR_NOT_DOWNLOADED_HEIGHT = 2; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | const BAR_MIN_HEIGHT = 4; | 
					
						
							|  |  |  | const BAR_MAX_HEIGHT = 20; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-03 17:43:44 -06:00
										 |  |  | const SPRING_CONFIG = { | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |   mass: 0.5, | 
					
						
							|  |  |  |   tension: 350, | 
					
						
							|  |  |  |   friction: 20, | 
					
						
							|  |  |  |   velocity: 0.01, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const DOT_DIV_WIDTH = 14; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | function PlayedDot({ | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |   played, | 
					
						
							|  |  |  |   onHide, | 
					
						
							|  |  |  | }: { | 
					
						
							|  |  |  |   played: boolean; | 
					
						
							|  |  |  |   onHide: () => void; | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | }) { | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |   const start = played ? 1 : 0; | 
					
						
							|  |  |  |   const end = played ? 0 : 1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-08-13 15:19:34 -07:00
										 |  |  |   // eslint-disable-next-line react-hooks/exhaustive-deps -- FIXME
 | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |   const [animProps] = useSpring( | 
					
						
							|  |  |  |     { | 
					
						
							| 
									
										
										
										
											2022-10-03 17:43:44 -06:00
										 |  |  |       config: SPRING_CONFIG, | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |       from: { scale: start, opacity: start, width: start }, | 
					
						
							|  |  |  |       to: { scale: end, opacity: end, width: end * DOT_DIV_WIDTH }, | 
					
						
							|  |  |  |       onRest: () => { | 
					
						
							|  |  |  |         if (played) { | 
					
						
							|  |  |  |           onHide(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     [played] | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return ( | 
					
						
							|  |  |  |     <animated.div | 
					
						
							|  |  |  |       style={animProps} | 
					
						
							|  |  |  |       aria-hidden="true" | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |       className={classNames( | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |         `${CSS_BASE}__dot`, | 
					
						
							|  |  |  |         `${CSS_BASE}__dot--${played ? 'played' : 'unplayed'}` | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |       )} | 
					
						
							|  |  |  |     /> | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Display message audio attachment along with its waveform, duration, and | 
					
						
							|  |  |  |  * toggle Play/Pause button. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * A global audio player is used for playback and access is managed by the | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |  * `active.content.current.id` and the `active.content.context` properties. Whenever both | 
					
						
							|  |  |  |  * are equal to `id` and `context` respectively the instance of the `MessageAudio` | 
					
						
							|  |  |  |  * assumes the ownership of the `Audio` instance and fully manages it. | 
					
						
							| 
									
										
										
										
											2021-06-29 12:58:29 -07:00
										 |  |  |  * | 
					
						
							|  |  |  |  * `context` is required for displaying separate MessageAudio instances in | 
					
						
							|  |  |  |  * MessageDetails and Message React components. | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | export function MessageAudio(props: Props): JSX.Element { | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |   const { | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |     active, | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |     buttonRef, | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |     i18n, | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |     attachment, | 
					
						
							| 
									
										
										
										
											2022-03-08 08:32:42 -06:00
										 |  |  |     collapseMetadata, | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |     withContentAbove, | 
					
						
							|  |  |  |     withContentBelow, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |     direction, | 
					
						
							|  |  |  |     expirationLength, | 
					
						
							|  |  |  |     expirationTimestamp, | 
					
						
							|  |  |  |     id, | 
					
						
							| 
									
										
										
										
											2021-07-27 10:42:25 -05:00
										 |  |  |     played, | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |     status, | 
					
						
							|  |  |  |     textPending, | 
					
						
							|  |  |  |     timestamp, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |     cancelAttachmentDownload, | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |     kickOffAttachmentDownload, | 
					
						
							| 
									
										
										
										
											2021-03-22 11:51:53 -07:00
										 |  |  |     onCorrupted, | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |     setPlaybackRate, | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |     onPlayMessage, | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  |     pushPanelForConversation, | 
					
						
							| 
									
										
										
										
											2023-02-28 06:07:40 -07:00
										 |  |  |     setPosition, | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |     setIsPlaying, | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |   } = props; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |   const isPlaying = active?.playing ?? false; | 
					
						
							| 
									
										
										
										
											2022-08-18 09:43:44 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |   const [isPlayedDotVisible, setIsPlayedDotVisible] = React.useState(!played); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const audioUrl = isDownloaded(attachment) ? attachment.url : undefined; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const { duration, hasPeaks, peaks } = useComputePeaks({ | 
					
						
							|  |  |  |     audioUrl, | 
					
						
							|  |  |  |     activeDuration: active?.duration, | 
					
						
							|  |  |  |     barCount: BAR_COUNT, | 
					
						
							|  |  |  |     onCorrupted, | 
					
						
							|  |  |  |   }); | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   let state: State; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (attachment.pending) { | 
					
						
							|  |  |  |     state = State.Pending; | 
					
						
							| 
									
										
										
										
											2022-03-28 21:10:08 -04:00
										 |  |  |   } else if (!isDownloaded(attachment)) { | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |     state = State.NotDownloaded; | 
					
						
							| 
									
										
										
										
											2021-04-15 14:02:24 -07:00
										 |  |  |   } else if (!hasPeaks) { | 
					
						
							|  |  |  |     state = State.Computing; | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     state = State.Normal; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const toggleIsPlaying = useCallback(() => { | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |     if (!isPlaying) { | 
					
						
							| 
									
										
										
										
											2021-03-16 10:49:19 -07:00
										 |  |  |       if (!attachment.url) { | 
					
						
							|  |  |  |         throw new Error( | 
					
						
							|  |  |  |           'Expected attachment url in the MessageAudio with ' + | 
					
						
							|  |  |  |             `state: ${state}` | 
					
						
							|  |  |  |         ); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |       if (active) { | 
					
						
							|  |  |  |         setIsPlaying(true); | 
					
						
							|  |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |         onPlayMessage(id, 0); | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |       } | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       setIsPlaying(false); | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   }, [ | 
					
						
							|  |  |  |     isPlaying, | 
					
						
							|  |  |  |     attachment.url, | 
					
						
							|  |  |  |     active, | 
					
						
							|  |  |  |     state, | 
					
						
							|  |  |  |     setIsPlaying, | 
					
						
							|  |  |  |     id, | 
					
						
							|  |  |  |     onPlayMessage, | 
					
						
							|  |  |  |   ]); | 
					
						
							| 
									
										
										
										
											2021-03-24 16:08:57 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const currentTimeOrZero = active?.currentTime ?? 0; | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const updatePosition = useCallback( | 
					
						
							|  |  |  |     (newPosition: number) => { | 
					
						
							|  |  |  |       if (active) { | 
					
						
							|  |  |  |         setPosition(newPosition); | 
					
						
							|  |  |  |         if (!active.playing) { | 
					
						
							|  |  |  |           setIsPlaying(true); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2023-02-28 06:07:40 -07:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |       if (attachment.url) { | 
					
						
							|  |  |  |         onPlayMessage(id, newPosition); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         log.warn('Waveform clicked on attachment with no url'); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     [active, attachment.url, id, onPlayMessage, setIsPlaying, setPosition] | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2023-02-28 06:07:40 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const handleWaveformClick = useCallback( | 
					
						
							|  |  |  |     (positionAsRatio: number) => { | 
					
						
							|  |  |  |       if (state !== State.Normal) { | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |       updatePosition(positionAsRatio); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     [state, updatePosition] | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |   const handleWaveformScrub = useCallback( | 
					
						
							|  |  |  |     (amountInSeconds: number) => { | 
					
						
							|  |  |  |       const currentPosition = currentTimeOrZero / duration; | 
					
						
							|  |  |  |       const positionIncrement = amountInSeconds / duration; | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |       updatePosition( | 
					
						
							|  |  |  |         Math.min(Math.max(0, currentPosition + positionIncrement), duration) | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     }, | 
					
						
							|  |  |  |     [currentTimeOrZero, duration, updatePosition] | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const waveform = ( | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |     <WaveformScrubber | 
					
						
							|  |  |  |       i18n={i18n} | 
					
						
							|  |  |  |       peaks={peaks} | 
					
						
							|  |  |  |       duration={duration} | 
					
						
							|  |  |  |       currentTime={currentTimeOrZero} | 
					
						
							|  |  |  |       barMinHeight={ | 
					
						
							|  |  |  |         state !== State.Normal ? BAR_NOT_DOWNLOADED_HEIGHT : BAR_MIN_HEIGHT | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       barMaxHeight={BAR_MAX_HEIGHT} | 
					
						
							|  |  |  |       onClick={handleWaveformClick} | 
					
						
							|  |  |  |       onScrub={handleWaveformScrub} | 
					
						
							|  |  |  |     /> | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   let button: React.ReactElement; | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |   if (state === State.Computing) { | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |     // Not really a button, but who cares?
 | 
					
						
							|  |  |  |     button = ( | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |       <PlaybackButton | 
					
						
							|  |  |  |         variant="message" | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |         mod="computing" | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |         onClick={noop} | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         label={i18n('icu:MessageAudio--pending')} | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |         context={direction} | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |       /> | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |   } else if (state === State.Pending) { | 
					
						
							|  |  |  |     // Not really a button, but who cares?
 | 
					
						
							|  |  |  |     const downloadFraction = | 
					
						
							|  |  |  |       attachment.size && attachment.totalDownloaded | 
					
						
							|  |  |  |         ? roundFractionForProgressBar( | 
					
						
							|  |  |  |             attachment.totalDownloaded / attachment.size | 
					
						
							|  |  |  |           ) | 
					
						
							|  |  |  |         : undefined; | 
					
						
							|  |  |  |     button = ( | 
					
						
							|  |  |  |       <PlaybackButton | 
					
						
							|  |  |  |         variant="message" | 
					
						
							|  |  |  |         mod="downloading" | 
					
						
							|  |  |  |         downloadFraction={downloadFraction} | 
					
						
							|  |  |  |         onClick={cancelAttachmentDownload} | 
					
						
							|  |  |  |         label={i18n('icu:MessageAudio--pending')} | 
					
						
							|  |  |  |         context={direction} | 
					
						
							|  |  |  |       /> | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |   } else if (state === State.NotDownloaded) { | 
					
						
							|  |  |  |     button = ( | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |       <PlaybackButton | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |         ref={buttonRef} | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |         variant="message" | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |         mod="not-downloaded" | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |         label={i18n('icu:MessageAudio--download')} | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |         onClick={kickOffAttachmentDownload} | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |         context={direction} | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |       /> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     // State.Normal
 | 
					
						
							|  |  |  |     button = ( | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |       <PlaybackButton | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |         ref={buttonRef} | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |         variant="message" | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |         mod={isPlaying ? 'pause' : 'play'} | 
					
						
							| 
									
										
										
										
											2023-02-21 10:44:31 -07:00
										 |  |  |         label={ | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |           isPlaying | 
					
						
							|  |  |  |             ? i18n('icu:MessageAudio--pause') | 
					
						
							|  |  |  |             : i18n('icu:MessageAudio--play') | 
					
						
							| 
									
										
										
										
											2023-02-21 10:44:31 -07:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |         onClick={toggleIsPlaying} | 
					
						
							| 
									
										
										
										
											2023-03-02 13:55:40 -07:00
										 |  |  |         context={direction} | 
					
						
							| 
									
										
										
										
											2021-03-15 17:59:48 -07:00
										 |  |  |       /> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |   const countDown = Math.max(0, duration - (active?.currentTime ?? 0)); | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |   const fileSizeOrDuration = | 
					
						
							|  |  |  |     state === State.NotDownloaded || state === State.Pending || duration < 1 | 
					
						
							|  |  |  |       ? formatFileSize(attachment.size) | 
					
						
							|  |  |  |       : durationToPlaybackText(countDown); | 
					
						
							| 
									
										
										
										
											2021-03-22 11:15:59 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |   const metadata = ( | 
					
						
							|  |  |  |     <div className={`${CSS_BASE}__metadata`}> | 
					
						
							| 
									
										
										
										
											2022-03-09 15:45:18 -06:00
										 |  |  |       <div | 
					
						
							| 
									
										
										
										
											2022-04-25 14:12:22 -07:00
										 |  |  |         aria-hidden="true" | 
					
						
							| 
									
										
										
										
											2022-03-09 15:45:18 -06:00
										 |  |  |         className={classNames( | 
					
						
							|  |  |  |           `${CSS_BASE}__countdown`, | 
					
						
							|  |  |  |           `${CSS_BASE}__countdown--${played ? 'played' : 'unplayed'}` | 
					
						
							|  |  |  |         )} | 
					
						
							|  |  |  |       > | 
					
						
							| 
									
										
										
										
											2025-03-04 10:09:43 +10:00
										 |  |  |         {fileSizeOrDuration} | 
					
						
							| 
									
										
										
										
											2022-03-09 15:45:18 -06:00
										 |  |  |       </div> | 
					
						
							| 
									
										
										
										
											2022-08-31 14:42:09 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |       <div className={`${CSS_BASE}__controls`}> | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |         <PlayedDot | 
					
						
							|  |  |  |           played={played} | 
					
						
							|  |  |  |           onHide={() => setIsPlayedDotVisible(false)} | 
					
						
							| 
									
										
										
										
											2022-08-31 14:42:09 -06:00
										 |  |  |         /> | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |         <PlaybackRateButton | 
					
						
							|  |  |  |           i18n={i18n} | 
					
						
							|  |  |  |           variant={`message-${direction}`} | 
					
						
							|  |  |  |           playbackRate={active?.playbackRate} | 
					
						
							| 
									
										
										
										
											2022-10-03 17:43:44 -06:00
										 |  |  |           visible={isPlaying && (!played || !isPlayedDotVisible)} | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |           onClick={() => { | 
					
						
							|  |  |  |             if (active) { | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |               setPlaybackRate( | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |                 PlaybackRateButton.nextPlaybackRate(active.playbackRate) | 
					
						
							| 
									
										
										
										
											2022-09-15 14:10:46 -06:00
										 |  |  |               ); | 
					
						
							| 
									
										
										
										
											2022-09-19 18:28:10 -06:00
										 |  |  |             } | 
					
						
							|  |  |  |           }} | 
					
						
							| 
									
										
										
										
											2023-02-24 16:18:57 -07:00
										 |  |  |         /> | 
					
						
							| 
									
										
										
										
											2022-08-31 14:42:09 -06:00
										 |  |  |       </div> | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-08 08:32:42 -06:00
										 |  |  |       {!withContentBelow && !collapseMetadata && ( | 
					
						
							| 
									
										
										
										
											2022-08-31 14:42:09 -06:00
										 |  |  |         <MessageMetadata | 
					
						
							|  |  |  |           direction={direction} | 
					
						
							|  |  |  |           expirationLength={expirationLength} | 
					
						
							|  |  |  |           expirationTimestamp={expirationTimestamp} | 
					
						
							|  |  |  |           hasText={withContentBelow} | 
					
						
							|  |  |  |           i18n={i18n} | 
					
						
							|  |  |  |           id={id} | 
					
						
							|  |  |  |           isShowingImage={false} | 
					
						
							|  |  |  |           isSticker={false} | 
					
						
							| 
									
										
										
										
											2022-12-20 19:25:10 -08:00
										 |  |  |           pushPanelForConversation={pushPanelForConversation} | 
					
						
							| 
									
										
										
										
											2023-04-20 12:31:59 -04:00
										 |  |  |           retryMessageSend={shouldNeverBeCalled} | 
					
						
							| 
									
										
										
										
											2022-08-31 14:42:09 -06:00
										 |  |  |           status={status} | 
					
						
							|  |  |  |           textPending={textPending} | 
					
						
							|  |  |  |           timestamp={timestamp} | 
					
						
							|  |  |  |         /> | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |       )} | 
					
						
							|  |  |  |     </div> | 
					
						
							|  |  |  |   ); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |   return ( | 
					
						
							|  |  |  |     <div | 
					
						
							|  |  |  |       className={classNames( | 
					
						
							|  |  |  |         CSS_BASE, | 
					
						
							|  |  |  |         `${CSS_BASE}--${direction}`, | 
					
						
							|  |  |  |         withContentBelow ? `${CSS_BASE}--with-content-below` : null, | 
					
						
							|  |  |  |         withContentAbove ? `${CSS_BASE}--with-content-above` : null | 
					
						
							|  |  |  |       )} | 
					
						
							|  |  |  |     > | 
					
						
							| 
									
										
										
										
											2021-07-09 15:27:16 -05:00
										 |  |  |       <div className={`${CSS_BASE}__button-and-waveform`}> | 
					
						
							|  |  |  |         {button} | 
					
						
							|  |  |  |         {waveform} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |       {metadata} | 
					
						
							| 
									
										
										
										
											2021-03-10 12:36:58 -08:00
										 |  |  |     </div> | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-11-17 16:45:19 -08:00
										 |  |  | } |