Skip to content

Commit

Permalink
finish adding push to talk
Browse files Browse the repository at this point in the history
  • Loading branch information
SupertigerDev committed Oct 19, 2024
1 parent 0bf00f4 commit f5ee700
Show file tree
Hide file tree
Showing 5 changed files with 147 additions and 37 deletions.
19 changes: 13 additions & 6 deletions src/chat-api/events/voiceEvents.ts
Original file line number Diff line number Diff line change
Expand Up @@ -4,13 +4,16 @@ import useAccount from "../store/useAccount";
import useVoiceUsers from "../store/useVoiceUsers";

export function onVoiceUserJoined(payload: RawVoice) {
const {createVoiceUser} = useVoiceUsers();
const { createVoiceUser } = useVoiceUsers();

createVoiceUser(payload);
}
export function onVoiceUserLeft(payload: {userId: string, channelId: string}) {
const {removeVoiceUser, setCurrentChannelId} = useVoiceUsers();
const {user} = useAccount();
export function onVoiceUserLeft(payload: {
userId: string;
channelId: string;
}) {
const { removeVoiceUser, setCurrentChannelId } = useVoiceUsers();
const { user } = useAccount();

if (user()?.id === payload.userId) {
setCurrentChannelId(null);
Expand All @@ -27,12 +30,16 @@ interface VoiceSignalReceivedPayload {

export function onVoiceSignalReceived(payload: VoiceSignalReceivedPayload) {
const voiceUsers = useVoiceUsers();
const voiceUser = voiceUsers.getVoiceUser(payload.channelId, payload.fromUserId);

const voiceUser = voiceUsers.getVoiceUser(
payload.channelId,
payload.fromUserId
);
if (!voiceUser) return;

if (!voiceUser.peer) {
return voiceUsers.createPeer(voiceUser, payload.signal);
}

voiceUsers.signal(voiceUser, payload.signal);
}
}
86 changes: 78 additions & 8 deletions src/chat-api/store/useVoiceUsers.ts
Original file line number Diff line number Diff line change
@@ -1,16 +1,23 @@
import { createStore, reconcile } from "solid-js/store";
import { RawVoice } from "../RawData";
import { batch, createEffect, createSignal } from "solid-js";
import { batch, createEffect, createMemo, createSignal, on } from "solid-js";
import { getCachedCredentials } from "../services/VoiceService";
import { emitVoiceSignal } from "../emits/voiceEmits";

import type SimplePeer from "@thaunknown/simple-peer";
import useUsers, { User } from "./useUsers";
import LazySimplePeer from "@/components/LazySimplePeer";
import { getStorageString, StorageKeys } from "@/common/localStorage";
import {
getStorageObject,
getStorageString,
StorageKeys,
useVoiceInputMode,
} from "@/common/localStorage";
import useAccount from "./useAccount";
import { set } from "idb-keyval";
import vad from "voice-activity-detection";
import { downKeys, useGlobalKey } from "@/common/GlobalKey";
import { arrayEquals } from "@/common/arrayEquals";

const createIceServers = () => [
getCachedCredentials(),
Expand Down Expand Up @@ -73,6 +80,49 @@ const [currentVoiceUser, setCurrentVoiceUser] = createSignal<
CurrentVoiceUser | undefined
>(undefined);

const { start, stop } = useGlobalKey();
const [voiceMode] = useVoiceInputMode();

createEffect(
on(currentVoiceUser, (current) => {
stop();
if (!current?.channelId) return;
if (voiceMode() !== "PTT") return;
start();
})
);

const micTrack = createMemo(() => {
const current = currentVoiceUser();
return current?.audioStream?.getAudioTracks()[0];
});

createEffect(
on(
() => downKeys.length,
() => {
const bound = getStorageObject(StorageKeys.PTTBoundKeys, []);
if (!bound.length) return;
const mic = micTrack();
if (!mic) return;
const current = currentVoiceUser();
if (!current) return;

if (!arrayEquals(downKeys, bound)) {
mic.enabled = false;
setVoiceUsers(current.channelId, useAccount().user()?.id!, {
voiceActivity: false,
});
return;
}
mic.enabled = true;
setVoiceUsers(current.channelId, useAccount().user()?.id!, {
voiceActivity: true,
});
}
)
);

const setCurrentChannelId = (channelId: string | null) => {
const current = currentVoiceUser();
if (current?.channelId) {
Expand Down Expand Up @@ -328,8 +378,10 @@ function createVadInstance(
}
: {
minNoiseLevel: 0,
noiseCaptureDuration: 0,
noiseCaptureDuration: 100,
avgNoiseMultiplier: 0.1,
maxNoiseLevel: 0.01,
onUpdate: console.log,
}),

onVoiceStart: function () {
Expand Down Expand Up @@ -413,12 +465,30 @@ const toggleMic = async () => {
audio: !deviceId ? true : { deviceId: JSON.parse(deviceId) },
video: false,
});
const vadStream = await navigator.mediaDevices.getUserMedia({
audio: !deviceId ? true : { deviceId: JSON.parse(deviceId) },
video: false,
});

let vadStream: MediaStream | undefined;
let vadInstance: ReturnType<typeof vad> | undefined;

if (voiceMode() === "OPEN") {
setVoiceUsers(current.channelId, useAccount().user()?.id!, {
voiceActivity: true,
});
}

if (voiceMode() !== "OPEN") {
stream.getAudioTracks()[0]!.enabled = false;
}

if (voiceMode() === "VOICE_ACTIVITY") {
vadStream = await navigator.mediaDevices.getUserMedia({
audio: !deviceId ? true : { deviceId: JSON.parse(deviceId) },
video: false,
});
vadInstance = createVadInstance(vadStream, stream);
}

addStreamToPeers(stream);
const vadInstance = createVadInstance(vadStream, stream);

setCurrentVoiceUser({
...current,
audioStream: stream,
Expand Down
6 changes: 6 additions & 0 deletions src/common/arrayEquals.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
export function arrayEquals<A extends unknown[], B extends unknown[]>(
a: A,
b: B
) {
return a.length === b.length && a.every((v, i) => v === b[i]);
}
12 changes: 11 additions & 1 deletion src/components/settings/CallSettings.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -136,12 +136,22 @@ const InputModeRadioBoxContainer = styled(FlexColumn)`

function InputMode() {
const [inputMode, setInputMode] = useVoiceInputMode();
const store = useStore();

const isInCall = () => store.voiceUsers.currentUser()?.channelId;

return (
<div>
<SettingsBlock icon="steppers" label="Input Mode" header />
<InputModeRadioBoxContainer>
<InputModeRadioBoxContainer
onClick={() => {
if (isInCall()) {
alert("You must leave the call first.");
}
}}
>
<RadioBox
style={isInCall() ? { "pointer-events": "none" } : {}}
initialId={inputMode()}
onChange={(e) => setInputMode(e.id)}
items={[
Expand Down
61 changes: 39 additions & 22 deletions src/components/ui/RadioBox.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { styled } from "solid-styled-components";
import { FlexColumn, FlexRow } from "./Flexbox";
import Text from "./Text";
import { For, createEffect, createSignal } from "solid-js";
import { For, JSX, createEffect, createSignal } from "solid-js";
import { classNames, conditionalClass } from "@/common/classNames";

export interface RadioBoxItem {
Expand All @@ -11,12 +11,12 @@ export interface RadioBoxItem {

interface RadioBoxProps {
onChange?(item: RadioBoxItem): void;
items: RadioBoxItem[]
items: RadioBoxItem[];
initialId: string | number;
style?: JSX.CSSProperties;
}

const RadioBoxContainer = styled(FlexColumn)`
`;
const RadioBoxContainer = styled(FlexColumn)``;

export function RadioBox(props: RadioBoxProps) {
const [selectedId, setSelectedId] = createSignal(props.initialId);
Expand All @@ -32,28 +32,33 @@ export function RadioBox(props: RadioBoxProps) {
};

return (
<RadioBoxContainer>
<RadioBoxContainer style={props.style}>
<For each={props.items}>
{item => <RadioBoxItem onClick={() => onClick(item)} item={item} selected={item.id === selectedId()} />}
{(item) => (
<RadioBoxItem
onClick={() => onClick(item)}
item={item}
selected={item.id === selectedId()}
/>
)}
</For>
</RadioBoxContainer>
);
}


export const RadioBoxItemCheckBox = styled(FlexRow)<{size?: number}>`
export const RadioBoxItemCheckBox = styled(FlexRow)<{ size?: number }>`
position: relative;
width: ${props => props.size || 10}px;
height: ${props => props.size || 10}px;
width: ${(props) => props.size || 10}px;
height: ${(props) => props.size || 10}px;
border-radius: 50%;
background-color: rgba(255, 255, 255, 0.1);
transition: 0.2s;
border: solid ${props => (props.size || 10) / 2}px transparent;
border: solid ${(props) => (props.size || 10) / 2}px transparent;
&:after {
position: absolute;
content: "";
inset: -${props => (props.size || 10) / 2}px;
inset: -${(props) => (props.size || 10) / 2}px;
border: solid 1px rgba(255, 255, 255, 0.2);
border-radius: 50%;
}
Expand All @@ -63,10 +68,6 @@ export const RadioBoxItemCheckBox = styled(FlexRow)<{size?: number}>`
border-color: var(--primary-color);
}
flex-shrink: 0;
`;

const RadioBoxItemContainer = styled(FlexRow)`
Expand All @@ -84,8 +85,8 @@ const RadioBoxItemContainer = styled(FlexRow)`
&:not(.selected):hover {
.radio-box-circle {
background-color: rgba(255,255,255,0.6);
border-color: rgba(0,0,0,0.4);
background-color: rgba(255, 255, 255, 0.6);
border-color: rgba(0, 0, 0, 0.4);
}
}
`;
Expand All @@ -101,9 +102,25 @@ interface RadioBoxItemProps {

export function RadioBoxItem(props: RadioBoxItemProps) {
return (
<RadioBoxItemContainer class={classNames(props.class, conditionalClass(props.selected, "selected"))} gap={5} onClick={props.onClick}>
<RadioBoxItemCheckBox size={props.checkboxSize} class={classNames("radio-box-circle", conditionalClass(props.selected, "selected"))} classList={{selected: props.selected}} />
<Text class="label" size={props.labelSize}>{props.item.label}</Text>
<RadioBoxItemContainer
class={classNames(
props.class,
conditionalClass(props.selected, "selected")
)}
gap={5}
onClick={props.onClick}
>
<RadioBoxItemCheckBox
size={props.checkboxSize}
class={classNames(
"radio-box-circle",
conditionalClass(props.selected, "selected")
)}
classList={{ selected: props.selected }}
/>
<Text class="label" size={props.labelSize}>
{props.item.label}
</Text>
</RadioBoxItemContainer>
);
}
}

0 comments on commit f5ee700

Please sign in to comment.