mirror of
https://github.com/vector-im/element-call.git
synced 2026-03-22 06:30:25 +00:00
105 lines
2.9 KiB
TypeScript
105 lines
2.9 KiB
TypeScript
/*
|
|
Copyright 2024 New Vector Ltd.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial
|
|
Please see LICENSE in the repository root for full details.
|
|
*/
|
|
|
|
import { type ReactNode, useEffect } from "react";
|
|
import { filter, interval, throttle } from "rxjs";
|
|
|
|
import { type CallViewModel } from "../state/CallViewModel";
|
|
import joinCallSoundMp3 from "../sound/join_call.mp3";
|
|
import joinCallSoundOgg from "../sound/join_call.ogg";
|
|
import leftCallSoundMp3 from "../sound/left_call.mp3";
|
|
import leftCallSoundOgg from "../sound/left_call.ogg";
|
|
import handSoundOgg from "../sound/raise_hand.ogg";
|
|
import handSoundMp3 from "../sound/raise_hand.mp3";
|
|
import screenShareStartedOgg from "../sound/screen_share_started.ogg";
|
|
import screenShareStartedMp3 from "../sound/screen_share_started.mp3";
|
|
import { useAudioContext } from "../useAudioContext";
|
|
import { prefetchSounds } from "../soundUtils";
|
|
import { useLatest } from "../useLatest";
|
|
|
|
// Do not play any sounds if the participant count has exceeded this
|
|
// number.
|
|
export const MAX_PARTICIPANT_COUNT_FOR_SOUND = 8;
|
|
export const THROTTLE_SOUND_EFFECT_MS = 500;
|
|
|
|
export const callEventAudioSounds = prefetchSounds({
|
|
join: {
|
|
mp3: joinCallSoundMp3,
|
|
ogg: joinCallSoundOgg,
|
|
},
|
|
left: {
|
|
mp3: leftCallSoundMp3,
|
|
ogg: leftCallSoundOgg,
|
|
},
|
|
raiseHand: {
|
|
mp3: handSoundMp3,
|
|
ogg: handSoundOgg,
|
|
},
|
|
screenshareStarted: {
|
|
mp3: screenShareStartedMp3,
|
|
ogg: screenShareStartedOgg,
|
|
},
|
|
});
|
|
|
|
export function CallEventAudioRenderer({
|
|
vm,
|
|
muted,
|
|
}: {
|
|
vm: CallViewModel;
|
|
muted?: boolean;
|
|
}): ReactNode {
|
|
const audioEngineCtx = useAudioContext({
|
|
sounds: callEventAudioSounds,
|
|
latencyHint: "interactive",
|
|
muted,
|
|
});
|
|
const audioEngineRef = useLatest(audioEngineCtx);
|
|
|
|
useEffect(() => {
|
|
const joinSub = vm.memberChanges$
|
|
.pipe(
|
|
filter(
|
|
({ joined, ids }) =>
|
|
ids.length <= MAX_PARTICIPANT_COUNT_FOR_SOUND && joined.length > 0,
|
|
),
|
|
throttle(() => interval(THROTTLE_SOUND_EFFECT_MS)),
|
|
)
|
|
.subscribe(() => {
|
|
void audioEngineRef.current?.playSound("join");
|
|
});
|
|
|
|
const leftSub = vm.memberChanges$
|
|
.pipe(
|
|
filter(
|
|
({ ids, left }) =>
|
|
ids.length <= MAX_PARTICIPANT_COUNT_FOR_SOUND && left.length > 0,
|
|
),
|
|
throttle(() => interval(THROTTLE_SOUND_EFFECT_MS)),
|
|
)
|
|
.subscribe(() => {
|
|
void audioEngineRef.current?.playSound("left");
|
|
});
|
|
|
|
const handRaisedSub = vm.newHandRaised$.subscribe(() => {
|
|
void audioEngineRef.current?.playSound("raiseHand");
|
|
});
|
|
|
|
const screenshareSub = vm.newScreenShare$.subscribe(() => {
|
|
void audioEngineRef.current?.playSound("screenshareStarted");
|
|
});
|
|
|
|
return (): void => {
|
|
joinSub.unsubscribe();
|
|
leftSub.unsubscribe();
|
|
handRaisedSub.unsubscribe();
|
|
screenshareSub.unsubscribe();
|
|
};
|
|
}, [audioEngineRef, vm]);
|
|
|
|
return <></>;
|
|
}
|