2021-06-23 09:52:33 -04:00
|
|
|
import { createContext } from "preact";
|
|
|
|
import { useContext, useEffect, useMemo, useRef, useState } from "preact/hooks";
|
2021-07-05 06:23:23 -04:00
|
|
|
|
|
|
|
import type { ProduceType, VoiceUser } from "../lib/vortex/Types";
|
|
|
|
import type VoiceClient from "../lib/vortex/VoiceClient";
|
|
|
|
|
|
|
|
import { Children } from "../types/Preact";
|
2021-06-24 09:26:18 -04:00
|
|
|
import { SoundContext } from "./Settings";
|
2021-07-05 06:23:23 -04:00
|
|
|
import { AppContext } from "./revoltjs/RevoltClient";
|
2021-06-23 09:52:33 -04:00
|
|
|
|
|
|
|
export enum VoiceStatus {
|
2021-07-05 06:25:20 -04:00
|
|
|
LOADING = 0,
|
|
|
|
UNAVAILABLE,
|
|
|
|
ERRORED,
|
|
|
|
READY = 3,
|
|
|
|
CONNECTING = 4,
|
|
|
|
AUTHENTICATING,
|
|
|
|
RTC_CONNECTING,
|
|
|
|
CONNECTED,
|
|
|
|
// RECONNECTING
|
2021-06-23 09:52:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface VoiceOperations {
|
2021-07-05 06:25:20 -04:00
|
|
|
connect: (channelId: string) => Promise<void>;
|
|
|
|
disconnect: () => void;
|
|
|
|
isProducing: (type: ProduceType) => boolean;
|
|
|
|
startProducing: (type: ProduceType) => Promise<void>;
|
|
|
|
stopProducing: (type: ProduceType) => Promise<void> | undefined;
|
2021-06-23 09:52:33 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface VoiceState {
|
2021-07-05 06:25:20 -04:00
|
|
|
roomId?: string;
|
|
|
|
status: VoiceStatus;
|
|
|
|
participants?: Readonly<Map<string, VoiceUser>>;
|
2021-06-23 09:52:33 -04:00
|
|
|
}
|
|
|
|
|
2021-07-04 21:22:33 -04:00
|
|
|
// They should be present from first render. - insert's words
|
2021-07-04 07:09:39 -04:00
|
|
|
export const VoiceContext = createContext<VoiceState>(null!);
|
|
|
|
export const VoiceOperationsContext = createContext<VoiceOperations>(null!);
|
2021-06-23 09:52:33 -04:00
|
|
|
|
|
|
|
type Props = {
|
2021-07-05 06:25:20 -04:00
|
|
|
children: Children;
|
2021-06-23 09:52:33 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
export default function Voice({ children }: Props) {
|
2021-07-05 06:25:20 -04:00
|
|
|
const revoltClient = useContext(AppContext);
|
|
|
|
const [client, setClient] = useState<VoiceClient | undefined>(undefined);
|
|
|
|
const [state, setState] = useState<VoiceState>({
|
|
|
|
status: VoiceStatus.LOADING,
|
|
|
|
participants: new Map(),
|
|
|
|
});
|
|
|
|
|
|
|
|
function setStatus(status: VoiceStatus, roomId?: string) {
|
|
|
|
setState({
|
|
|
|
status,
|
|
|
|
roomId: roomId ?? client?.roomId,
|
|
|
|
participants: client?.participants ?? new Map(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
import("../lib/vortex/VoiceClient")
|
|
|
|
.then(({ default: VoiceClient }) => {
|
|
|
|
const client = new VoiceClient();
|
|
|
|
setClient(client);
|
|
|
|
|
|
|
|
if (!client?.supported()) {
|
|
|
|
setStatus(VoiceStatus.UNAVAILABLE);
|
|
|
|
} else {
|
|
|
|
setStatus(VoiceStatus.READY);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
.catch((err) => {
|
|
|
|
console.error("Failed to load voice library!", err);
|
|
|
|
setStatus(VoiceStatus.UNAVAILABLE);
|
|
|
|
});
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const isConnecting = useRef(false);
|
|
|
|
const operations: VoiceOperations = useMemo(() => {
|
|
|
|
return {
|
|
|
|
connect: async (channelId) => {
|
|
|
|
if (!client?.supported()) throw new Error("RTC is unavailable");
|
|
|
|
|
|
|
|
isConnecting.current = true;
|
|
|
|
setStatus(VoiceStatus.CONNECTING, channelId);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const call = await revoltClient.channels.joinCall(
|
|
|
|
channelId,
|
|
|
|
);
|
|
|
|
|
|
|
|
if (!isConnecting.current) {
|
|
|
|
setStatus(VoiceStatus.READY);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ! FIXME: use configuration to check if voso is enabled
|
|
|
|
// await client.connect("wss://voso.revolt.chat/ws");
|
|
|
|
await client.connect(
|
|
|
|
"wss://voso.revolt.chat/ws",
|
|
|
|
channelId,
|
|
|
|
);
|
|
|
|
|
|
|
|
setStatus(VoiceStatus.AUTHENTICATING);
|
|
|
|
|
|
|
|
await client.authenticate(call.token);
|
|
|
|
setStatus(VoiceStatus.RTC_CONNECTING);
|
|
|
|
|
|
|
|
await client.initializeTransports();
|
|
|
|
} catch (error) {
|
|
|
|
console.error(error);
|
|
|
|
setStatus(VoiceStatus.READY);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
setStatus(VoiceStatus.CONNECTED);
|
|
|
|
isConnecting.current = false;
|
|
|
|
},
|
|
|
|
disconnect: () => {
|
|
|
|
if (!client?.supported()) throw new Error("RTC is unavailable");
|
|
|
|
|
|
|
|
// if (status <= VoiceStatus.READY) return;
|
|
|
|
// this will not update in this context
|
|
|
|
|
|
|
|
isConnecting.current = false;
|
|
|
|
client.disconnect();
|
|
|
|
setStatus(VoiceStatus.READY);
|
|
|
|
},
|
|
|
|
isProducing: (type: ProduceType) => {
|
|
|
|
switch (type) {
|
|
|
|
case "audio":
|
|
|
|
return client?.audioProducer !== undefined;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
startProducing: async (type: ProduceType) => {
|
|
|
|
switch (type) {
|
|
|
|
case "audio": {
|
|
|
|
if (client?.audioProducer !== undefined)
|
|
|
|
return console.log("No audio producer."); // ! FIXME: let the user know
|
|
|
|
if (navigator.mediaDevices === undefined)
|
|
|
|
return console.log("No media devices."); // ! FIXME: let the user know
|
|
|
|
const mediaStream =
|
|
|
|
await navigator.mediaDevices.getUserMedia({
|
|
|
|
audio: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
await client?.startProduce(
|
|
|
|
mediaStream.getAudioTracks()[0],
|
|
|
|
"audio",
|
|
|
|
);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
stopProducing: (type: ProduceType) => {
|
|
|
|
return client?.stopProduce(type);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}, [client]);
|
|
|
|
|
|
|
|
const playSound = useContext(SoundContext);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!client?.supported()) return;
|
|
|
|
|
|
|
|
// ! FIXME: message for fatal:
|
|
|
|
// ! get rid of these force updates
|
|
|
|
// ! handle it through state or smth
|
|
|
|
|
2021-07-27 06:45:45 -04:00
|
|
|
function stateUpdate() {
|
|
|
|
setStatus(state.status);
|
|
|
|
}
|
|
|
|
|
|
|
|
client.on("startProduce", stateUpdate);
|
|
|
|
client.on("stopProduce", stateUpdate);
|
2021-07-05 06:25:20 -04:00
|
|
|
|
|
|
|
client.on("userJoined", () => {
|
|
|
|
playSound("call_join");
|
2021-07-27 06:45:45 -04:00
|
|
|
stateUpdate();
|
2021-07-05 06:25:20 -04:00
|
|
|
});
|
|
|
|
client.on("userLeft", () => {
|
|
|
|
playSound("call_leave");
|
2021-07-27 06:45:45 -04:00
|
|
|
stateUpdate();
|
2021-07-05 06:25:20 -04:00
|
|
|
});
|
2021-07-27 06:45:45 -04:00
|
|
|
|
|
|
|
client.on("userStartProduce", stateUpdate);
|
|
|
|
client.on("userStopProduce", stateUpdate);
|
|
|
|
|
|
|
|
client.on("close", stateUpdate);
|
2021-07-05 06:25:20 -04:00
|
|
|
|
|
|
|
return () => {
|
2021-07-27 06:45:45 -04:00
|
|
|
client.removeListener("startProduce", stateUpdate);
|
|
|
|
client.removeListener("stopProduce", stateUpdate);
|
|
|
|
|
|
|
|
client.removeListener("userJoined", stateUpdate);
|
|
|
|
client.removeListener("userLeft", stateUpdate);
|
|
|
|
client.removeListener("userStartProduce", stateUpdate);
|
|
|
|
client.removeListener("userStopProduce", stateUpdate);
|
|
|
|
client.removeListener("close", stateUpdate);
|
2021-07-05 06:25:20 -04:00
|
|
|
};
|
|
|
|
}, [client, state]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<VoiceContext.Provider value={state}>
|
|
|
|
<VoiceOperationsContext.Provider value={operations}>
|
|
|
|
{children}
|
|
|
|
</VoiceOperationsContext.Provider>
|
|
|
|
</VoiceContext.Provider>
|
|
|
|
);
|
2021-06-23 09:52:33 -04:00
|
|
|
}
|