Skip to content

Commit

Permalink
Merge pull request #327 from vector-im/dbkr/end_talk_sound
Browse files Browse the repository at this point in the history
Add sound when speaker stops speaking
  • Loading branch information
dbkr authored May 16, 2022
2 parents 24a1091 + 9fd7329 commit c13040f
Show file tree
Hide file tree
Showing 8 changed files with 31 additions and 11 deletions.
10 changes: 8 additions & 2 deletions src/room/PTTCallView.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -109,8 +109,13 @@ export const PTTCallView: React.FC<Props> = ({

const { audioOutput } = useMediaHandler();

const { startTalkingLocalRef, startTalkingRemoteRef, blockedRef, playClip } =
usePTTSounds();
const {
startTalkingLocalRef,
startTalkingRemoteRef,
blockedRef,
endTalkingRef,
playClip,
} = usePTTSounds();

const {
pttButtonHeld,
Expand Down Expand Up @@ -146,6 +151,7 @@ export const PTTCallView: React.FC<Props> = ({
<PTTClips
startTalkingLocalRef={startTalkingLocalRef}
startTalkingRemoteRef={startTalkingRemoteRef}
endTalkingRef={endTalkingRef}
blockedRef={blockedRef}
/>
<Header className={styles.header}>
Expand Down
15 changes: 7 additions & 8 deletions src/room/usePTT.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,17 +27,14 @@ import { PlayClipFunction, PTTClipID } from "../sound/usePttSounds";
function getActiveSpeakerFeed(
feeds: CallFeed[],
groupCall: GroupCall
): CallFeed {
): CallFeed | null {
const activeSpeakerFeeds = feeds.filter((f) => !f.isAudioMuted());

let activeSpeakerFeed;
let highestPowerLevel;
let activeSpeakerFeed = null;
let highestPowerLevel = null;
for (const feed of activeSpeakerFeeds) {
const member = groupCall.room.getMember(feed.userId);
if (
highestPowerLevel === undefined ||
member.powerLevel > highestPowerLevel
) {
if (highestPowerLevel === null || member.powerLevel > highestPowerLevel) {
highestPowerLevel = member.powerLevel;
activeSpeakerFeed = feed;
}
Expand Down Expand Up @@ -110,12 +107,14 @@ export const usePTT = (
const activeSpeakerFeed = getActiveSpeakerFeed(userMediaFeeds, groupCall);

let blocked = false;
if (activeSpeakerUserId === null && activeSpeakerFeed.userId !== null) {
if (activeSpeakerUserId === null && activeSpeakerFeed !== null) {
if (activeSpeakerFeed.userId === client.getUserId()) {
playClip(PTTClipID.START_TALKING_LOCAL);
} else {
playClip(PTTClipID.START_TALKING_REMOTE);
}
} else if (activeSpeakerUserId !== null && activeSpeakerFeed === null) {
playClip(PTTClipID.END_TALKING);
} else if (
pttButtonHeld &&
activeSpeakerUserId === client.getUserId() &&
Expand Down
8 changes: 8 additions & 0 deletions src/sound/PTTClips.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -20,19 +20,23 @@ import startTalkLocalOggUrl from "./start_talk_local.ogg";
import startTalkLocalMp3Url from "./start_talk_local.mp3";
import startTalkRemoteOggUrl from "./start_talk_remote.ogg";
import startTalkRemoteMp3Url from "./start_talk_remote.mp3";
import endTalkOggUrl from "./end_talk.ogg";
import endTalkMp3Url from "./end_talk.mp3";
import blockedOggUrl from "./blocked.ogg";
import blockedMp3Url from "./blocked.mp3";
import styles from "./PTTClips.module.css";

interface Props {
startTalkingLocalRef: React.RefObject<HTMLAudioElement>;
startTalkingRemoteRef: React.RefObject<HTMLAudioElement>;
endTalkingRef: React.RefObject<HTMLAudioElement>;
blockedRef: React.RefObject<HTMLAudioElement>;
}

export const PTTClips: React.FC<Props> = ({
startTalkingLocalRef,
startTalkingRemoteRef,
endTalkingRef,
blockedRef,
}) => {
return (
Expand All @@ -53,6 +57,10 @@ export const PTTClips: React.FC<Props> = ({
<source type="audio/ogg" src={startTalkRemoteOggUrl} />
<source type="audio/mpeg" src={startTalkRemoteMp3Url} />
</audio>
<audio preload="true" className={styles.pttClip} ref={endTalkingRef}>
<source type="audio/ogg" src={endTalkOggUrl} />
<source type="audio/mpeg" src={endTalkMp3Url} />
</audio>
<audio preload="true" className={styles.pttClip} ref={blockedRef}>
<source type="audio/ogg" src={blockedOggUrl} />
<source type="audio/mpeg" src={blockedMp3Url} />
Expand Down
Binary file modified src/sound/blocked.mp3
Binary file not shown.
Binary file modified src/sound/blocked.ogg
Binary file not shown.
Binary file added src/sound/end_talk.mp3
Binary file not shown.
Binary file added src/sound/end_talk.ogg
Binary file not shown.
9 changes: 8 additions & 1 deletion src/sound/usePttSounds.ts
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ import React, { useCallback, useState } from "react";
export enum PTTClipID {
START_TALKING_LOCAL,
START_TALKING_REMOTE,
END_TALKING,
BLOCKED,
}

Expand All @@ -27,13 +28,15 @@ export type PlayClipFunction = (clipID: PTTClipID) => void;
interface PTTSounds {
startTalkingLocalRef: React.RefObject<HTMLAudioElement>;
startTalkingRemoteRef: React.RefObject<HTMLAudioElement>;
endTalkingRef: React.RefObject<HTMLAudioElement>;
blockedRef: React.RefObject<HTMLAudioElement>;
playClip: PlayClipFunction;
}

export const usePTTSounds = (): PTTSounds => {
const [startTalkingLocalRef] = useState(React.createRef<HTMLAudioElement>());
const [startTalkingRemoteRef] = useState(React.createRef<HTMLAudioElement>());
const [endTalkingRef] = useState(React.createRef<HTMLAudioElement>());
const [blockedRef] = useState(React.createRef<HTMLAudioElement>());

const playClip = useCallback(
Expand All @@ -47,6 +50,9 @@ export const usePTTSounds = (): PTTSounds => {
case PTTClipID.START_TALKING_REMOTE:
ref = startTalkingRemoteRef;
break;
case PTTClipID.END_TALKING:
ref = endTalkingRef;
break;
case PTTClipID.BLOCKED:
ref = blockedRef;
break;
Expand All @@ -58,12 +64,13 @@ export const usePTTSounds = (): PTTSounds => {
console.log("No media element found");
}
},
[startTalkingLocalRef, startTalkingRemoteRef, blockedRef]
[startTalkingLocalRef, startTalkingRemoteRef, endTalkingRef, blockedRef]
);

return {
startTalkingLocalRef,
startTalkingRemoteRef,
endTalkingRef,
blockedRef,
playClip,
};
Expand Down

0 comments on commit c13040f

Please sign in to comment.