matrix-react-sdk
Version:
SDK for matrix.org using React
104 lines (100 loc) • 17.3 kB
JavaScript
"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 = _interopRequireWildcard(require("react"));
var _callFeed = require("matrix-js-sdk/src/webrtc/callFeed");
var _logger = require("matrix-js-sdk/src/logger");
var _MediaDeviceHandler = _interopRequireWildcard(require("../../../MediaDeviceHandler"));
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 2021 Šimon Brandner <simon.bra.ag@gmail.com>
SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only
Please see LICENSE files in the repository root for full details.
*/
class AudioFeed extends _react.default.Component {
constructor(props) {
super(props);
(0, _defineProperty2.default)(this, "element", /*#__PURE__*/(0, _react.createRef)());
(0, _defineProperty2.default)(this, "onAudioOutputChanged", audioOutput => {
const element = this.element.current;
if (audioOutput) {
try {
// This seems quite unreliable in Chrome, although I haven't yet managed to make a jsfiddle where
// it fails.
// It seems reliable if you set the sink ID after setting the srcObject and then set the sink ID
// back to the default after the call is over - Dave
element.setSinkId(audioOutput);
} catch (e) {
_logger.logger.error("Couldn't set requested audio output device: using default", e);
_logger.logger.warn("Couldn't set requested audio output device: using default", e);
}
}
});
(0, _defineProperty2.default)(this, "onNewStream", () => {
this.setState({
audioMuted: this.props.feed.isAudioMuted()
});
this.playMedia();
});
this.state = {
audioMuted: this.props.feed.isAudioMuted()
};
}
componentDidMount() {
_MediaDeviceHandler.default.instance.addListener(_MediaDeviceHandler.MediaDeviceHandlerEvent.AudioOutputChanged, this.onAudioOutputChanged);
this.props.feed.addListener(_callFeed.CallFeedEvent.NewStream, this.onNewStream);
this.playMedia();
}
componentWillUnmount() {
_MediaDeviceHandler.default.instance.removeListener(_MediaDeviceHandler.MediaDeviceHandlerEvent.AudioOutputChanged, this.onAudioOutputChanged);
this.props.feed.removeListener(_callFeed.CallFeedEvent.NewStream, this.onNewStream);
this.stopMedia();
}
async playMedia() {
const element = this.element.current;
if (!element) return;
this.onAudioOutputChanged(_MediaDeviceHandler.default.getAudioOutput());
element.muted = false;
element.srcObject = this.props.feed.stream;
element.autoplay = true;
try {
// A note on calling methods on media elements:
// We used to have queues per media element to serialise all calls on those elements.
// The reason given for this was that load() and play() were racing. However, we now
// never call load() explicitly so this seems unnecessary. However, serialising every
// operation was causing bugs where video would not resume because some play command
// had got stuck and all media operations were queued up behind it. If necessary, we
// should serialise the ones that need to be serialised but then be able to interrupt
// them with another load() which will cancel the pending one, but since we don't call
// load() explicitly, it shouldn't be a problem. - Dave
await element.load();
} catch (e) {
_logger.logger.info(`Failed to play media element with feed for userId ` + `${this.props.feed.userId} with purpose ${this.props.feed.purpose}`, e);
}
}
stopMedia() {
const element = this.element.current;
if (!element) return;
element.pause();
element.removeAttribute("src");
// As per comment in componentDidMount, setting the sink ID back to the
// default once the call is over makes setSinkId work reliably. - Dave
// Since we are not using the same element anymore, the above doesn't
// seem to be necessary - Šimon
}
render() {
// Do not render the audio element if there is no audio track
if (this.state.audioMuted) return null;
return /*#__PURE__*/_react.default.createElement("audio", {
ref: this.element
});
}
}
exports.default = AudioFeed;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_react","_interopRequireWildcard","require","_callFeed","_logger","_MediaDeviceHandler","_getRequireWildcardCache","e","WeakMap","r","t","__esModule","default","has","get","n","__proto__","a","Object","defineProperty","getOwnPropertyDescriptor","u","hasOwnProperty","call","i","set","AudioFeed","React","Component","constructor","props","_defineProperty2","createRef","audioOutput","element","current","setSinkId","logger","error","warn","setState","audioMuted","feed","isAudioMuted","playMedia","state","componentDidMount","MediaDeviceHandler","instance","addListener","MediaDeviceHandlerEvent","AudioOutputChanged","onAudioOutputChanged","CallFeedEvent","NewStream","onNewStream","componentWillUnmount","removeListener","stopMedia","getAudioOutput","muted","srcObject","stream","autoplay","load","info","userId","purpose","pause","removeAttribute","render","createElement","ref","exports"],"sources":["../../../../src/components/views/voip/AudioFeed.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2021 Šimon Brandner <simon.bra.ag@gmail.com>\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, { createRef } from \"react\";\nimport { CallFeed, CallFeedEvent } from \"matrix-js-sdk/src/webrtc/callFeed\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\n\nimport MediaDeviceHandler, { MediaDeviceHandlerEvent } from \"../../../MediaDeviceHandler\";\n\ninterface IProps {\n    feed: CallFeed;\n}\n\ninterface IState {\n    audioMuted: boolean;\n}\n\nexport default class AudioFeed extends React.Component<IProps, IState> {\n    private element = createRef<HTMLAudioElement>();\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            audioMuted: this.props.feed.isAudioMuted(),\n        };\n    }\n\n    public componentDidMount(): void {\n        MediaDeviceHandler.instance.addListener(MediaDeviceHandlerEvent.AudioOutputChanged, this.onAudioOutputChanged);\n        this.props.feed.addListener(CallFeedEvent.NewStream, this.onNewStream);\n        this.playMedia();\n    }\n\n    public componentWillUnmount(): void {\n        MediaDeviceHandler.instance.removeListener(\n            MediaDeviceHandlerEvent.AudioOutputChanged,\n            this.onAudioOutputChanged,\n        );\n        this.props.feed.removeListener(CallFeedEvent.NewStream, this.onNewStream);\n        this.stopMedia();\n    }\n\n    private onAudioOutputChanged = (audioOutput: string): void => {\n        const element = this.element.current;\n        if (audioOutput) {\n            try {\n                // This seems quite unreliable in Chrome, although I haven't yet managed to make a jsfiddle where\n                // it fails.\n                // It seems reliable if you set the sink ID after setting the srcObject and then set the sink ID\n                // back to the default after the call is over - Dave\n                element!.setSinkId(audioOutput);\n            } catch (e) {\n                logger.error(\"Couldn't set requested audio output device: using default\", e);\n                logger.warn(\"Couldn't set requested audio output device: using default\", e);\n            }\n        }\n    };\n\n    private async playMedia(): Promise<void> {\n        const element = this.element.current;\n        if (!element) return;\n        this.onAudioOutputChanged(MediaDeviceHandler.getAudioOutput());\n        element.muted = false;\n        element.srcObject = this.props.feed.stream;\n        element.autoplay = true;\n\n        try {\n            // A note on calling methods on media elements:\n            // We used to have queues per media element to serialise all calls on those elements.\n            // The reason given for this was that load() and play() were racing. However, we now\n            // never call load() explicitly so this seems unnecessary. However, serialising every\n            // operation was causing bugs where video would not resume because some play command\n            // had got stuck and all media operations were queued up behind it. If necessary, we\n            // should serialise the ones that need to be serialised but then be able to interrupt\n            // them with another load() which will cancel the pending one, but since we don't call\n            // load() explicitly, it shouldn't be a problem. - Dave\n            await element.load();\n        } catch (e) {\n            logger.info(\n                `Failed to play media element with feed for userId ` +\n                    `${this.props.feed.userId} with purpose ${this.props.feed.purpose}`,\n                e,\n            );\n        }\n    }\n\n    private stopMedia(): void {\n        const element = this.element.current;\n        if (!element) return;\n\n        element.pause();\n        element.removeAttribute(\"src\");\n\n        // As per comment in componentDidMount, setting the sink ID back to the\n        // default once the call is over makes setSinkId work reliably. - Dave\n        // Since we are not using the same element anymore, the above doesn't\n        // seem to be necessary - Šimon\n    }\n\n    private onNewStream = (): void => {\n        this.setState({\n            audioMuted: this.props.feed.isAudioMuted(),\n        });\n        this.playMedia();\n    };\n\n    public render(): React.ReactNode {\n        // Do not render the audio element if there is no audio track\n        if (this.state.audioMuted) return null;\n\n        return <audio ref={this.element} />;\n    }\n}\n"],"mappings":"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","ignoreList":[]}