mirror of
https://github.com/vector-im/element-call.git
synced 2026-02-11 04:27:03 +00:00
This probably should have been part of https://github.com/element-hq/element-call/pull/2984
110 lines
2.6 KiB
TypeScript
110 lines
2.6 KiB
TypeScript
/*
|
|
Copyright 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 ChangeEvent,
|
|
type FC,
|
|
type ReactElement,
|
|
type ReactNode,
|
|
useCallback,
|
|
useId,
|
|
} from "react";
|
|
import {
|
|
Heading,
|
|
InlineField,
|
|
Label,
|
|
RadioControl,
|
|
Separator,
|
|
} from "@vector-im/compound-web";
|
|
import { Trans, useTranslation } from "react-i18next";
|
|
|
|
import { type MediaDevice } from "../livekit/MediaDevicesContext";
|
|
import styles from "./DeviceSelection.module.css";
|
|
|
|
interface Props {
|
|
devices: MediaDevice;
|
|
title: string;
|
|
numberedLabel: (number: number) => string;
|
|
}
|
|
|
|
export const DeviceSelection: FC<Props> = ({
|
|
devices,
|
|
title,
|
|
numberedLabel,
|
|
}) => {
|
|
const { t } = useTranslation();
|
|
const groupId = useId();
|
|
const onChange = useCallback(
|
|
(e: ChangeEvent<HTMLInputElement>) => {
|
|
devices.select(e.target.value);
|
|
},
|
|
[devices],
|
|
);
|
|
|
|
if (devices.available.size == 0) return null;
|
|
|
|
return (
|
|
<div className={styles.selection}>
|
|
<Heading
|
|
type="body"
|
|
weight="semibold"
|
|
size="sm"
|
|
as="h4"
|
|
className={styles.title}
|
|
>
|
|
{title}
|
|
</Heading>
|
|
<Separator className={styles.separator} />
|
|
<div className={styles.options}>
|
|
{[...devices.available].map(([id, label]) => {
|
|
let labelText: ReactNode;
|
|
switch (label.type) {
|
|
case "name":
|
|
labelText = label.name;
|
|
break;
|
|
case "number":
|
|
labelText = numberedLabel(label.number);
|
|
break;
|
|
case "default":
|
|
labelText =
|
|
label.name === null ? (
|
|
t("settings.devices.default")
|
|
) : (
|
|
<Trans
|
|
i18nKey="settings.devices.default_named"
|
|
name={label.name}
|
|
>
|
|
Default{" "}
|
|
<span className={styles.secondary}>
|
|
({{ name: label.name } as unknown as ReactElement})
|
|
</span>
|
|
</Trans>
|
|
);
|
|
break;
|
|
}
|
|
|
|
return (
|
|
<InlineField
|
|
key={id}
|
|
name={groupId}
|
|
control={
|
|
<RadioControl
|
|
checked={id === devices.selectedId}
|
|
onChange={onChange}
|
|
value={id}
|
|
/>
|
|
}
|
|
>
|
|
<Label>{labelText}</Label>
|
|
</InlineField>
|
|
);
|
|
})}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|