matrix-react-sdk
Version:
SDK for matrix.org using React
179 lines (140 loc) • 20.5 kB
JavaScript
"use strict";
var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault");
var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard");
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.default = void 0;
var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty"));
var _react = _interopRequireDefault(require("react"));
var _CallView = _interopRequireDefault(require("./CallView"));
var _RoomViewStore = _interopRequireDefault(require("../../../stores/RoomViewStore"));
var _CallHandler = _interopRequireWildcard(require("../../../CallHandler"));
var _dispatcher = _interopRequireDefault(require("../../../dispatcher/dispatcher"));
var _PersistentApp = _interopRequireDefault(require("../elements/PersistentApp"));
var _SettingsStore = _interopRequireDefault(require("../../../settings/SettingsStore"));
var _call = require("matrix-js-sdk/src/webrtc/call");
var _MatrixClientPeg = require("../../../MatrixClientPeg");
var _replaceableComponent = require("../../../utils/replaceableComponent");
var _dec, _class, _temp;
const SHOW_CALL_IN_STATES = [_call.CallState.Connected, _call.CallState.InviteSent, _call.CallState.Connecting, _call.CallState.CreateAnswer, _call.CallState.CreateOffer, _call.CallState.WaitLocalMedia];
// Splits a list of calls into one 'primary' one and a list
// (which should be a single element) of other calls.
// The primary will be the one not on hold, or an arbitrary one
// if they're all on hold)
function getPrimarySecondaryCalls(calls
/*: MatrixCall[]*/
)
/*: [MatrixCall, MatrixCall[]]*/
{
let primary
/*: MatrixCall*/
= null;
let secondaries
/*: MatrixCall[]*/
= [];
for (const call of calls) {
if (!SHOW_CALL_IN_STATES.includes(call.state)) continue;
if (!call.isRemoteOnHold() && primary === null) {
primary = call;
} else {
secondaries.push(call);
}
}
if (primary === null && secondaries.length > 0) {
primary = secondaries[0];
secondaries = secondaries.slice(1);
}
if (secondaries.length > 1) {
// We should never be in more than two calls so this shouldn't happen
console.log("Found more than 1 secondary call! Other calls will not be shown.");
}
return [primary, secondaries];
}
/**
* CallPreview shows a small version of CallView hovering over the UI in 'picture-in-picture'
* (PiP mode). It displays the call(s) which is *not* in the room the user is currently viewing.
*/
let CallPreview = (_dec = (0, _replaceableComponent.replaceableComponent)("views.voip.CallPreview"), _dec(_class = (_temp = class CallPreview extends _react.default.Component
/*:: <IProps, IState>*/
{
constructor(props
/*: IProps*/
) {
super(props);
(0, _defineProperty2.default)(this, "roomStoreToken", void 0);
(0, _defineProperty2.default)(this, "dispatcherRef", void 0);
(0, _defineProperty2.default)(this, "settingsWatcherRef", void 0);
(0, _defineProperty2.default)(this, "onRoomViewStoreUpdate", payload => {
if (_RoomViewStore.default.getRoomId() === this.state.roomId) return;
const roomId = _RoomViewStore.default.getRoomId();
const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(_CallHandler.default.sharedInstance().getAllActiveCallsNotInRoom(roomId));
this.setState({
roomId,
primaryCall: primaryCall,
secondaryCall: secondaryCalls[0]
});
});
(0, _defineProperty2.default)(this, "onAction", (payload
/*: ActionPayload*/
) => {
switch (payload.action) {
// listen for call state changes to prod the render method, which
// may hide the global CallView if the call it is tracking is dead
case 'call_state':
{
this.updateCalls();
break;
}
}
});
(0, _defineProperty2.default)(this, "updateCalls", () => {
const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(_CallHandler.default.sharedInstance().getAllActiveCallsNotInRoom(this.state.roomId));
this.setState({
primaryCall: primaryCall,
secondaryCall: secondaryCalls[0]
});
});
(0, _defineProperty2.default)(this, "onCallRemoteHold", () => {
const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(_CallHandler.default.sharedInstance().getAllActiveCallsNotInRoom(this.state.roomId));
this.setState({
primaryCall: primaryCall,
secondaryCall: secondaryCalls[0]
});
});
const _roomId = _RoomViewStore.default.getRoomId();
const [_primaryCall, _secondaryCalls] = getPrimarySecondaryCalls(_CallHandler.default.sharedInstance().getAllActiveCallsNotInRoom(_roomId));
this.state = {
roomId: _roomId,
primaryCall: _primaryCall,
secondaryCall: _secondaryCalls[0]
};
}
componentDidMount() {
_CallHandler.default.sharedInstance().addListener(_CallHandler.CallHandlerEvent.CallChangeRoom, this.updateCalls);
this.roomStoreToken = _RoomViewStore.default.addListener(this.onRoomViewStoreUpdate);
this.dispatcherRef = _dispatcher.default.register(this.onAction);
_MatrixClientPeg.MatrixClientPeg.get().on(_call.CallEvent.RemoteHoldUnhold, this.onCallRemoteHold);
}
componentWillUnmount() {
_CallHandler.default.sharedInstance().removeListener(_CallHandler.CallHandlerEvent.CallChangeRoom, this.updateCalls);
_MatrixClientPeg.MatrixClientPeg.get().removeListener(_call.CallEvent.RemoteHoldUnhold, this.onCallRemoteHold);
if (this.roomStoreToken) {
this.roomStoreToken.remove();
}
_dispatcher.default.unregister(this.dispatcherRef);
_SettingsStore.default.unwatchSetting(this.settingsWatcherRef);
}
render() {
if (this.state.primaryCall) {
return /*#__PURE__*/_react.default.createElement(_CallView.default, {
call: this.state.primaryCall,
secondaryCall: this.state.secondaryCall,
pipMode: true
});
}
return /*#__PURE__*/_react.default.createElement(_PersistentApp.default, null);
}
}, _temp)) || _class);
exports.default = CallPreview;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../../../../src/components/views/voip/CallPreview.tsx"],"names":["SHOW_CALL_IN_STATES","CallState","Connected","InviteSent","Connecting","CreateAnswer","CreateOffer","WaitLocalMedia","getPrimarySecondaryCalls","calls","primary","secondaries","call","includes","state","isRemoteOnHold","push","length","slice","console","log","CallPreview","React","Component","constructor","props","payload","RoomViewStore","getRoomId","roomId","primaryCall","secondaryCalls","CallHandler","sharedInstance","getAllActiveCallsNotInRoom","setState","secondaryCall","action","updateCalls","componentDidMount","addListener","CallHandlerEvent","CallChangeRoom","roomStoreToken","onRoomViewStoreUpdate","dispatcherRef","dis","register","onAction","MatrixClientPeg","get","on","CallEvent","RemoteHoldUnhold","onCallRemoteHold","componentWillUnmount","removeListener","remove","unregister","SettingsStore","unwatchSetting","settingsWatcherRef","render"],"mappings":"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","sourcesContent":["/*\nCopyright 2017, 2018 New Vector Ltd\nCopyright 2019, 2020 The Matrix.org Foundation C.I.C.\n\nLicensed under the Apache License, Version 2.0 (the \"License\");\nyou may not use this file except in compliance with the License.\nYou may obtain a copy of the License at\n\n    http://www.apache.org/licenses/LICENSE-2.0\n\nUnless required by applicable law or agreed to in writing, software\ndistributed under the License is distributed on an \"AS IS\" BASIS,\nWITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.\nSee the License for the specific language governing permissions and\nlimitations under the License.\n*/\n\nimport React from 'react';\n\nimport CallView from \"./CallView\";\nimport RoomViewStore from '../../../stores/RoomViewStore';\nimport CallHandler, { CallHandlerEvent } from '../../../CallHandler';\nimport dis from '../../../dispatcher/dispatcher';\nimport { ActionPayload } from '../../../dispatcher/payloads';\nimport PersistentApp from \"../elements/PersistentApp\";\nimport SettingsStore from \"../../../settings/SettingsStore\";\nimport { CallEvent, CallState, MatrixCall } from 'matrix-js-sdk/src/webrtc/call';\nimport { MatrixClientPeg } from '../../../MatrixClientPeg';\nimport {replaceableComponent} from \"../../../utils/replaceableComponent\";\n\nconst SHOW_CALL_IN_STATES = [\n    CallState.Connected,\n    CallState.InviteSent,\n    CallState.Connecting,\n    CallState.CreateAnswer,\n    CallState.CreateOffer,\n    CallState.WaitLocalMedia,\n];\n\ninterface IProps {\n}\n\ninterface IState {\n    roomId: string;\n\n    // The main call that we are displaying (ie. not including the call in the room being viewed, if any)\n    primaryCall: MatrixCall;\n\n    // Any other call we're displaying: only if the user is on two calls and not viewing either of the rooms\n    // they belong to\n    secondaryCall: MatrixCall;\n}\n\n// Splits a list of calls into one 'primary' one and a list\n// (which should be a single element) of other calls.\n// The primary will be the one not on hold, or an arbitrary one\n// if they're all on hold)\nfunction getPrimarySecondaryCalls(calls: MatrixCall[]): [MatrixCall, MatrixCall[]] {\n    let primary: MatrixCall = null;\n    let secondaries: MatrixCall[] = [];\n\n    for (const call of calls) {\n        if (!SHOW_CALL_IN_STATES.includes(call.state)) continue;\n\n        if (!call.isRemoteOnHold() && primary === null) {\n            primary = call;\n        } else {\n            secondaries.push(call);\n        }\n    }\n\n    if (primary === null && secondaries.length > 0) {\n        primary = secondaries[0];\n        secondaries = secondaries.slice(1);\n    }\n\n    if (secondaries.length > 1) {\n        // We should never be in more than two calls so this shouldn't happen\n        console.log(\"Found more than 1 secondary call! Other calls will not be shown.\");\n    }\n\n    return [primary, secondaries];\n}\n\n/**\n * CallPreview shows a small version of CallView hovering over the UI in 'picture-in-picture'\n * (PiP mode). It displays the call(s) which is *not* in the room the user is currently viewing.\n */\n@replaceableComponent(\"views.voip.CallPreview\")\nexport default class CallPreview extends React.Component<IProps, IState> {\n    private roomStoreToken: any;\n    private dispatcherRef: string;\n    private settingsWatcherRef: string;\n\n    constructor(props: IProps) {\n        super(props);\n\n        const roomId = RoomViewStore.getRoomId();\n\n        const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(\n            CallHandler.sharedInstance().getAllActiveCallsNotInRoom(roomId),\n        );\n\n        this.state = {\n            roomId,\n            primaryCall: primaryCall,\n            secondaryCall: secondaryCalls[0],\n        };\n    }\n\n    public componentDidMount() {\n        CallHandler.sharedInstance().addListener(CallHandlerEvent.CallChangeRoom, this.updateCalls);\n        this.roomStoreToken = RoomViewStore.addListener(this.onRoomViewStoreUpdate);\n        this.dispatcherRef = dis.register(this.onAction);\n        MatrixClientPeg.get().on(CallEvent.RemoteHoldUnhold, this.onCallRemoteHold);\n    }\n\n    public componentWillUnmount() {\n        CallHandler.sharedInstance().removeListener(CallHandlerEvent.CallChangeRoom, this.updateCalls);\n        MatrixClientPeg.get().removeListener(CallEvent.RemoteHoldUnhold, this.onCallRemoteHold);\n        if (this.roomStoreToken) {\n            this.roomStoreToken.remove();\n        }\n        dis.unregister(this.dispatcherRef);\n        SettingsStore.unwatchSetting(this.settingsWatcherRef);\n    }\n\n    private onRoomViewStoreUpdate = (payload) => {\n        if (RoomViewStore.getRoomId() === this.state.roomId) return;\n\n        const roomId = RoomViewStore.getRoomId();\n        const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(\n            CallHandler.sharedInstance().getAllActiveCallsNotInRoom(roomId),\n        );\n\n        this.setState({\n            roomId,\n            primaryCall: primaryCall,\n            secondaryCall: secondaryCalls[0],\n        });\n    };\n\n    private onAction = (payload: ActionPayload) => {\n        switch (payload.action) {\n            // listen for call state changes to prod the render method, which\n            // may hide the global CallView if the call it is tracking is dead\n            case 'call_state': {\n                this.updateCalls();\n                break;\n            }\n        }\n    };\n\n    private updateCalls = () => {\n        const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(\n            CallHandler.sharedInstance().getAllActiveCallsNotInRoom(this.state.roomId),\n        );\n\n        this.setState({\n            primaryCall: primaryCall,\n            secondaryCall: secondaryCalls[0],\n        });\n    };\n\n    private onCallRemoteHold = () => {\n        const [primaryCall, secondaryCalls] = getPrimarySecondaryCalls(\n            CallHandler.sharedInstance().getAllActiveCallsNotInRoom(this.state.roomId),\n        );\n\n        this.setState({\n            primaryCall: primaryCall,\n            secondaryCall: secondaryCalls[0],\n        });\n    }\n\n    public render() {\n        if (this.state.primaryCall) {\n            return (\n                <CallView call={this.state.primaryCall} secondaryCall={this.state.secondaryCall} pipMode={true} />\n            );\n        }\n\n        return <PersistentApp />;\n    }\n}\n\n"]}