* Refactor media devices to live outside React as Observables This moves the media devices state out of React to further our transition to a MVVM architecture in which we can more easily model and store complex application state. I have created an AppViewModel to act as the overarching state holder for any future non-React state we end up creating, and the MediaDevices reside within this. We should move more application logic (including the CallViewModel itself) there in the future. * Address review feedback * Fixes from ios debugging session: (#3342) - dont use preferred vs selected concept in controlled media. Its not needed since we dont use the id for actual browser media devices (the id's are not even actual browser media devices) - add more logging - add more conditions to not accidently set a deviceId that is not a browser deviceId but one provided via controlled. --------- Co-authored-by: Timo <16718859+toger5@users.noreply.github.com>
237 lines
7.2 KiB
TypeScript
237 lines
7.2 KiB
TypeScript
/*
|
|
Copyright 2022-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 FC, type ReactNode, useEffect, useState } from "react";
|
|
import { useTranslation } from "react-i18next";
|
|
import { type MatrixClient } from "matrix-js-sdk";
|
|
import { Button, Root as Form, Separator } from "@vector-im/compound-web";
|
|
import { type Room as LivekitRoom } from "livekit-client";
|
|
import { useObservableEagerState } from "observable-hooks";
|
|
|
|
import { Modal } from "../Modal";
|
|
import styles from "./SettingsModal.module.css";
|
|
import { type Tab, TabContainer } from "../tabs/Tabs";
|
|
import { ProfileSettingsTab } from "./ProfileSettingsTab";
|
|
import { FeedbackSettingsTab } from "./FeedbackSettingsTab";
|
|
import { iosDeviceMenu$ } from "../state/MediaDevices";
|
|
import { useMediaDevices } from "../MediaDevicesContext";
|
|
import { widget } from "../widget";
|
|
import {
|
|
useSetting,
|
|
soundEffectVolume as soundEffectVolumeSetting,
|
|
backgroundBlur as backgroundBlurSetting,
|
|
developerMode,
|
|
} from "./settings";
|
|
import { PreferencesSettingsTab } from "./PreferencesSettingsTab";
|
|
import { Slider } from "../Slider";
|
|
import { DeviceSelection } from "./DeviceSelection";
|
|
import { useTrackProcessor } from "../livekit/TrackProcessorContext";
|
|
import { DeveloperSettingsTab } from "./DeveloperSettingsTab";
|
|
import { FieldRow, InputField } from "../input/Input";
|
|
import { useSubmitRageshake } from "./submit-rageshake";
|
|
import { useUrlParams } from "../UrlParams";
|
|
|
|
type SettingsTab =
|
|
| "audio"
|
|
| "video"
|
|
| "profile"
|
|
| "preferences"
|
|
| "feedback"
|
|
| "more"
|
|
| "developer";
|
|
|
|
interface Props {
|
|
open: boolean;
|
|
onDismiss: () => void;
|
|
tab: SettingsTab;
|
|
onTabChange: (tab: SettingsTab) => void;
|
|
client: MatrixClient;
|
|
roomId?: string;
|
|
livekitRoom?: LivekitRoom;
|
|
}
|
|
|
|
export const defaultSettingsTab: SettingsTab = "audio";
|
|
|
|
export const SettingsModal: FC<Props> = ({
|
|
open,
|
|
onDismiss,
|
|
tab,
|
|
onTabChange,
|
|
client,
|
|
roomId,
|
|
livekitRoom,
|
|
}) => {
|
|
const { t } = useTranslation();
|
|
|
|
// Generate a `Checkbox` input to turn blur on or off.
|
|
const BlurCheckbox: React.FC = (): ReactNode => {
|
|
const { supported } = useTrackProcessor();
|
|
|
|
const [blurActive, setBlurActive] = useSetting(backgroundBlurSetting);
|
|
|
|
return (
|
|
<>
|
|
<h4>{t("settings.background_blur_header")}</h4>
|
|
|
|
<FieldRow>
|
|
<InputField
|
|
id="activateBackgroundBlur"
|
|
label={t("settings.background_blur_label")}
|
|
description={
|
|
supported ? "" : t("settings.blur_not_supported_by_browser")
|
|
}
|
|
type="checkbox"
|
|
checked={!!blurActive}
|
|
onChange={(b): void => setBlurActive(b.target.checked)}
|
|
disabled={!supported}
|
|
/>
|
|
</FieldRow>
|
|
</>
|
|
);
|
|
};
|
|
|
|
const devices = useMediaDevices();
|
|
useEffect(() => {
|
|
if (open) devices.requestDeviceNames();
|
|
}, [open, devices]);
|
|
|
|
const [soundVolume, setSoundVolume] = useSetting(soundEffectVolumeSetting);
|
|
const [soundVolumeRaw, setSoundVolumeRaw] = useState(soundVolume);
|
|
const [showDeveloperSettingsTab] = useSetting(developerMode);
|
|
|
|
const { available: isRageshakeAvailable } = useSubmitRageshake();
|
|
|
|
// For controlled devices, we will not show the input section:
|
|
// Controlled media devices are used on mobile platforms, where input and output are grouped into
|
|
// a single device. These are called "headset" or "speaker" (or similar) but contain both input and output.
|
|
// On EC, we decided that it is less confusing for the user if they see those options in the output section
|
|
// rather than the input section.
|
|
const { controlledAudioDevices } = useUrlParams();
|
|
// If we are on iOS we will show a button to open the native audio device picker.
|
|
const iosDeviceMenu = useObservableEagerState(iosDeviceMenu$);
|
|
|
|
const audioTab: Tab<SettingsTab> = {
|
|
key: "audio",
|
|
name: t("common.audio"),
|
|
content: (
|
|
<>
|
|
<Form>
|
|
{!controlledAudioDevices && (
|
|
<DeviceSelection
|
|
device={devices.audioInput}
|
|
title={t("settings.devices.microphone")}
|
|
numberedLabel={(n) =>
|
|
t("settings.devices.microphone_numbered", { n })
|
|
}
|
|
/>
|
|
)}
|
|
{iosDeviceMenu && controlledAudioDevices && (
|
|
<Button
|
|
onClick={(e): void => {
|
|
e.preventDefault();
|
|
window.controls.showNativeAudioDevicePicker?.();
|
|
// call deprecated method for backwards compatibility.
|
|
window.controls.showNativeOutputDevicePicker?.();
|
|
}}
|
|
>
|
|
{t("settings.devices.change_device_button")}
|
|
</Button>
|
|
)}
|
|
<DeviceSelection
|
|
device={devices.audioOutput}
|
|
title={t("settings.devices.speaker")}
|
|
numberedLabel={(n) => t("settings.devices.speaker_numbered", { n })}
|
|
/>
|
|
|
|
<div className={styles.volumeSlider}>
|
|
<label>{t("settings.audio_tab.effect_volume_label")}</label>
|
|
<p>{t("settings.audio_tab.effect_volume_description")}</p>
|
|
<Slider
|
|
label={t("video_tile.volume")}
|
|
value={soundVolumeRaw}
|
|
onValueChange={setSoundVolumeRaw}
|
|
onValueCommit={setSoundVolume}
|
|
min={0}
|
|
max={1}
|
|
step={0.01}
|
|
/>
|
|
</div>
|
|
</Form>
|
|
</>
|
|
),
|
|
};
|
|
|
|
const videoTab: Tab<SettingsTab> = {
|
|
key: "video",
|
|
name: t("common.video"),
|
|
content: (
|
|
<>
|
|
<Form>
|
|
<DeviceSelection
|
|
device={devices.videoInput}
|
|
title={t("settings.devices.camera")}
|
|
numberedLabel={(n) => t("settings.devices.camera_numbered", { n })}
|
|
/>
|
|
</Form>
|
|
<Separator />
|
|
<BlurCheckbox />
|
|
</>
|
|
),
|
|
};
|
|
|
|
const preferencesTab: Tab<SettingsTab> = {
|
|
key: "preferences",
|
|
name: t("common.preferences"),
|
|
content: <PreferencesSettingsTab />,
|
|
};
|
|
|
|
const profileTab: Tab<SettingsTab> = {
|
|
key: "profile",
|
|
name: t("common.profile"),
|
|
content: <ProfileSettingsTab client={client} />,
|
|
};
|
|
|
|
const feedbackTab: Tab<SettingsTab> = {
|
|
key: "feedback",
|
|
name: t("settings.feedback_tab_title"),
|
|
content: <FeedbackSettingsTab roomId={roomId} />,
|
|
};
|
|
|
|
const developerTab: Tab<SettingsTab> = {
|
|
key: "developer",
|
|
name: t("settings.developer_tab_title"),
|
|
content: <DeveloperSettingsTab client={client} livekitRoom={livekitRoom} />,
|
|
};
|
|
|
|
const tabs = [audioTab, videoTab];
|
|
if (widget === null) tabs.push(profileTab);
|
|
tabs.push(preferencesTab);
|
|
if (isRageshakeAvailable || import.meta.env.VITE_PACKAGE === "full") {
|
|
// for full package we want to show the analytics consent checkbox
|
|
// even if rageshake is not available
|
|
tabs.push(feedbackTab);
|
|
}
|
|
if (showDeveloperSettingsTab) tabs.push(developerTab);
|
|
|
|
return (
|
|
<Modal
|
|
title={t("common.settings")}
|
|
className={styles.settingsModal}
|
|
open={open}
|
|
onDismiss={onDismiss}
|
|
tabbed
|
|
>
|
|
<TabContainer
|
|
label={t("common.settings")}
|
|
tab={tab}
|
|
onTabChange={onTabChange}
|
|
tabs={tabs}
|
|
/>
|
|
</Modal>
|
|
);
|
|
};
|