2023-03-02 20:55:40 +00:00
|
|
|
// Copyright 2023 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
|
|
|
import React, { useState, useCallback, useRef } from 'react';
|
|
|
|
import { useComputePeaks } from '../hooks/useComputePeaks';
|
|
|
|
import type { LocalizerType } from '../types/Util';
|
|
|
|
import { WaveformScrubber } from './conversation/WaveformScrubber';
|
|
|
|
import { PlaybackButton } from './PlaybackButton';
|
|
|
|
import { RecordingComposer } from './RecordingComposer';
|
|
|
|
import * as log from '../logging/log';
|
2023-07-25 23:56:56 +00:00
|
|
|
import type { Size } from '../hooks/useSizeObserver';
|
|
|
|
import { SizeObserver } from '../hooks/useSizeObserver';
|
2023-03-02 20:55:40 +00:00
|
|
|
|
|
|
|
type Props = {
|
|
|
|
i18n: LocalizerType;
|
|
|
|
audioUrl: string | undefined;
|
|
|
|
active:
|
|
|
|
| {
|
|
|
|
playing: boolean;
|
|
|
|
duration: number | undefined;
|
|
|
|
currentTime: number;
|
|
|
|
}
|
|
|
|
| undefined;
|
|
|
|
onCancel: () => void;
|
|
|
|
onSend: () => void;
|
|
|
|
onPlay: (positionAsRatio?: number) => void;
|
|
|
|
onPause: () => void;
|
|
|
|
onScrub: (positionAsRatio: number) => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
export function CompositionRecordingDraft({
|
|
|
|
i18n,
|
|
|
|
audioUrl,
|
|
|
|
active,
|
|
|
|
onCancel,
|
|
|
|
onSend,
|
|
|
|
onPlay,
|
|
|
|
onPause,
|
|
|
|
onScrub,
|
|
|
|
}: Props): JSX.Element {
|
|
|
|
const [state, setState] = useState<{
|
|
|
|
calculatingWidth: boolean;
|
|
|
|
width: undefined | number;
|
|
|
|
}>({ calculatingWidth: false, width: undefined });
|
|
|
|
|
|
|
|
const timeout = useRef<undefined | NodeJS.Timeout>(undefined);
|
|
|
|
|
|
|
|
const handleResize = useCallback(
|
2023-07-25 23:56:56 +00:00
|
|
|
(size: Size) => {
|
|
|
|
if (size.width === state.width) {
|
2023-03-02 20:55:40 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!state.calculatingWidth) {
|
|
|
|
setState({ ...state, calculatingWidth: true });
|
|
|
|
}
|
|
|
|
|
|
|
|
if (timeout.current) {
|
|
|
|
clearTimeout(timeout.current);
|
|
|
|
}
|
|
|
|
|
2023-07-25 23:56:56 +00:00
|
|
|
const newWidth = size.width;
|
2023-03-02 20:55:40 +00:00
|
|
|
|
|
|
|
// if mounting, set width immediately
|
|
|
|
// otherwise debounce
|
|
|
|
if (state.width === undefined) {
|
|
|
|
setState({ calculatingWidth: false, width: newWidth });
|
|
|
|
} else {
|
|
|
|
timeout.current = setTimeout(() => {
|
|
|
|
setState({ calculatingWidth: false, width: newWidth });
|
|
|
|
}, 500);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
[state]
|
|
|
|
);
|
|
|
|
|
|
|
|
const handlePlaybackClick = useCallback(() => {
|
|
|
|
if (active?.playing) {
|
|
|
|
onPause();
|
|
|
|
} else {
|
|
|
|
onPlay();
|
|
|
|
}
|
|
|
|
}, [active, onPause, onPlay]);
|
|
|
|
|
|
|
|
const scrubber = (
|
|
|
|
<SizedWaveformScrubber
|
|
|
|
i18n={i18n}
|
|
|
|
audioUrl={audioUrl}
|
|
|
|
activeDuration={active?.duration}
|
|
|
|
currentTime={active?.currentTime ?? 0}
|
|
|
|
width={state.width}
|
|
|
|
onClick={onScrub}
|
|
|
|
onScrub={onScrub}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<RecordingComposer i18n={i18n} onCancel={onCancel} onSend={onSend}>
|
|
|
|
<PlaybackButton
|
|
|
|
variant="draft"
|
|
|
|
mod={active?.playing ? 'pause' : 'play'}
|
|
|
|
label={
|
|
|
|
active?.playing
|
2023-03-30 00:03:25 +00:00
|
|
|
? i18n('icu:MessageAudio--pause')
|
|
|
|
: i18n('icu:MessageAudio--play')
|
2023-03-02 20:55:40 +00:00
|
|
|
}
|
|
|
|
onClick={handlePlaybackClick}
|
|
|
|
/>
|
2023-07-25 23:56:56 +00:00
|
|
|
<SizeObserver onSizeChange={handleResize}>
|
|
|
|
{ref => (
|
|
|
|
<div ref={ref} className="CompositionRecordingDraft__sizer">
|
2023-03-02 20:55:40 +00:00
|
|
|
{scrubber}
|
|
|
|
</div>
|
|
|
|
)}
|
2023-07-25 23:56:56 +00:00
|
|
|
</SizeObserver>
|
2023-03-02 20:55:40 +00:00
|
|
|
</RecordingComposer>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
type SizedWaveformScrubberProps = {
|
|
|
|
i18n: LocalizerType;
|
|
|
|
audioUrl: string | undefined;
|
|
|
|
// undefined if we don't have a size yet
|
|
|
|
width: number | undefined;
|
|
|
|
// defined if we are playing
|
|
|
|
activeDuration: number | undefined;
|
|
|
|
currentTime: number;
|
|
|
|
onScrub: (progressAsRatio: number) => void;
|
|
|
|
onClick: (progressAsRatio: number) => void;
|
|
|
|
};
|
|
|
|
function SizedWaveformScrubber({
|
|
|
|
i18n,
|
|
|
|
audioUrl,
|
|
|
|
activeDuration,
|
|
|
|
currentTime,
|
|
|
|
onClick,
|
|
|
|
onScrub,
|
|
|
|
width,
|
|
|
|
}: SizedWaveformScrubberProps) {
|
2023-03-06 17:30:03 +00:00
|
|
|
const handleCorrupted = useCallback(() => {
|
2023-03-02 20:55:40 +00:00
|
|
|
log.warn('SizedWaveformScrubber: audio corrupted');
|
2023-03-06 17:30:03 +00:00
|
|
|
}, []);
|
|
|
|
|
2023-03-02 20:55:40 +00:00
|
|
|
const { peaks, duration } = useComputePeaks({
|
|
|
|
audioUrl,
|
|
|
|
activeDuration,
|
|
|
|
onCorrupted: handleCorrupted,
|
|
|
|
barCount: Math.floor((width ?? 800) / 4),
|
|
|
|
});
|
|
|
|
|
|
|
|
return (
|
|
|
|
<WaveformScrubber
|
|
|
|
i18n={i18n}
|
|
|
|
peaks={peaks}
|
|
|
|
currentTime={currentTime}
|
|
|
|
barMinHeight={2}
|
|
|
|
barMaxHeight={20}
|
|
|
|
duration={duration}
|
|
|
|
onClick={onClick}
|
|
|
|
onScrub={onScrub}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|