UNPKG

matrix-react-sdk

Version:
182 lines (177 loc) 28.3 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 _classnames = _interopRequireDefault(require("classnames")); var _react = _interopRequireDefault(require("react")); var _callFeed = require("matrix-js-sdk/src/webrtc/callFeed"); var _logger = require("matrix-js-sdk/src/logger"); var _callEventTypes = require("matrix-js-sdk/src/webrtc/callEventTypes"); var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore")); var _LegacyCallHandler = _interopRequireDefault(require("../../../LegacyCallHandler")); var _MatrixClientPeg = require("../../../MatrixClientPeg"); var _RoomAvatar = _interopRequireDefault(require("../avatars/RoomAvatar")); /* Copyright 2024 New Vector Ltd. Copyright 2021, 2022 Šimon Brandner <simon.bra.ag@gmail.com> Copyright 2015, 2016 , 2019, 2020, 2021 The Matrix.org Foundation C.I.C. SPDX-License-Identifier: AGPL-3.0-only OR GPL-3.0-only Please see LICENSE files in the repository root for full details. */ class VideoFeed extends _react.default.PureComponent { constructor(props) { super(props); (0, _defineProperty2.default)(this, "element", void 0); (0, _defineProperty2.default)(this, "setElementRef", element => { if (!element) { this.element?.removeEventListener("resize", this.onResize); return; } this.element = element; element.addEventListener("resize", this.onResize); }); (0, _defineProperty2.default)(this, "onNewStream", () => { this.setState({ audioMuted: this.props.feed.isAudioMuted(), videoMuted: this.props.feed.isVideoMuted() }); this.playMedia(); }); (0, _defineProperty2.default)(this, "onMuteStateChanged", () => { this.setState({ audioMuted: this.props.feed.isAudioMuted(), videoMuted: this.props.feed.isVideoMuted() }); }); (0, _defineProperty2.default)(this, "onResize", e => { if (this.props.onResize && !this.props.feed.isLocal()) { this.props.onResize(e); } }); this.state = { audioMuted: this.props.feed.isAudioMuted(), videoMuted: this.props.feed.isVideoMuted() }; } componentDidMount() { this.updateFeed(null, this.props.feed); this.playMedia(); } componentWillUnmount() { this.updateFeed(this.props.feed, null); } componentDidUpdate(prevProps, prevState) { this.updateFeed(prevProps.feed, this.props.feed); // If the mutes state has changed, we try to playMedia() if (prevState.videoMuted !== this.state.videoMuted || prevProps.feed.stream !== this.props.feed.stream) { this.playMedia(); } } static getDerivedStateFromProps(props) { return { audioMuted: props.feed.isAudioMuted(), videoMuted: props.feed.isVideoMuted() }; } updateFeed(oldFeed, newFeed) { if (oldFeed === newFeed) return; if (oldFeed) { this.props.feed.removeListener(_callFeed.CallFeedEvent.NewStream, this.onNewStream); this.props.feed.removeListener(_callFeed.CallFeedEvent.MuteStateChanged, this.onMuteStateChanged); if (this.props.feed.purpose === _callEventTypes.SDPStreamMetadataPurpose.Usermedia) { this.props.feed.measureVolumeActivity(false); } this.stopMedia(); } if (newFeed) { this.props.feed.addListener(_callFeed.CallFeedEvent.NewStream, this.onNewStream); this.props.feed.addListener(_callFeed.CallFeedEvent.MuteStateChanged, this.onMuteStateChanged); if (this.props.feed.purpose === _callEventTypes.SDPStreamMetadataPurpose.Usermedia) { this.props.feed.measureVolumeActivity(true); } this.playMedia(); } } async playMedia() { const element = this.element; if (!element) return; // We play audio in AudioFeed, not here element.muted = true; 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.play(); } 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; 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() { const { pipMode, primary, secondary, feed } = this.props; const wrapperClasses = (0, _classnames.default)("mx_VideoFeed", { mx_VideoFeed_primary: primary, mx_VideoFeed_secondary: secondary, mx_VideoFeed_voice: this.state.videoMuted }); const micIconClasses = (0, _classnames.default)("mx_VideoFeed_mic", { mx_VideoFeed_mic_muted: this.state.audioMuted, mx_VideoFeed_mic_unmuted: !this.state.audioMuted }); let micIcon; if (feed.purpose !== _callEventTypes.SDPStreamMetadataPurpose.Screenshare && !pipMode) { micIcon = /*#__PURE__*/_react.default.createElement("div", { className: micIconClasses }); } let content; if (this.state.videoMuted) { const callRoomId = _LegacyCallHandler.default.instance.roomIdForCall(this.props.call); const callRoom = (callRoomId ? _MatrixClientPeg.MatrixClientPeg.safeGet().getRoom(callRoomId) : undefined) ?? undefined; let avatarSize; if (pipMode && primary) avatarSize = "76px";else if (pipMode && !primary) avatarSize = "16px";else if (!pipMode && primary) avatarSize = "160px";else ; // TBD content = /*#__PURE__*/_react.default.createElement(_RoomAvatar.default, { room: callRoom, size: avatarSize }); } else { const videoClasses = (0, _classnames.default)("mx_VideoFeed_video", { mx_VideoFeed_video_mirror: this.props.feed.isLocal() && this.props.feed.purpose === _callEventTypes.SDPStreamMetadataPurpose.Usermedia && _SettingsStore.default.getValue("VideoView.flipVideoHorizontally") }); content = /*#__PURE__*/_react.default.createElement("video", { className: videoClasses, ref: this.setElementRef }); } return /*#__PURE__*/_react.default.createElement("div", { className: wrapperClasses }, micIcon, content); } } exports.default = VideoFeed; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_classnames","_interopRequireDefault","require","_react","_callFeed","_logger","_callEventTypes","_SettingsStore","_LegacyCallHandler","_MatrixClientPeg","_RoomAvatar","VideoFeed","React","PureComponent","constructor","props","_defineProperty2","default","element","removeEventListener","onResize","addEventListener","setState","audioMuted","feed","isAudioMuted","videoMuted","isVideoMuted","playMedia","e","isLocal","state","componentDidMount","updateFeed","componentWillUnmount","componentDidUpdate","prevProps","prevState","stream","getDerivedStateFromProps","oldFeed","newFeed","removeListener","CallFeedEvent","NewStream","onNewStream","MuteStateChanged","onMuteStateChanged","purpose","SDPStreamMetadataPurpose","Usermedia","measureVolumeActivity","stopMedia","addListener","muted","srcObject","autoplay","play","logger","info","userId","pause","removeAttribute","render","pipMode","primary","secondary","wrapperClasses","classnames","mx_VideoFeed_primary","mx_VideoFeed_secondary","mx_VideoFeed_voice","micIconClasses","mx_VideoFeed_mic_muted","mx_VideoFeed_mic_unmuted","micIcon","Screenshare","createElement","className","content","callRoomId","LegacyCallHandler","instance","roomIdForCall","call","callRoom","MatrixClientPeg","safeGet","getRoom","undefined","avatarSize","room","size","videoClasses","mx_VideoFeed_video_mirror","SettingsStore","getValue","ref","setElementRef","exports"],"sources":["../../../../src/components/views/voip/VideoFeed.tsx"],"sourcesContent":["/*\nCopyright 2024 New Vector Ltd.\nCopyright 2021, 2022 Šimon Brandner <simon.bra.ag@gmail.com>\nCopyright 2015, 2016 , 2019, 2020, 2021 The Matrix.org Foundation C.I.C.\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 classnames from \"classnames\";\nimport { MatrixCall } from \"matrix-js-sdk/src/webrtc/call\";\nimport React from \"react\";\nimport { CallFeed, CallFeedEvent } from \"matrix-js-sdk/src/webrtc/callFeed\";\nimport { logger } from \"matrix-js-sdk/src/logger\";\nimport { SDPStreamMetadataPurpose } from \"matrix-js-sdk/src/webrtc/callEventTypes\";\n\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport LegacyCallHandler from \"../../../LegacyCallHandler\";\nimport { MatrixClientPeg } from \"../../../MatrixClientPeg\";\nimport RoomAvatar from \"../avatars/RoomAvatar\";\n\ninterface IProps {\n    call: MatrixCall;\n\n    feed: CallFeed;\n\n    // Whether this call view is for picture-in-picture mode\n    // otherwise, it's the larger call view when viewing the room the call is in.\n    // This is sort of a proxy for a number of things but we currently have no\n    // need to control those things separately, so this is simpler.\n    pipMode?: boolean;\n\n    // a callback which is called when the video element is resized\n    // due to a change in video metadata\n    onResize?: (e: Event) => void;\n\n    primary?: boolean;\n    secondary?: boolean;\n}\n\ninterface IState {\n    audioMuted: boolean;\n    videoMuted: boolean;\n}\n\nexport default class VideoFeed extends React.PureComponent<IProps, IState> {\n    private element?: HTMLVideoElement;\n\n    public constructor(props: IProps) {\n        super(props);\n\n        this.state = {\n            audioMuted: this.props.feed.isAudioMuted(),\n            videoMuted: this.props.feed.isVideoMuted(),\n        };\n    }\n\n    public componentDidMount(): void {\n        this.updateFeed(null, this.props.feed);\n        this.playMedia();\n    }\n\n    public componentWillUnmount(): void {\n        this.updateFeed(this.props.feed, null);\n    }\n\n    public componentDidUpdate(prevProps: IProps, prevState: IState): void {\n        this.updateFeed(prevProps.feed, this.props.feed);\n        // If the mutes state has changed, we try to playMedia()\n        if (prevState.videoMuted !== this.state.videoMuted || prevProps.feed.stream !== this.props.feed.stream) {\n            this.playMedia();\n        }\n    }\n\n    public static getDerivedStateFromProps(props: IProps): IState {\n        return {\n            audioMuted: props.feed.isAudioMuted(),\n            videoMuted: props.feed.isVideoMuted(),\n        };\n    }\n\n    private setElementRef = (element: HTMLVideoElement): void => {\n        if (!element) {\n            this.element?.removeEventListener(\"resize\", this.onResize);\n            return;\n        }\n\n        this.element = element;\n        element.addEventListener(\"resize\", this.onResize);\n    };\n\n    private updateFeed(oldFeed: CallFeed | null, newFeed: CallFeed | null): void {\n        if (oldFeed === newFeed) return;\n\n        if (oldFeed) {\n            this.props.feed.removeListener(CallFeedEvent.NewStream, this.onNewStream);\n            this.props.feed.removeListener(CallFeedEvent.MuteStateChanged, this.onMuteStateChanged);\n            if (this.props.feed.purpose === SDPStreamMetadataPurpose.Usermedia) {\n                this.props.feed.measureVolumeActivity(false);\n            }\n            this.stopMedia();\n        }\n        if (newFeed) {\n            this.props.feed.addListener(CallFeedEvent.NewStream, this.onNewStream);\n            this.props.feed.addListener(CallFeedEvent.MuteStateChanged, this.onMuteStateChanged);\n            if (this.props.feed.purpose === SDPStreamMetadataPurpose.Usermedia) {\n                this.props.feed.measureVolumeActivity(true);\n            }\n            this.playMedia();\n        }\n    }\n\n    private async playMedia(): Promise<void> {\n        const element = this.element;\n        if (!element) return;\n        // We play audio in AudioFeed, not here\n        element.muted = true;\n        element.srcObject = this.props.feed.stream;\n        element.autoplay = true;\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.play();\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;\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            videoMuted: this.props.feed.isVideoMuted(),\n        });\n        this.playMedia();\n    };\n\n    private onMuteStateChanged = (): void => {\n        this.setState({\n            audioMuted: this.props.feed.isAudioMuted(),\n            videoMuted: this.props.feed.isVideoMuted(),\n        });\n    };\n\n    private onResize = (e: Event): void => {\n        if (this.props.onResize && !this.props.feed.isLocal()) {\n            this.props.onResize(e);\n        }\n    };\n\n    public render(): React.ReactNode {\n        const { pipMode, primary, secondary, feed } = this.props;\n\n        const wrapperClasses = classnames(\"mx_VideoFeed\", {\n            mx_VideoFeed_primary: primary,\n            mx_VideoFeed_secondary: secondary,\n            mx_VideoFeed_voice: this.state.videoMuted,\n        });\n        const micIconClasses = classnames(\"mx_VideoFeed_mic\", {\n            mx_VideoFeed_mic_muted: this.state.audioMuted,\n            mx_VideoFeed_mic_unmuted: !this.state.audioMuted,\n        });\n\n        let micIcon;\n        if (feed.purpose !== SDPStreamMetadataPurpose.Screenshare && !pipMode) {\n            micIcon = <div className={micIconClasses} />;\n        }\n\n        let content;\n        if (this.state.videoMuted) {\n            const callRoomId = LegacyCallHandler.instance.roomIdForCall(this.props.call);\n            const callRoom = (callRoomId ? MatrixClientPeg.safeGet().getRoom(callRoomId) : undefined) ?? undefined;\n\n            let avatarSize;\n            if (pipMode && primary) avatarSize = \"76px\";\n            else if (pipMode && !primary) avatarSize = \"16px\";\n            else if (!pipMode && primary) avatarSize = \"160px\";\n            else; // TBD\n\n            content = <RoomAvatar room={callRoom} size={avatarSize} />;\n        } else {\n            const videoClasses = classnames(\"mx_VideoFeed_video\", {\n                mx_VideoFeed_video_mirror:\n                    this.props.feed.isLocal() &&\n                    this.props.feed.purpose === SDPStreamMetadataPurpose.Usermedia &&\n                    SettingsStore.getValue(\"VideoView.flipVideoHorizontally\"),\n            });\n\n            content = <video className={videoClasses} ref={this.setElementRef} />;\n        }\n\n        return (\n            <div className={wrapperClasses}>\n                {micIcon}\n                {content}\n            </div>\n        );\n    }\n}\n"],"mappings":"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","ignoreList":[]}