UNPKG

matrix-react-sdk

Version:
201 lines (198 loc) 39.2 kB
"use strict"; 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":[]}