matrix-react-sdk
Version:
SDK for matrix.org using React
201 lines (198 loc) • 39.2 kB
JavaScript
;
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _logger = require("matrix-js-sdk/src/logger");
var _call = require("matrix-js-sdk/src/webrtc/call");
var _languageHandler = require("../../../../../languageHandler");
var _MediaDeviceHandler = _interopRequireWildcard(require("../../../../../MediaDeviceHandler"));
var _Field = _interopRequireDefault(require("../../../elements/Field"));
var _AccessibleButton = _interopRequireDefault(require("../../../elements/AccessibleButton"));
var _SettingLevel = require("../../../../../settings/SettingLevel");
var _SettingsFlag = _interopRequireDefault(require("../../../elements/SettingsFlag"));
var _LabelledToggleSwitch = _interopRequireDefault(require("../../../elements/LabelledToggleSwitch"));
var _requestMediaPermissions = require("../../../../../utils/media/requestMediaPermissions");
var _SettingsTab = _interopRequireDefault(require("../SettingsTab"));
var _SettingsSection = require("../../shared/SettingsSection");
var _SettingsSubsection = _interopRequireDefault(require("../../shared/SettingsSubsection"));
var _MatrixClientContext = _interopRequireDefault(require("../../../../../contexts/MatrixClientContext"));
function _getRequireWildcardCache(e) { if ("function" != typeof WeakMap) return null; var r = new WeakMap(), t = new WeakMap(); return (_getRequireWildcardCache = function (e) { return e ? t : r; })(e); }
function _interopRequireWildcard(e, r) { if (!r && e && e.__esModule) return e; if (null === e || "object" != typeof e && "function" != typeof e) return { default: e }; var t = _getRequireWildcardCache(r); if (t && t.has(e)) return t.get(e); var n = { __proto__: null }, a = Object.defineProperty && Object.getOwnPropertyDescriptor; for (var u in e) if ("default" !== u && {}.hasOwnProperty.call(e, u)) { var i = a ? Object.getOwnPropertyDescriptor(e, u) : null; i && (i.get || i.set) ? Object.defineProperty(n, u, i) : n[u] = e[u]; } return n.default = e, t && t.set(e, n), n; }
/*
Copyright 2024 New Vector Ltd.
Copyright 2020 The Matrix.org Foundation C.I.C.
Copyright 2019 New Vector Ltd
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
/**
* Maps deviceKind to the right get method on MediaDeviceHandler
* Helpful for setting state
*/
const mapDeviceKindToHandlerValue = deviceKind => {
switch (deviceKind) {
case _MediaDeviceHandler.MediaDeviceKindEnum.AudioOutput:
return _MediaDeviceHandler.default.getAudioOutput();
case _MediaDeviceHandler.MediaDeviceKindEnum.AudioInput:
return _MediaDeviceHandler.default.getAudioInput();
case _MediaDeviceHandler.MediaDeviceKindEnum.VideoInput:
return _MediaDeviceHandler.default.getVideoInput();
}
};
class VoiceUserSettingsTab extends _react.default.Component {
constructor(props, context) {
super(props, context);
(0, _defineProperty2.default)(this, "refreshMediaDevices", async stream => {
this.setState({
mediaDevices: (await _MediaDeviceHandler.default.getDevices()) ?? null,
[_MediaDeviceHandler.MediaDeviceKindEnum.AudioOutput]: mapDeviceKindToHandlerValue(_MediaDeviceHandler.MediaDeviceKindEnum.AudioOutput),
[_MediaDeviceHandler.MediaDeviceKindEnum.AudioInput]: mapDeviceKindToHandlerValue(_MediaDeviceHandler.MediaDeviceKindEnum.AudioInput),
[_MediaDeviceHandler.MediaDeviceKindEnum.VideoInput]: mapDeviceKindToHandlerValue(_MediaDeviceHandler.MediaDeviceKindEnum.VideoInput)
});
if (stream) {
// kill stream (after we've enumerated the devices, otherwise we'd get empty labels again)
// so that we don't leave it lingering around with webcam enabled etc
// as here we called gUM to ask user for permission to their device names only
stream.getTracks().forEach(track => track.stop());
}
});
(0, _defineProperty2.default)(this, "requestMediaPermissions", async () => {
const stream = await (0, _requestMediaPermissions.requestMediaPermissions)();
if (stream) {
await this.refreshMediaDevices(stream);
}
});
(0, _defineProperty2.default)(this, "setDevice", async (deviceId, kind) => {
// set state immediately so UI is responsive
this.setState({
[kind]: deviceId
});
try {
await _MediaDeviceHandler.default.instance.setDevice(deviceId, kind);
} catch (error) {
_logger.logger.error(`Failed to set device ${kind}: ${deviceId}`);
// reset state to current value
this.setState({
[kind]: mapDeviceKindToHandlerValue(kind)
});
}
});
(0, _defineProperty2.default)(this, "changeWebRtcMethod", p2p => {
this.context.setForceTURN(!p2p);
});
this.state = {
mediaDevices: null,
[_MediaDeviceHandler.MediaDeviceKindEnum.AudioOutput]: null,
[_MediaDeviceHandler.MediaDeviceKindEnum.AudioInput]: null,
[_MediaDeviceHandler.MediaDeviceKindEnum.VideoInput]: null,
audioAutoGainControl: _MediaDeviceHandler.default.getAudioAutoGainControl(),
audioEchoCancellation: _MediaDeviceHandler.default.getAudioEchoCancellation(),
audioNoiseSuppression: _MediaDeviceHandler.default.getAudioNoiseSuppression()
};
}
async componentDidMount() {
const canSeeDeviceLabels = await _MediaDeviceHandler.default.hasAnyLabeledDevices();
if (canSeeDeviceLabels) {
await this.refreshMediaDevices();
}
}
renderDeviceOptions(devices, category) {
return devices.map(d => {
return /*#__PURE__*/_react.default.createElement("option", {
key: `${category}-${d.deviceId}`,
value: d.deviceId
}, d.label);
});
}
renderDropdown(kind, label) {
const devices = this.state.mediaDevices?.[kind].slice(0);
if (!devices?.length) return null;
const defaultDevice = _MediaDeviceHandler.default.getDefaultDevice(devices);
return /*#__PURE__*/_react.default.createElement(_Field.default, {
element: "select",
label: label,
value: this.state[kind] || defaultDevice,
onChange: e => this.setDevice(e.target.value, kind)
}, this.renderDeviceOptions(devices, kind));
}
render() {
let requestButton;
let speakerDropdown;
let microphoneDropdown;
let webcamDropdown;
if (!this.state.mediaDevices) {
requestButton = /*#__PURE__*/_react.default.createElement("div", null, /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("settings|voip|missing_permissions_prompt")), /*#__PURE__*/_react.default.createElement(_AccessibleButton.default, {
onClick: this.requestMediaPermissions,
kind: "primary"
}, (0, _languageHandler._t)("settings|voip|request_permissions")));
} else if (this.state.mediaDevices) {
speakerDropdown = this.renderDropdown(_MediaDeviceHandler.MediaDeviceKindEnum.AudioOutput, (0, _languageHandler._t)("settings|voip|audio_output")) || /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("settings|voip|audio_output_empty"));
microphoneDropdown = this.renderDropdown(_MediaDeviceHandler.MediaDeviceKindEnum.AudioInput, (0, _languageHandler._t)("common|microphone")) || /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("settings|voip|audio_input_empty"));
webcamDropdown = this.renderDropdown(_MediaDeviceHandler.MediaDeviceKindEnum.VideoInput, (0, _languageHandler._t)("common|camera")) || /*#__PURE__*/_react.default.createElement("p", null, (0, _languageHandler._t)("settings|voip|video_input_empty"));
}
return /*#__PURE__*/_react.default.createElement(_SettingsTab.default, null, /*#__PURE__*/_react.default.createElement(_SettingsSection.SettingsSection, null, requestButton, /*#__PURE__*/_react.default.createElement(_SettingsSubsection.default, {
heading: (0, _languageHandler._t)("settings|voip|voice_section"),
stretchContent: true
}, speakerDropdown, microphoneDropdown, /*#__PURE__*/_react.default.createElement(_LabelledToggleSwitch.default, {
value: this.state.audioAutoGainControl,
onChange: async v => {
await _MediaDeviceHandler.default.setAudioAutoGainControl(v);
this.setState({
audioAutoGainControl: _MediaDeviceHandler.default.getAudioAutoGainControl()
});
},
label: (0, _languageHandler._t)("settings|voip|voice_agc"),
"data-testid": "voice-auto-gain"
})), /*#__PURE__*/_react.default.createElement(_SettingsSubsection.default, {
heading: (0, _languageHandler._t)("settings|voip|video_section"),
stretchContent: true
}, webcamDropdown, /*#__PURE__*/_react.default.createElement(_SettingsFlag.default, {
name: "VideoView.flipVideoHorizontally",
level: _SettingLevel.SettingLevel.ACCOUNT
}))), /*#__PURE__*/_react.default.createElement(_SettingsSection.SettingsSection, {
heading: (0, _languageHandler._t)("common|advanced")
}, /*#__PURE__*/_react.default.createElement(_SettingsSubsection.default, {
heading: (0, _languageHandler._t)("settings|voip|voice_processing")
}, /*#__PURE__*/_react.default.createElement(_LabelledToggleSwitch.default, {
value: this.state.audioNoiseSuppression,
onChange: async v => {
await _MediaDeviceHandler.default.setAudioNoiseSuppression(v);
this.setState({
audioNoiseSuppression: _MediaDeviceHandler.default.getAudioNoiseSuppression()
});
},
label: (0, _languageHandler._t)("settings|voip|noise_suppression"),
"data-testid": "voice-noise-suppression"
}), /*#__PURE__*/_react.default.createElement(_LabelledToggleSwitch.default, {
value: this.state.audioEchoCancellation,
onChange: async v => {
await _MediaDeviceHandler.default.setAudioEchoCancellation(v);
this.setState({
audioEchoCancellation: _MediaDeviceHandler.default.getAudioEchoCancellation()
});
},
label: (0, _languageHandler._t)("settings|voip|echo_cancellation"),
"data-testid": "voice-echo-cancellation"
})), /*#__PURE__*/_react.default.createElement(_SettingsSubsection.default, {
heading: (0, _languageHandler._t)("settings|voip|connection_section")
}, /*#__PURE__*/_react.default.createElement(_SettingsFlag.default, {
name: "webRtcAllowPeerToPeer",
level: _SettingLevel.SettingLevel.DEVICE,
onChange: this.changeWebRtcMethod
}), /*#__PURE__*/_react.default.createElement(_SettingsFlag.default, {
name: "fallbackICEServerAllowed",
label: (0, _languageHandler._t)("settings|voip|enable_fallback_ice_server", {
server: new URL(_call.FALLBACK_ICE_SERVER).pathname
}),
level: _SettingLevel.SettingLevel.DEVICE,
hideIfCannotSet: true
}))));
}
}
exports.default = VoiceUserSettingsTab;
(0, _defineProperty2.default)(VoiceUserSettingsTab, "contextType", _MatrixClientContext.default);
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireDefault","require","_logger","_call","_languageHandler","_MediaDeviceHandler","_interopRequireWildcard","_Field","_AccessibleButton","_SettingLevel","_SettingsFlag","_LabelledToggleSwitch","_requestMediaPermissions","_SettingsTab","_SettingsSection","_SettingsSubsection","_MatrixClientContext","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","mapDeviceKindToHandlerValue","deviceKind","MediaDeviceKindEnum","AudioOutput","MediaDeviceHandler","getAudioOutput","AudioInput","getAudioInput","VideoInput","getVideoInput","VoiceUserSettingsTab","React","Component","constructor","props","context","_defineProperty2","stream","setState","mediaDevices","getDevices","getTracks","forEach","track","stop","requestMediaPermissions","refreshMediaDevices","deviceId","kind","instance","setDevice","error","logger","p2p","setForceTURN","state","audioAutoGainControl","getAudioAutoGainControl","audioEchoCancellation","getAudioEchoCancellation","audioNoiseSuppression","getAudioNoiseSuppression","componentDidMount","canSeeDeviceLabels","hasAnyLabeledDevices","renderDeviceOptions","devices","category","map","d","createElement","key","value","label","renderDropdown","slice","length","defaultDevice","getDefaultDevice","element","onChange","target","render","requestButton","speakerDropdown","microphoneDropdown","webcamDropdown","_t","onClick","SettingsSection","heading","stretchContent","v","setAudioAutoGainControl","name","level","SettingLevel","ACCOUNT","setAudioNoiseSuppression","setAudioEchoCancellation","DEVICE","changeWebRtcMethod","server","URL","FALLBACK_ICE_SERVER","pathname","hideIfCannotSet","exports","MatrixClientContext"],"sources":["../../../../../../src/components/views/settings/tabs/user/VoiceUserSettingsTab.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2020 The Matrix.org Foundation C.I.C.\nCopyright 2019 New Vector Ltd\n\nSPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only\nPlease see LICENSE files in the repository root for full details.\n*/\n\nimport React, { ReactNode } from \"react\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport { FALLBACK_ICE_SERVER } from \"matrix-js-sdk/src/webrtc/call\";\n\nimport { _t } from \"../../../../../languageHandler\";\nimport MediaDeviceHandler, { IMediaDevices, MediaDeviceKindEnum } from \"../../../../../MediaDeviceHandler\";\nimport Field from \"../../../elements/Field\";\nimport AccessibleButton from \"../../../elements/AccessibleButton\";\nimport { SettingLevel } from \"../../../../../settings/SettingLevel\";\nimport SettingsFlag from \"../../../elements/SettingsFlag\";\nimport LabelledToggleSwitch from \"../../../elements/LabelledToggleSwitch\";\nimport { requestMediaPermissions } from \"../../../../../utils/media/requestMediaPermissions\";\nimport SettingsTab from \"../SettingsTab\";\nimport { SettingsSection } from \"../../shared/SettingsSection\";\nimport SettingsSubsection from \"../../shared/SettingsSubsection\";\nimport MatrixClientContext from \"../../../../../contexts/MatrixClientContext\";\n\ninterface IState {\n    mediaDevices: IMediaDevices | null;\n    [MediaDeviceKindEnum.AudioOutput]: string | null;\n    [MediaDeviceKindEnum.AudioInput]: string | null;\n    [MediaDeviceKindEnum.VideoInput]: string | null;\n    audioAutoGainControl: boolean;\n    audioEchoCancellation: boolean;\n    audioNoiseSuppression: boolean;\n}\n\n/**\n * Maps deviceKind to the right get method on MediaDeviceHandler\n * Helpful for setting state\n */\nconst mapDeviceKindToHandlerValue = (deviceKind: MediaDeviceKindEnum): string | null => {\n    switch (deviceKind) {\n        case MediaDeviceKindEnum.AudioOutput:\n            return MediaDeviceHandler.getAudioOutput();\n        case MediaDeviceKindEnum.AudioInput:\n            return MediaDeviceHandler.getAudioInput();\n        case MediaDeviceKindEnum.VideoInput:\n            return MediaDeviceHandler.getVideoInput();\n    }\n};\n\nexport default class VoiceUserSettingsTab extends React.Component<{}, IState> {\n    public static contextType = MatrixClientContext;\n    public declare context: React.ContextType<typeof MatrixClientContext>;\n\n    public constructor(props: {}, context: React.ContextType<typeof MatrixClientContext>) {\n        super(props, context);\n\n        this.state = {\n            mediaDevices: null,\n            [MediaDeviceKindEnum.AudioOutput]: null,\n            [MediaDeviceKindEnum.AudioInput]: null,\n            [MediaDeviceKindEnum.VideoInput]: null,\n            audioAutoGainControl: MediaDeviceHandler.getAudioAutoGainControl(),\n            audioEchoCancellation: MediaDeviceHandler.getAudioEchoCancellation(),\n            audioNoiseSuppression: MediaDeviceHandler.getAudioNoiseSuppression(),\n        };\n    }\n\n    public async componentDidMount(): Promise<void> {\n        const canSeeDeviceLabels = await MediaDeviceHandler.hasAnyLabeledDevices();\n        if (canSeeDeviceLabels) {\n            await this.refreshMediaDevices();\n        }\n    }\n\n    private refreshMediaDevices = async (stream?: MediaStream): Promise<void> => {\n        this.setState({\n            mediaDevices: (await MediaDeviceHandler.getDevices()) ?? null,\n            [MediaDeviceKindEnum.AudioOutput]: mapDeviceKindToHandlerValue(MediaDeviceKindEnum.AudioOutput),\n            [MediaDeviceKindEnum.AudioInput]: mapDeviceKindToHandlerValue(MediaDeviceKindEnum.AudioInput),\n            [MediaDeviceKindEnum.VideoInput]: mapDeviceKindToHandlerValue(MediaDeviceKindEnum.VideoInput),\n        });\n        if (stream) {\n            // kill stream (after we've enumerated the devices, otherwise we'd get empty labels again)\n            // so that we don't leave it lingering around with webcam enabled etc\n            // as here we called gUM to ask user for permission to their device names only\n            stream.getTracks().forEach((track) => track.stop());\n        }\n    };\n\n    private requestMediaPermissions = async (): Promise<void> => {\n        const stream = await requestMediaPermissions();\n        if (stream) {\n            await this.refreshMediaDevices(stream);\n        }\n    };\n\n    private setDevice = async (deviceId: string, kind: MediaDeviceKindEnum): Promise<void> => {\n        // set state immediately so UI is responsive\n        this.setState<any>({ [kind]: deviceId });\n        try {\n            await MediaDeviceHandler.instance.setDevice(deviceId, kind);\n        } catch (error) {\n            logger.error(`Failed to set device ${kind}: ${deviceId}`);\n            // reset state to current value\n            this.setState<any>({ [kind]: mapDeviceKindToHandlerValue(kind) });\n        }\n    };\n\n    private changeWebRtcMethod = (p2p: boolean): void => {\n        this.context.setForceTURN(!p2p);\n    };\n\n    private renderDeviceOptions(devices: Array<MediaDeviceInfo>, category: MediaDeviceKindEnum): Array<JSX.Element> {\n        return devices.map((d) => {\n            return (\n                <option key={`${category}-${d.deviceId}`} value={d.deviceId}>\n                    {d.label}\n                </option>\n            );\n        });\n    }\n\n    private renderDropdown(kind: MediaDeviceKindEnum, label: string): ReactNode {\n        const devices = this.state.mediaDevices?.[kind].slice(0);\n        if (!devices?.length) return null;\n\n        const defaultDevice = MediaDeviceHandler.getDefaultDevice(devices);\n        return (\n            <Field\n                element=\"select\"\n                label={label}\n                value={this.state[kind] || defaultDevice}\n                onChange={(e) => this.setDevice(e.target.value, kind)}\n            >\n                {this.renderDeviceOptions(devices, kind)}\n            </Field>\n        );\n    }\n\n    public render(): ReactNode {\n        let requestButton: ReactNode | undefined;\n        let speakerDropdown: ReactNode | undefined;\n        let microphoneDropdown: ReactNode | undefined;\n        let webcamDropdown: ReactNode | undefined;\n        if (!this.state.mediaDevices) {\n            requestButton = (\n                <div>\n                    <p>{_t(\"settings|voip|missing_permissions_prompt\")}</p>\n                    <AccessibleButton onClick={this.requestMediaPermissions} kind=\"primary\">\n                        {_t(\"settings|voip|request_permissions\")}\n                    </AccessibleButton>\n                </div>\n            );\n        } else if (this.state.mediaDevices) {\n            speakerDropdown = this.renderDropdown(\n                MediaDeviceKindEnum.AudioOutput,\n                _t(\"settings|voip|audio_output\"),\n            ) || <p>{_t(\"settings|voip|audio_output_empty\")}</p>;\n            microphoneDropdown = this.renderDropdown(MediaDeviceKindEnum.AudioInput, _t(\"common|microphone\")) || (\n                <p>{_t(\"settings|voip|audio_input_empty\")}</p>\n            );\n            webcamDropdown = this.renderDropdown(MediaDeviceKindEnum.VideoInput, _t(\"common|camera\")) || (\n                <p>{_t(\"settings|voip|video_input_empty\")}</p>\n            );\n        }\n\n        return (\n            <SettingsTab>\n                <SettingsSection>\n                    {requestButton}\n                    <SettingsSubsection heading={_t(\"settings|voip|voice_section\")} stretchContent>\n                        {speakerDropdown}\n                        {microphoneDropdown}\n                        <LabelledToggleSwitch\n                            value={this.state.audioAutoGainControl}\n                            onChange={async (v): Promise<void> => {\n                                await MediaDeviceHandler.setAudioAutoGainControl(v);\n                                this.setState({ audioAutoGainControl: MediaDeviceHandler.getAudioAutoGainControl() });\n                            }}\n                            label={_t(\"settings|voip|voice_agc\")}\n                            data-testid=\"voice-auto-gain\"\n                        />\n                    </SettingsSubsection>\n                    <SettingsSubsection heading={_t(\"settings|voip|video_section\")} stretchContent>\n                        {webcamDropdown}\n                        <SettingsFlag name=\"VideoView.flipVideoHorizontally\" level={SettingLevel.ACCOUNT} />\n                    </SettingsSubsection>\n                </SettingsSection>\n\n                <SettingsSection heading={_t(\"common|advanced\")}>\n                    <SettingsSubsection heading={_t(\"settings|voip|voice_processing\")}>\n                        <LabelledToggleSwitch\n                            value={this.state.audioNoiseSuppression}\n                            onChange={async (v): Promise<void> => {\n                                await MediaDeviceHandler.setAudioNoiseSuppression(v);\n                                this.setState({ audioNoiseSuppression: MediaDeviceHandler.getAudioNoiseSuppression() });\n                            }}\n                            label={_t(\"settings|voip|noise_suppression\")}\n                            data-testid=\"voice-noise-suppression\"\n                        />\n                        <LabelledToggleSwitch\n                            value={this.state.audioEchoCancellation}\n                            onChange={async (v): Promise<void> => {\n                                await MediaDeviceHandler.setAudioEchoCancellation(v);\n                                this.setState({ audioEchoCancellation: MediaDeviceHandler.getAudioEchoCancellation() });\n                            }}\n                            label={_t(\"settings|voip|echo_cancellation\")}\n                            data-testid=\"voice-echo-cancellation\"\n                        />\n                    </SettingsSubsection>\n                    <SettingsSubsection heading={_t(\"settings|voip|connection_section\")}>\n                        <SettingsFlag\n                            name=\"webRtcAllowPeerToPeer\"\n                            level={SettingLevel.DEVICE}\n                            onChange={this.changeWebRtcMethod}\n                        />\n                        <SettingsFlag\n                            name=\"fallbackICEServerAllowed\"\n                            label={_t(\"settings|voip|enable_fallback_ice_server\", {\n                                server: new URL(FALLBACK_ICE_SERVER).pathname,\n                            })}\n                            level={SettingLevel.DEVICE}\n                            hideIfCannotSet\n                        />\n                    </SettingsSubsection>\n                </SettingsSection>\n            </SettingsTab>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}