diff --git a/src/room/InCallView.jsx b/src/room/InCallView.jsx
index 8ecd54788..39cd59231 100644
--- a/src/room/InCallView.jsx
+++ b/src/room/InCallView.jsx
@@ -22,6 +22,7 @@ import { UserMenuContainer } from "../UserMenuContainer";
import { useRageshakeRequestModal } from "../settings/rageshake";
import { RageshakeRequestModal } from "./RageshakeRequestModal";
import { usePreventScroll } from "@react-aria/overlays";
+import { useMediaHandler } from "../settings/useMediaHandler";
const canScreenshare = "getDisplayMedia" in navigator.mediaDevices;
// There is currently a bug in Safari our our code with cloning and sending MediaStreams
@@ -51,6 +52,8 @@ export function InCallView({
usePreventScroll();
const [layout, setLayout] = useVideoGridLayout(screenshareFeeds.length > 0);
+ const { audioOutput } = useMediaHandler();
+
const items = useMemo(() => {
const participants = [];
@@ -159,6 +162,7 @@ export function InCallView({
item={item}
getAvatar={renderAvatar}
showName={items.length > 2 || item.focused}
+ audioOutputDevice={audioOutput}
{...rest}
/>
)}
diff --git a/src/room/LobbyView.jsx b/src/room/LobbyView.jsx
index fbeabcd6f..f92fe274b 100644
--- a/src/room/LobbyView.jsx
+++ b/src/room/LobbyView.jsx
@@ -14,6 +14,7 @@ import { useProfile } from "../profile/useProfile";
import useMeasure from "react-use-measure";
import { ResizeObserver } from "@juggle/resize-observer";
import { useLocationNavigation } from "../useLocationNavigation";
+import { useMediaHandler } from "../settings/useMediaHandler";
export function LobbyView({
client,
@@ -31,7 +32,8 @@ export function LobbyView({
roomId,
}) {
const { stream } = useCallFeed(localCallFeed);
- const videoRef = useMediaStream(stream, true);
+ const { audioOutput } = useMediaHandler();
+ const videoRef = useMediaStream(stream, audioOutput, true);
const { displayName, avatarUrl } = useProfile(client);
const [previewRef, previewBounds] = useMeasure({ polyfill: ResizeObserver });
const avatarSize = (previewBounds.height - 66) / 2;
diff --git a/src/room/OverflowMenu.jsx b/src/room/OverflowMenu.jsx
index 3f7608519..fd34bcd6e 100644
--- a/src/room/OverflowMenu.jsx
+++ b/src/room/OverflowMenu.jsx
@@ -75,7 +75,6 @@ export function OverflowMenu({
{...settingsModalProps}
setShowInspector={setShowInspector}
showInspector={showInspector}
- client={client}
/>
)}
{inviteModalState.isOpen && (
diff --git a/src/room/RoomPage.jsx b/src/room/RoomPage.jsx
index f421e6871..92e320e51 100644
--- a/src/room/RoomPage.jsx
+++ b/src/room/RoomPage.jsx
@@ -21,6 +21,7 @@ import { ErrorView, LoadingView } from "../FullScreenView";
import { RoomAuthView } from "./RoomAuthView";
import { GroupCallLoader } from "./GroupCallLoader";
import { GroupCallView } from "./GroupCallView";
+import { MediaHandlerProvider } from "../settings/useMediaHandler";
export function RoomPage() {
const { loading, isAuthenticated, error, client, isPasswordlessUser } =
@@ -47,16 +48,18 @@ export function RoomPage() {
}
return (
-
- {(groupCall) => (
-
- )}
-
+
+
+ {(groupCall) => (
+
+ )}
+
+
);
}
diff --git a/src/settings/SettingsModal.jsx b/src/settings/SettingsModal.jsx
index 36c33c023..2945daca4 100644
--- a/src/settings/SettingsModal.jsx
+++ b/src/settings/SettingsModal.jsx
@@ -13,12 +13,7 @@ import { Button } from "../button";
import { useDownloadDebugLog } from "./rageshake";
import { Body } from "../typography/Typography";
-export function SettingsModal({
- client,
- setShowInspector,
- showInspector,
- ...rest
-}) {
+export function SettingsModal({ setShowInspector, showInspector, ...rest }) {
const {
audioInput,
audioInputs,
@@ -26,7 +21,10 @@ export function SettingsModal({
videoInput,
videoInputs,
setVideoInput,
- } = useMediaHandler(client);
+ audioOutput,
+ audioOutputs,
+ setAudioOutput,
+ } = useMediaHandler();
const downloadDebugLog = useDownloadDebugLog();
@@ -56,6 +54,17 @@ export function SettingsModal({
- {label}
))}
+ {audioOutputs.length > 0 && (
+
+ {audioOutputs.map(({ deviceId, label }) => (
+ - {label}
+ ))}
+
+ )}
{
- const mediaHandler = client.getMediaHandler();
-
- return {
- audioInput: mediaHandler.audioInput,
- videoInput: mediaHandler.videoInput,
- audioInputs: [],
- videoInputs: [],
- };
- });
-
- useEffect(() => {
- const mediaHandler = client.getMediaHandler();
-
- function updateDevices() {
- navigator.mediaDevices.enumerateDevices().then((devices) => {
- const audioInputs = devices.filter(
- (device) => device.kind === "audioinput"
- );
- const videoInputs = devices.filter(
- (device) => device.kind === "videoinput"
- );
-
- setState(() => ({
- audioInput: mediaHandler.audioInput,
- videoInput: mediaHandler.videoInput,
- audioInputs,
- videoInputs,
- }));
- });
- }
-
- updateDevices();
-
- mediaHandler.on("local_streams_changed", updateDevices);
- navigator.mediaDevices.addEventListener("devicechange", updateDevices);
-
- return () => {
- mediaHandler.removeListener("local_streams_changed", updateDevices);
- navigator.mediaDevices.removeEventListener("devicechange", updateDevices);
- };
- }, []);
-
- const setAudioInput = useCallback(
- (deviceId) => {
- setState((prevState) => ({ ...prevState, audioInput: deviceId }));
- client.getMediaHandler().setAudioInput(deviceId);
- },
- [client]
- );
-
- const setVideoInput = useCallback(
- (deviceId) => {
- setState((prevState) => ({ ...prevState, videoInput: deviceId }));
- client.getMediaHandler().setVideoInput(deviceId);
- },
- [client]
- );
-
- return {
- audioInput,
- audioInputs,
- setAudioInput,
- videoInput,
- videoInputs,
- setVideoInput,
- };
-}
diff --git a/src/settings/useMediaHandler.jsx b/src/settings/useMediaHandler.jsx
new file mode 100644
index 000000000..1b5407365
--- /dev/null
+++ b/src/settings/useMediaHandler.jsx
@@ -0,0 +1,143 @@
+import React, {
+ useState,
+ useEffect,
+ useCallback,
+ useMemo,
+ useContext,
+ createContext,
+} from "react";
+
+const MediaHandlerContext = createContext();
+
+export function MediaHandlerProvider({ client, children }) {
+ const [
+ {
+ audioInput,
+ videoInput,
+ audioInputs,
+ videoInputs,
+ audioOutput,
+ audioOutputs,
+ },
+ setState,
+ ] = useState(() => {
+ const mediaHandler = client.getMediaHandler();
+
+ return {
+ audioInput: mediaHandler.audioInput,
+ videoInput: mediaHandler.videoInput,
+ audioOutput: undefined,
+ audioInputs: [],
+ videoInputs: [],
+ audioOutputs: [],
+ };
+ });
+
+ useEffect(() => {
+ const mediaHandler = client.getMediaHandler();
+
+ function updateDevices() {
+ navigator.mediaDevices.enumerateDevices().then((devices) => {
+ const audioInputs = devices.filter(
+ (device) => device.kind === "audioinput"
+ );
+ const videoInputs = devices.filter(
+ (device) => device.kind === "videoinput"
+ );
+ const audioOutputs = devices.filter(
+ (device) => device.kind === "audiooutput"
+ );
+
+ let audioOutput = undefined;
+
+ const audioOutputPreference = localStorage.getItem(
+ "matrix-audio-output"
+ );
+
+ if (
+ audioOutputPreference &&
+ audioOutputs.some(
+ (device) => device.deviceId === audioOutputPreference
+ )
+ ) {
+ audioOutput = audioOutputPreference;
+ }
+
+ setState({
+ audioInput: mediaHandler.audioInput,
+ videoInput: mediaHandler.videoInput,
+ audioOutput,
+ audioInputs,
+ audioOutputs,
+ videoInputs,
+ });
+ });
+ }
+
+ updateDevices();
+
+ mediaHandler.on("local_streams_changed", updateDevices);
+ navigator.mediaDevices.addEventListener("devicechange", updateDevices);
+
+ return () => {
+ mediaHandler.removeListener("local_streams_changed", updateDevices);
+ navigator.mediaDevices.removeEventListener("devicechange", updateDevices);
+ };
+ }, [client]);
+
+ const setAudioInput = useCallback(
+ (deviceId) => {
+ setState((prevState) => ({ ...prevState, audioInput: deviceId }));
+ client.getMediaHandler().setAudioInput(deviceId);
+ },
+ [client]
+ );
+
+ const setVideoInput = useCallback(
+ (deviceId) => {
+ setState((prevState) => ({ ...prevState, videoInput: deviceId }));
+ client.getMediaHandler().setVideoInput(deviceId);
+ },
+ [client]
+ );
+
+ const setAudioOutput = useCallback((deviceId) => {
+ localStorage.setItem("matrix-audio-output", deviceId);
+ setState((prevState) => ({ ...prevState, audioOutput: deviceId }));
+ }, []);
+
+ const context = useMemo(
+ () => ({
+ audioInput,
+ audioInputs,
+ setAudioInput,
+ videoInput,
+ videoInputs,
+ setVideoInput,
+ audioOutput,
+ audioOutputs,
+ setAudioOutput,
+ }),
+ [
+ audioInput,
+ audioInputs,
+ setAudioInput,
+ videoInput,
+ videoInputs,
+ setVideoInput,
+ audioOutput,
+ audioOutputs,
+ setAudioOutput,
+ ]
+ );
+
+ return (
+
+ {children}
+
+ );
+}
+
+export function useMediaHandler() {
+ return useContext(MediaHandlerContext);
+}