@unwanted/matrix-sdk-mini 34.12.0-1 → 34.12.0-2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/git-revision.txt +1 -1
- package/lib/@types/event.d.ts +0 -19
- package/lib/@types/event.d.ts.map +1 -1
- package/lib/@types/event.js.map +1 -1
- package/lib/client.d.ts +2 -50
- package/lib/client.d.ts.map +1 -1
- package/lib/client.js +391 -501
- package/lib/client.js.map +1 -1
- package/lib/embedded.d.ts.map +1 -1
- package/lib/embedded.js +0 -1
- package/lib/embedded.js.map +1 -1
- package/lib/matrix.d.ts +0 -6
- package/lib/matrix.d.ts.map +1 -1
- package/lib/matrix.js +1 -5
- package/lib/matrix.js.map +1 -1
- package/package.json +1 -1
- package/src/@types/event.ts +2 -36
- package/src/client.ts +1 -150
- package/src/embedded.ts +0 -2
- package/src/matrix.ts +0 -13
- package/lib/matrixrtc/CallMembership.d.ts +0 -66
- package/lib/matrixrtc/CallMembership.d.ts.map +0 -1
- package/lib/matrixrtc/CallMembership.js +0 -197
- package/lib/matrixrtc/CallMembership.js.map +0 -1
- package/lib/matrixrtc/LivekitFocus.d.ts +0 -16
- package/lib/matrixrtc/LivekitFocus.d.ts.map +0 -1
- package/lib/matrixrtc/LivekitFocus.js +0 -20
- package/lib/matrixrtc/LivekitFocus.js.map +0 -1
- package/lib/matrixrtc/MatrixRTCSession.d.ts +0 -295
- package/lib/matrixrtc/MatrixRTCSession.d.ts.map +0 -1
- package/lib/matrixrtc/MatrixRTCSession.js +0 -1043
- package/lib/matrixrtc/MatrixRTCSession.js.map +0 -1
- package/lib/matrixrtc/MatrixRTCSessionManager.d.ts +0 -40
- package/lib/matrixrtc/MatrixRTCSessionManager.d.ts.map +0 -1
- package/lib/matrixrtc/MatrixRTCSessionManager.js +0 -146
- package/lib/matrixrtc/MatrixRTCSessionManager.js.map +0 -1
- package/lib/matrixrtc/focus.d.ts +0 -10
- package/lib/matrixrtc/focus.d.ts.map +0 -1
- package/lib/matrixrtc/focus.js +0 -1
- package/lib/matrixrtc/focus.js.map +0 -1
- package/lib/matrixrtc/index.d.ts +0 -7
- package/lib/matrixrtc/index.d.ts.map +0 -1
- package/lib/matrixrtc/index.js +0 -21
- package/lib/matrixrtc/index.js.map +0 -1
- package/lib/matrixrtc/types.d.ts +0 -19
- package/lib/matrixrtc/types.d.ts.map +0 -1
- package/lib/matrixrtc/types.js +0 -1
- package/lib/matrixrtc/types.js.map +0 -1
- package/lib/webrtc/audioContext.d.ts +0 -15
- package/lib/webrtc/audioContext.d.ts.map +0 -1
- package/lib/webrtc/audioContext.js +0 -46
- package/lib/webrtc/audioContext.js.map +0 -1
- package/lib/webrtc/call.d.ts +0 -560
- package/lib/webrtc/call.d.ts.map +0 -1
- package/lib/webrtc/call.js +0 -2541
- package/lib/webrtc/call.js.map +0 -1
- package/lib/webrtc/callEventHandler.d.ts +0 -37
- package/lib/webrtc/callEventHandler.d.ts.map +0 -1
- package/lib/webrtc/callEventHandler.js +0 -344
- package/lib/webrtc/callEventHandler.js.map +0 -1
- package/lib/webrtc/callEventTypes.d.ts +0 -73
- package/lib/webrtc/callEventTypes.d.ts.map +0 -1
- package/lib/webrtc/callEventTypes.js +0 -13
- package/lib/webrtc/callEventTypes.js.map +0 -1
- package/lib/webrtc/callFeed.d.ts +0 -128
- package/lib/webrtc/callFeed.d.ts.map +0 -1
- package/lib/webrtc/callFeed.js +0 -289
- package/lib/webrtc/callFeed.js.map +0 -1
- package/lib/webrtc/groupCall.d.ts +0 -323
- package/lib/webrtc/groupCall.d.ts.map +0 -1
- package/lib/webrtc/groupCall.js +0 -1337
- package/lib/webrtc/groupCall.js.map +0 -1
- package/lib/webrtc/groupCallEventHandler.d.ts +0 -31
- package/lib/webrtc/groupCallEventHandler.d.ts.map +0 -1
- package/lib/webrtc/groupCallEventHandler.js +0 -178
- package/lib/webrtc/groupCallEventHandler.js.map +0 -1
- package/lib/webrtc/mediaHandler.d.ts +0 -89
- package/lib/webrtc/mediaHandler.d.ts.map +0 -1
- package/lib/webrtc/mediaHandler.js +0 -437
- package/lib/webrtc/mediaHandler.js.map +0 -1
- package/lib/webrtc/stats/callFeedStatsReporter.d.ts +0 -8
- package/lib/webrtc/stats/callFeedStatsReporter.d.ts.map +0 -1
- package/lib/webrtc/stats/callFeedStatsReporter.js +0 -82
- package/lib/webrtc/stats/callFeedStatsReporter.js.map +0 -1
- package/lib/webrtc/stats/callStatsReportGatherer.d.ts +0 -25
- package/lib/webrtc/stats/callStatsReportGatherer.d.ts.map +0 -1
- package/lib/webrtc/stats/callStatsReportGatherer.js +0 -199
- package/lib/webrtc/stats/callStatsReportGatherer.js.map +0 -1
- package/lib/webrtc/stats/callStatsReportSummary.d.ts +0 -17
- package/lib/webrtc/stats/callStatsReportSummary.d.ts.map +0 -1
- package/lib/webrtc/stats/callStatsReportSummary.js +0 -1
- package/lib/webrtc/stats/callStatsReportSummary.js.map +0 -1
- package/lib/webrtc/stats/connectionStats.d.ts +0 -28
- package/lib/webrtc/stats/connectionStats.d.ts.map +0 -1
- package/lib/webrtc/stats/connectionStats.js +0 -26
- package/lib/webrtc/stats/connectionStats.js.map +0 -1
- package/lib/webrtc/stats/connectionStatsBuilder.d.ts +0 -5
- package/lib/webrtc/stats/connectionStatsBuilder.d.ts.map +0 -1
- package/lib/webrtc/stats/connectionStatsBuilder.js +0 -27
- package/lib/webrtc/stats/connectionStatsBuilder.js.map +0 -1
- package/lib/webrtc/stats/connectionStatsReportBuilder.d.ts +0 -7
- package/lib/webrtc/stats/connectionStatsReportBuilder.d.ts.map +0 -1
- package/lib/webrtc/stats/connectionStatsReportBuilder.js +0 -121
- package/lib/webrtc/stats/connectionStatsReportBuilder.js.map +0 -1
- package/lib/webrtc/stats/groupCallStats.d.ts +0 -22
- package/lib/webrtc/stats/groupCallStats.d.ts.map +0 -1
- package/lib/webrtc/stats/groupCallStats.js +0 -78
- package/lib/webrtc/stats/groupCallStats.js.map +0 -1
- package/lib/webrtc/stats/media/mediaSsrcHandler.d.ts +0 -10
- package/lib/webrtc/stats/media/mediaSsrcHandler.d.ts.map +0 -1
- package/lib/webrtc/stats/media/mediaSsrcHandler.js +0 -57
- package/lib/webrtc/stats/media/mediaSsrcHandler.js.map +0 -1
- package/lib/webrtc/stats/media/mediaTrackHandler.d.ts +0 -12
- package/lib/webrtc/stats/media/mediaTrackHandler.d.ts.map +0 -1
- package/lib/webrtc/stats/media/mediaTrackHandler.js +0 -62
- package/lib/webrtc/stats/media/mediaTrackHandler.js.map +0 -1
- package/lib/webrtc/stats/media/mediaTrackStats.d.ts +0 -86
- package/lib/webrtc/stats/media/mediaTrackStats.d.ts.map +0 -1
- package/lib/webrtc/stats/media/mediaTrackStats.js +0 -142
- package/lib/webrtc/stats/media/mediaTrackStats.js.map +0 -1
- package/lib/webrtc/stats/media/mediaTrackStatsHandler.d.ts +0 -22
- package/lib/webrtc/stats/media/mediaTrackStatsHandler.d.ts.map +0 -1
- package/lib/webrtc/stats/media/mediaTrackStatsHandler.js +0 -76
- package/lib/webrtc/stats/media/mediaTrackStatsHandler.js.map +0 -1
- package/lib/webrtc/stats/statsReport.d.ts +0 -99
- package/lib/webrtc/stats/statsReport.d.ts.map +0 -1
- package/lib/webrtc/stats/statsReport.js +0 -32
- package/lib/webrtc/stats/statsReport.js.map +0 -1
- package/lib/webrtc/stats/statsReportEmitter.d.ts +0 -15
- package/lib/webrtc/stats/statsReportEmitter.d.ts.map +0 -1
- package/lib/webrtc/stats/statsReportEmitter.js +0 -33
- package/lib/webrtc/stats/statsReportEmitter.js.map +0 -1
- package/lib/webrtc/stats/summaryStatsReportGatherer.d.ts +0 -16
- package/lib/webrtc/stats/summaryStatsReportGatherer.d.ts.map +0 -1
- package/lib/webrtc/stats/summaryStatsReportGatherer.js +0 -116
- package/lib/webrtc/stats/summaryStatsReportGatherer.js.map +0 -1
- package/lib/webrtc/stats/trackStatsBuilder.d.ts +0 -19
- package/lib/webrtc/stats/trackStatsBuilder.d.ts.map +0 -1
- package/lib/webrtc/stats/trackStatsBuilder.js +0 -168
- package/lib/webrtc/stats/trackStatsBuilder.js.map +0 -1
- package/lib/webrtc/stats/transportStats.d.ts +0 -11
- package/lib/webrtc/stats/transportStats.d.ts.map +0 -1
- package/lib/webrtc/stats/transportStats.js +0 -1
- package/lib/webrtc/stats/transportStats.js.map +0 -1
- package/lib/webrtc/stats/transportStatsBuilder.d.ts +0 -5
- package/lib/webrtc/stats/transportStatsBuilder.d.ts.map +0 -1
- package/lib/webrtc/stats/transportStatsBuilder.js +0 -34
- package/lib/webrtc/stats/transportStatsBuilder.js.map +0 -1
- package/lib/webrtc/stats/valueFormatter.d.ts +0 -4
- package/lib/webrtc/stats/valueFormatter.d.ts.map +0 -1
- package/lib/webrtc/stats/valueFormatter.js +0 -25
- package/lib/webrtc/stats/valueFormatter.js.map +0 -1
- package/src/matrixrtc/CallMembership.ts +0 -247
- package/src/matrixrtc/LivekitFocus.ts +0 -39
- package/src/matrixrtc/MatrixRTCSession.ts +0 -1319
- package/src/matrixrtc/MatrixRTCSessionManager.ts +0 -166
- package/src/matrixrtc/focus.ts +0 -25
- package/src/matrixrtc/index.ts +0 -22
- package/src/matrixrtc/types.ts +0 -36
- package/src/webrtc/audioContext.ts +0 -44
- package/src/webrtc/call.ts +0 -3074
- package/src/webrtc/callEventHandler.ts +0 -425
- package/src/webrtc/callEventTypes.ts +0 -93
- package/src/webrtc/callFeed.ts +0 -364
- package/src/webrtc/groupCall.ts +0 -1735
- package/src/webrtc/groupCallEventHandler.ts +0 -234
- package/src/webrtc/mediaHandler.ts +0 -484
- package/src/webrtc/stats/callFeedStatsReporter.ts +0 -94
- package/src/webrtc/stats/callStatsReportGatherer.ts +0 -219
- package/src/webrtc/stats/callStatsReportSummary.ts +0 -30
- package/src/webrtc/stats/connectionStats.ts +0 -47
- package/src/webrtc/stats/connectionStatsBuilder.ts +0 -28
- package/src/webrtc/stats/connectionStatsReportBuilder.ts +0 -140
- package/src/webrtc/stats/groupCallStats.ts +0 -93
- package/src/webrtc/stats/media/mediaSsrcHandler.ts +0 -57
- package/src/webrtc/stats/media/mediaTrackHandler.ts +0 -76
- package/src/webrtc/stats/media/mediaTrackStats.ts +0 -176
- package/src/webrtc/stats/media/mediaTrackStatsHandler.ts +0 -90
- package/src/webrtc/stats/statsReport.ts +0 -133
- package/src/webrtc/stats/statsReportEmitter.ts +0 -49
- package/src/webrtc/stats/summaryStatsReportGatherer.ts +0 -148
- package/src/webrtc/stats/trackStatsBuilder.ts +0 -207
- package/src/webrtc/stats/transportStats.ts +0 -26
- package/src/webrtc/stats/transportStatsBuilder.ts +0 -48
- package/src/webrtc/stats/valueFormatter.ts +0 -27
package/lib/webrtc/groupCall.js
DELETED
@@ -1,1337 +0,0 @@
|
|
1
|
-
import _asyncToGenerator from "@babel/runtime/helpers/asyncToGenerator";
|
2
|
-
import _defineProperty from "@babel/runtime/helpers/defineProperty";
|
3
|
-
function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
|
4
|
-
function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
|
5
|
-
import { TypedEventEmitter } from "../models/typed-event-emitter.js";
|
6
|
-
import { CallFeed, SPEAKING_THRESHOLD } from "./callFeed.js";
|
7
|
-
import { CallErrorCode, CallEvent, CallState, genCallID, setTracksEnabled, createNewMatrixCall, CallError } from "./call.js";
|
8
|
-
import { RoomStateEvent } from "../models/room-state.js";
|
9
|
-
import { logger } from "../logger.js";
|
10
|
-
import { ReEmitter } from "../ReEmitter.js";
|
11
|
-
import { SDPStreamMetadataPurpose } from "./callEventTypes.js";
|
12
|
-
import { EventType } from "../@types/event.js";
|
13
|
-
import { CallEventHandlerEvent } from "./callEventHandler.js";
|
14
|
-
import { GroupCallEventHandlerEvent } from "./groupCallEventHandler.js";
|
15
|
-
import { mapsEqual } from "../utils.js";
|
16
|
-
import { GroupCallStats } from "./stats/groupCallStats.js";
|
17
|
-
import { StatsReport } from "./stats/statsReport.js";
|
18
|
-
import { SummaryStatsReportGatherer } from "./stats/summaryStatsReportGatherer.js";
|
19
|
-
import { CallFeedStatsReporter } from "./stats/callFeedStatsReporter.js";
|
20
|
-
import { KnownMembership } from "../@types/membership.js";
|
21
|
-
export var GroupCallIntent = /*#__PURE__*/function (GroupCallIntent) {
|
22
|
-
GroupCallIntent["Ring"] = "m.ring";
|
23
|
-
GroupCallIntent["Prompt"] = "m.prompt";
|
24
|
-
GroupCallIntent["Room"] = "m.room";
|
25
|
-
return GroupCallIntent;
|
26
|
-
}({});
|
27
|
-
export var GroupCallType = /*#__PURE__*/function (GroupCallType) {
|
28
|
-
GroupCallType["Video"] = "m.video";
|
29
|
-
GroupCallType["Voice"] = "m.voice";
|
30
|
-
return GroupCallType;
|
31
|
-
}({});
|
32
|
-
export var GroupCallTerminationReason = /*#__PURE__*/function (GroupCallTerminationReason) {
|
33
|
-
GroupCallTerminationReason["CallEnded"] = "call_ended";
|
34
|
-
return GroupCallTerminationReason;
|
35
|
-
}({});
|
36
|
-
/**
|
37
|
-
* Because event names are just strings, they do need
|
38
|
-
* to be unique over all event types of event emitter.
|
39
|
-
* Some objects could emit more then one set of events.
|
40
|
-
*/
|
41
|
-
export var GroupCallEvent = /*#__PURE__*/function (GroupCallEvent) {
|
42
|
-
GroupCallEvent["GroupCallStateChanged"] = "group_call_state_changed";
|
43
|
-
GroupCallEvent["ActiveSpeakerChanged"] = "active_speaker_changed";
|
44
|
-
GroupCallEvent["CallsChanged"] = "calls_changed";
|
45
|
-
GroupCallEvent["UserMediaFeedsChanged"] = "user_media_feeds_changed";
|
46
|
-
GroupCallEvent["ScreenshareFeedsChanged"] = "screenshare_feeds_changed";
|
47
|
-
GroupCallEvent["LocalScreenshareStateChanged"] = "local_screenshare_state_changed";
|
48
|
-
GroupCallEvent["LocalMuteStateChanged"] = "local_mute_state_changed";
|
49
|
-
GroupCallEvent["ParticipantsChanged"] = "participants_changed";
|
50
|
-
GroupCallEvent["Error"] = "group_call_error";
|
51
|
-
return GroupCallEvent;
|
52
|
-
}({});
|
53
|
-
export var GroupCallStatsReportEvent = /*#__PURE__*/function (GroupCallStatsReportEvent) {
|
54
|
-
GroupCallStatsReportEvent["ConnectionStats"] = "GroupCall.connection_stats";
|
55
|
-
GroupCallStatsReportEvent["ByteSentStats"] = "GroupCall.byte_sent_stats";
|
56
|
-
GroupCallStatsReportEvent["SummaryStats"] = "GroupCall.summary_stats";
|
57
|
-
GroupCallStatsReportEvent["CallFeedStats"] = "GroupCall.call_feed_stats";
|
58
|
-
return GroupCallStatsReportEvent;
|
59
|
-
}({});
|
60
|
-
|
61
|
-
/**
|
62
|
-
* The final report-events that get consumed by client.
|
63
|
-
*/
|
64
|
-
|
65
|
-
export var GroupCallErrorCode = /*#__PURE__*/function (GroupCallErrorCode) {
|
66
|
-
GroupCallErrorCode["NoUserMedia"] = "no_user_media";
|
67
|
-
GroupCallErrorCode["UnknownDevice"] = "unknown_device";
|
68
|
-
GroupCallErrorCode["PlaceCallFailed"] = "place_call_failed";
|
69
|
-
return GroupCallErrorCode;
|
70
|
-
}({});
|
71
|
-
export class GroupCallError extends Error {
|
72
|
-
constructor(code, msg, err) {
|
73
|
-
// Still don't think there's any way to have proper nested errors
|
74
|
-
if (err) {
|
75
|
-
super(msg + ": " + err);
|
76
|
-
_defineProperty(this, "code", void 0);
|
77
|
-
} else {
|
78
|
-
super(msg);
|
79
|
-
_defineProperty(this, "code", void 0);
|
80
|
-
}
|
81
|
-
this.code = code;
|
82
|
-
}
|
83
|
-
}
|
84
|
-
export class GroupCallUnknownDeviceError extends GroupCallError {
|
85
|
-
constructor(userId) {
|
86
|
-
super(GroupCallErrorCode.UnknownDevice, "No device found for " + userId);
|
87
|
-
this.userId = userId;
|
88
|
-
}
|
89
|
-
}
|
90
|
-
export class OtherUserSpeakingError extends Error {
|
91
|
-
constructor() {
|
92
|
-
super("Cannot unmute: another user is speaking");
|
93
|
-
}
|
94
|
-
}
|
95
|
-
|
96
|
-
// XXX: this hasn't made it into the MSC yet
|
97
|
-
|
98
|
-
export var GroupCallState = /*#__PURE__*/function (GroupCallState) {
|
99
|
-
GroupCallState["LocalCallFeedUninitialized"] = "local_call_feed_uninitialized";
|
100
|
-
GroupCallState["InitializingLocalCallFeed"] = "initializing_local_call_feed";
|
101
|
-
GroupCallState["LocalCallFeedInitialized"] = "local_call_feed_initialized";
|
102
|
-
GroupCallState["Entered"] = "entered";
|
103
|
-
GroupCallState["Ended"] = "ended";
|
104
|
-
return GroupCallState;
|
105
|
-
}({});
|
106
|
-
var DEVICE_TIMEOUT = 1000 * 60 * 60; // 1 hour
|
107
|
-
|
108
|
-
function getCallUserId(call) {
|
109
|
-
var _call$getOpponentMemb;
|
110
|
-
return ((_call$getOpponentMemb = call.getOpponentMember()) === null || _call$getOpponentMemb === void 0 ? void 0 : _call$getOpponentMemb.userId) || call.invitee || null;
|
111
|
-
}
|
112
|
-
export class GroupCall extends TypedEventEmitter {
|
113
|
-
constructor(client, room, type, isPtt, intent, groupCallId, dataChannelsEnabled, dataChannelOptions, isCallWithoutVideoAndAudio) {
|
114
|
-
var _room$currentState$ge, _room$currentState$ge2;
|
115
|
-
var useLivekit = arguments.length > 9 && arguments[9] !== undefined ? arguments[9] : false;
|
116
|
-
var livekitServiceURL = arguments.length > 10 ? arguments[10] : undefined;
|
117
|
-
super();
|
118
|
-
this.client = client;
|
119
|
-
this.room = room;
|
120
|
-
this.type = type;
|
121
|
-
this.isPtt = isPtt;
|
122
|
-
this.intent = intent;
|
123
|
-
this.dataChannelsEnabled = dataChannelsEnabled;
|
124
|
-
this.dataChannelOptions = dataChannelOptions;
|
125
|
-
this.useLivekit = useLivekit;
|
126
|
-
// Config
|
127
|
-
_defineProperty(this, "activeSpeakerInterval", 1000);
|
128
|
-
_defineProperty(this, "retryCallInterval", 5000);
|
129
|
-
_defineProperty(this, "participantTimeout", 1000 * 15);
|
130
|
-
_defineProperty(this, "pttMaxTransmitTime", 1000 * 20);
|
131
|
-
_defineProperty(this, "activeSpeaker", void 0);
|
132
|
-
_defineProperty(this, "localCallFeed", void 0);
|
133
|
-
_defineProperty(this, "localScreenshareFeed", void 0);
|
134
|
-
_defineProperty(this, "localDesktopCapturerSourceId", void 0);
|
135
|
-
_defineProperty(this, "userMediaFeeds", []);
|
136
|
-
_defineProperty(this, "screenshareFeeds", []);
|
137
|
-
_defineProperty(this, "groupCallId", void 0);
|
138
|
-
_defineProperty(this, "allowCallWithoutVideoAndAudio", void 0);
|
139
|
-
_defineProperty(this, "calls", new Map());
|
140
|
-
// user_id -> device_id -> MatrixCall
|
141
|
-
_defineProperty(this, "callHandlers", new Map());
|
142
|
-
// user_id -> device_id -> ICallHandlers
|
143
|
-
_defineProperty(this, "activeSpeakerLoopInterval", void 0);
|
144
|
-
_defineProperty(this, "retryCallLoopInterval", void 0);
|
145
|
-
_defineProperty(this, "retryCallCounts", new Map());
|
146
|
-
// user_id -> device_id -> count
|
147
|
-
_defineProperty(this, "reEmitter", void 0);
|
148
|
-
_defineProperty(this, "transmitTimer", null);
|
149
|
-
_defineProperty(this, "participantsExpirationTimer", null);
|
150
|
-
_defineProperty(this, "resendMemberStateTimer", null);
|
151
|
-
_defineProperty(this, "initWithAudioMuted", false);
|
152
|
-
_defineProperty(this, "initWithVideoMuted", false);
|
153
|
-
_defineProperty(this, "initCallFeedPromise", void 0);
|
154
|
-
_defineProperty(this, "_livekitServiceURL", void 0);
|
155
|
-
_defineProperty(this, "stats", void 0);
|
156
|
-
/**
|
157
|
-
* Configure default webrtc stats collection interval in ms
|
158
|
-
* Disable collecting webrtc stats by setting interval to 0
|
159
|
-
*/
|
160
|
-
_defineProperty(this, "statsCollectIntervalTime", 0);
|
161
|
-
_defineProperty(this, "onConnectionStats", report => {
|
162
|
-
// Final emit of the summary event, to be consumed by the client
|
163
|
-
this.emit(GroupCallStatsReportEvent.ConnectionStats, {
|
164
|
-
report
|
165
|
-
});
|
166
|
-
});
|
167
|
-
_defineProperty(this, "onByteSentStats", report => {
|
168
|
-
// Final emit of the summary event, to be consumed by the client
|
169
|
-
this.emit(GroupCallStatsReportEvent.ByteSentStats, {
|
170
|
-
report
|
171
|
-
});
|
172
|
-
});
|
173
|
-
_defineProperty(this, "onSummaryStats", report => {
|
174
|
-
SummaryStatsReportGatherer.extendSummaryReport(report, this.participants);
|
175
|
-
// Final emit of the summary event, to be consumed by the client
|
176
|
-
this.emit(GroupCallStatsReportEvent.SummaryStats, {
|
177
|
-
report
|
178
|
-
});
|
179
|
-
});
|
180
|
-
_defineProperty(this, "onCallFeedReport", report => {
|
181
|
-
if (this.localCallFeed) {
|
182
|
-
report = CallFeedStatsReporter.expandCallFeedReport(report, [this.localCallFeed], "from-local-feed");
|
183
|
-
}
|
184
|
-
var callFeeds = [];
|
185
|
-
this.forEachCall(call => {
|
186
|
-
if (call.callId === report.callId) {
|
187
|
-
call.getFeeds().forEach(f => callFeeds.push(f));
|
188
|
-
}
|
189
|
-
});
|
190
|
-
report = CallFeedStatsReporter.expandCallFeedReport(report, callFeeds, "from-call-feed");
|
191
|
-
this.emit(GroupCallStatsReportEvent.CallFeedStats, {
|
192
|
-
report
|
193
|
-
});
|
194
|
-
});
|
195
|
-
_defineProperty(this, "_state", GroupCallState.LocalCallFeedUninitialized);
|
196
|
-
_defineProperty(this, "_participants", new Map());
|
197
|
-
_defineProperty(this, "_creationTs", null);
|
198
|
-
_defineProperty(this, "_enteredViaAnotherSession", false);
|
199
|
-
/*
|
200
|
-
* Call Setup
|
201
|
-
*
|
202
|
-
* There are two different paths for calls to be created:
|
203
|
-
* 1. Incoming calls triggered by the Call.incoming event.
|
204
|
-
* 2. Outgoing calls to the initial members of a room or new members
|
205
|
-
* as they are observed by the RoomState.members event.
|
206
|
-
*/
|
207
|
-
_defineProperty(this, "onIncomingCall", newCall => {
|
208
|
-
var _newCall$getOpponentM, _this$calls$get;
|
209
|
-
// The incoming calls may be for another room, which we will ignore.
|
210
|
-
if (newCall.roomId !== this.room.roomId) {
|
211
|
-
return;
|
212
|
-
}
|
213
|
-
if (newCall.state !== CallState.Ringing) {
|
214
|
-
logger.warn("GroupCall ".concat(this.groupCallId, " onIncomingCall() incoming call no longer in ringing state - ignoring"));
|
215
|
-
return;
|
216
|
-
}
|
217
|
-
if (!newCall.groupCallId || newCall.groupCallId !== this.groupCallId) {
|
218
|
-
logger.log("GroupCall ".concat(this.groupCallId, " onIncomingCall() ignored because it doesn't match the current group call"));
|
219
|
-
newCall.reject();
|
220
|
-
return;
|
221
|
-
}
|
222
|
-
var opponentUserId = (_newCall$getOpponentM = newCall.getOpponentMember()) === null || _newCall$getOpponentM === void 0 ? void 0 : _newCall$getOpponentM.userId;
|
223
|
-
if (opponentUserId === undefined) {
|
224
|
-
logger.warn("GroupCall ".concat(this.groupCallId, " onIncomingCall() incoming call with no member - ignoring"));
|
225
|
-
return;
|
226
|
-
}
|
227
|
-
if (this.useLivekit) {
|
228
|
-
logger.info("Received incoming call whilst in signaling-only mode! Ignoring.");
|
229
|
-
return;
|
230
|
-
}
|
231
|
-
var deviceMap = (_this$calls$get = this.calls.get(opponentUserId)) !== null && _this$calls$get !== void 0 ? _this$calls$get : new Map();
|
232
|
-
var prevCall = deviceMap.get(newCall.getOpponentDeviceId());
|
233
|
-
if ((prevCall === null || prevCall === void 0 ? void 0 : prevCall.callId) === newCall.callId) return;
|
234
|
-
logger.log("GroupCall ".concat(this.groupCallId, " onIncomingCall() incoming call (userId=").concat(opponentUserId, ", callId=").concat(newCall.callId, ")"));
|
235
|
-
if (prevCall) prevCall.hangup(CallErrorCode.Replaced, false);
|
236
|
-
// We must do this before we start initialising / answering the call as we
|
237
|
-
// need to know it is the active call for this user+deviceId and to not ignore
|
238
|
-
// events from it.
|
239
|
-
deviceMap.set(newCall.getOpponentDeviceId(), newCall);
|
240
|
-
this.calls.set(opponentUserId, deviceMap);
|
241
|
-
this.initCall(newCall);
|
242
|
-
var feeds = this.getLocalFeeds().map(feed => feed.clone());
|
243
|
-
if (!this.callExpected(newCall)) {
|
244
|
-
// Disable our tracks for users not explicitly participating in the
|
245
|
-
// call but trying to receive the feeds
|
246
|
-
for (var feed of feeds) {
|
247
|
-
setTracksEnabled(feed.stream.getAudioTracks(), false);
|
248
|
-
setTracksEnabled(feed.stream.getVideoTracks(), false);
|
249
|
-
}
|
250
|
-
}
|
251
|
-
newCall.answerWithCallFeeds(feeds);
|
252
|
-
this.emit(GroupCallEvent.CallsChanged, this.calls);
|
253
|
-
});
|
254
|
-
_defineProperty(this, "onRetryCallLoop", () => {
|
255
|
-
var needsRetry = false;
|
256
|
-
for (var [{
|
257
|
-
userId: _userId
|
258
|
-
}, participantMap] of this.participants) {
|
259
|
-
var callMap = this.calls.get(_userId);
|
260
|
-
var retriesMap = this.retryCallCounts.get(_userId);
|
261
|
-
for (var [deviceId, participant] of participantMap) {
|
262
|
-
var _retriesMap$get, _retriesMap;
|
263
|
-
var call = callMap === null || callMap === void 0 ? void 0 : callMap.get(deviceId);
|
264
|
-
var retries = (_retriesMap$get = (_retriesMap = retriesMap) === null || _retriesMap === void 0 ? void 0 : _retriesMap.get(deviceId)) !== null && _retriesMap$get !== void 0 ? _retriesMap$get : 0;
|
265
|
-
if ((call === null || call === void 0 ? void 0 : call.getOpponentSessionId()) !== participant.sessionId && this.wantsOutgoingCall(_userId, deviceId) && retries < 3) {
|
266
|
-
if (retriesMap === undefined) {
|
267
|
-
retriesMap = new Map();
|
268
|
-
this.retryCallCounts.set(_userId, retriesMap);
|
269
|
-
}
|
270
|
-
retriesMap.set(deviceId, retries + 1);
|
271
|
-
needsRetry = true;
|
272
|
-
}
|
273
|
-
}
|
274
|
-
}
|
275
|
-
if (needsRetry) this.placeOutgoingCalls();
|
276
|
-
});
|
277
|
-
_defineProperty(this, "onCallFeedsChanged", call => {
|
278
|
-
var opponentMemberId = getCallUserId(call);
|
279
|
-
var opponentDeviceId = call.getOpponentDeviceId();
|
280
|
-
if (!opponentMemberId) {
|
281
|
-
throw new Error("Cannot change call feeds without user id");
|
282
|
-
}
|
283
|
-
var currentUserMediaFeed = this.getUserMediaFeed(opponentMemberId, opponentDeviceId);
|
284
|
-
var remoteUsermediaFeed = call.remoteUsermediaFeed;
|
285
|
-
var remoteFeedChanged = remoteUsermediaFeed !== currentUserMediaFeed;
|
286
|
-
var deviceMap = this.calls.get(opponentMemberId);
|
287
|
-
var currentCallForUserDevice = deviceMap === null || deviceMap === void 0 ? void 0 : deviceMap.get(opponentDeviceId);
|
288
|
-
if ((currentCallForUserDevice === null || currentCallForUserDevice === void 0 ? void 0 : currentCallForUserDevice.callId) !== call.callId) {
|
289
|
-
// the call in question is not the current call for this user/deviceId
|
290
|
-
// so ignore feed events from it otherwise we'll remove our real feeds
|
291
|
-
return;
|
292
|
-
}
|
293
|
-
if (remoteFeedChanged) {
|
294
|
-
if (!currentUserMediaFeed && remoteUsermediaFeed) {
|
295
|
-
this.addUserMediaFeed(remoteUsermediaFeed);
|
296
|
-
} else if (currentUserMediaFeed && remoteUsermediaFeed) {
|
297
|
-
this.replaceUserMediaFeed(currentUserMediaFeed, remoteUsermediaFeed);
|
298
|
-
} else if (currentUserMediaFeed && !remoteUsermediaFeed) {
|
299
|
-
this.removeUserMediaFeed(currentUserMediaFeed);
|
300
|
-
}
|
301
|
-
}
|
302
|
-
var currentScreenshareFeed = this.getScreenshareFeed(opponentMemberId, opponentDeviceId);
|
303
|
-
var remoteScreensharingFeed = call.remoteScreensharingFeed;
|
304
|
-
var remoteScreenshareFeedChanged = remoteScreensharingFeed !== currentScreenshareFeed;
|
305
|
-
if (remoteScreenshareFeedChanged) {
|
306
|
-
if (!currentScreenshareFeed && remoteScreensharingFeed) {
|
307
|
-
this.addScreenshareFeed(remoteScreensharingFeed);
|
308
|
-
} else if (currentScreenshareFeed && remoteScreensharingFeed) {
|
309
|
-
this.replaceScreenshareFeed(currentScreenshareFeed, remoteScreensharingFeed);
|
310
|
-
} else if (currentScreenshareFeed && !remoteScreensharingFeed) {
|
311
|
-
this.removeScreenshareFeed(currentScreenshareFeed);
|
312
|
-
}
|
313
|
-
}
|
314
|
-
});
|
315
|
-
_defineProperty(this, "onCallStateChanged", (call, state, _oldState) => {
|
316
|
-
var _call$getOpponentMemb2;
|
317
|
-
if (state === CallState.Ended) return;
|
318
|
-
var audioMuted = this.localCallFeed.isAudioMuted();
|
319
|
-
if (call.localUsermediaStream && call.isMicrophoneMuted() !== audioMuted) {
|
320
|
-
call.setMicrophoneMuted(audioMuted);
|
321
|
-
}
|
322
|
-
var videoMuted = this.localCallFeed.isVideoMuted();
|
323
|
-
if (call.localUsermediaStream && call.isLocalVideoMuted() !== videoMuted) {
|
324
|
-
call.setLocalVideoMuted(videoMuted);
|
325
|
-
}
|
326
|
-
var opponentUserId = (_call$getOpponentMemb2 = call.getOpponentMember()) === null || _call$getOpponentMemb2 === void 0 ? void 0 : _call$getOpponentMemb2.userId;
|
327
|
-
if (state === CallState.Connected && opponentUserId) {
|
328
|
-
var retriesMap = this.retryCallCounts.get(opponentUserId);
|
329
|
-
retriesMap === null || retriesMap === void 0 || retriesMap.delete(call.getOpponentDeviceId());
|
330
|
-
if ((retriesMap === null || retriesMap === void 0 ? void 0 : retriesMap.size) === 0) this.retryCallCounts.delete(opponentUserId);
|
331
|
-
}
|
332
|
-
});
|
333
|
-
_defineProperty(this, "onCallHangup", call => {
|
334
|
-
var _call$getOpponentMemb3, _call$getOpponentMemb4;
|
335
|
-
if (call.hangupReason === CallErrorCode.Replaced) return;
|
336
|
-
var opponentUserId = (_call$getOpponentMemb3 = (_call$getOpponentMemb4 = call.getOpponentMember()) === null || _call$getOpponentMemb4 === void 0 ? void 0 : _call$getOpponentMemb4.userId) !== null && _call$getOpponentMemb3 !== void 0 ? _call$getOpponentMemb3 : this.room.getMember(call.invitee).userId;
|
337
|
-
var deviceMap = this.calls.get(opponentUserId);
|
338
|
-
|
339
|
-
// Sanity check that this call is in fact in the map
|
340
|
-
if ((deviceMap === null || deviceMap === void 0 ? void 0 : deviceMap.get(call.getOpponentDeviceId())) === call) {
|
341
|
-
this.disposeCall(call, call.hangupReason);
|
342
|
-
deviceMap.delete(call.getOpponentDeviceId());
|
343
|
-
if (deviceMap.size === 0) this.calls.delete(opponentUserId);
|
344
|
-
this.emit(GroupCallEvent.CallsChanged, this.calls);
|
345
|
-
}
|
346
|
-
});
|
347
|
-
_defineProperty(this, "onCallReplaced", (prevCall, newCall) => {
|
348
|
-
var opponentUserId = prevCall.getOpponentMember().userId;
|
349
|
-
var deviceMap = this.calls.get(opponentUserId);
|
350
|
-
if (deviceMap === undefined) {
|
351
|
-
deviceMap = new Map();
|
352
|
-
this.calls.set(opponentUserId, deviceMap);
|
353
|
-
}
|
354
|
-
prevCall.hangup(CallErrorCode.Replaced, false);
|
355
|
-
this.initCall(newCall);
|
356
|
-
deviceMap.set(prevCall.getOpponentDeviceId(), newCall);
|
357
|
-
this.emit(GroupCallEvent.CallsChanged, this.calls);
|
358
|
-
});
|
359
|
-
_defineProperty(this, "onActiveSpeakerLoop", () => {
|
360
|
-
var topAvg = undefined;
|
361
|
-
var nextActiveSpeaker = undefined;
|
362
|
-
for (var callFeed of this.userMediaFeeds) {
|
363
|
-
if (callFeed.isLocal() && this.userMediaFeeds.length > 1) continue;
|
364
|
-
var total = callFeed.speakingVolumeSamples.reduce((acc, volume) => acc + Math.max(volume, SPEAKING_THRESHOLD));
|
365
|
-
var avg = total / callFeed.speakingVolumeSamples.length;
|
366
|
-
if (!topAvg || avg > topAvg) {
|
367
|
-
topAvg = avg;
|
368
|
-
nextActiveSpeaker = callFeed;
|
369
|
-
}
|
370
|
-
}
|
371
|
-
if (nextActiveSpeaker && this.activeSpeaker !== nextActiveSpeaker && topAvg && topAvg > SPEAKING_THRESHOLD) {
|
372
|
-
this.activeSpeaker = nextActiveSpeaker;
|
373
|
-
this.emit(GroupCallEvent.ActiveSpeakerChanged, this.activeSpeaker);
|
374
|
-
}
|
375
|
-
});
|
376
|
-
_defineProperty(this, "onRoomState", () => this.updateParticipants());
|
377
|
-
_defineProperty(this, "onParticipantsChanged", () => {
|
378
|
-
// Re-run setTracksEnabled on all calls, so that participants that just
|
379
|
-
// left get denied access to our media, and participants that just
|
380
|
-
// joined get granted access
|
381
|
-
this.forEachCall(call => {
|
382
|
-
var expected = this.callExpected(call);
|
383
|
-
for (var feed of call.getLocalFeeds()) {
|
384
|
-
setTracksEnabled(feed.stream.getAudioTracks(), !feed.isAudioMuted() && expected);
|
385
|
-
setTracksEnabled(feed.stream.getVideoTracks(), !feed.isVideoMuted() && expected);
|
386
|
-
}
|
387
|
-
});
|
388
|
-
if (this.state === GroupCallState.Entered && !this.useLivekit) this.placeOutgoingCalls();
|
389
|
-
|
390
|
-
// Update the participants stored in the stats object
|
391
|
-
});
|
392
|
-
_defineProperty(this, "onStateChanged", (newState, oldState) => {
|
393
|
-
if (newState === GroupCallState.Entered || oldState === GroupCallState.Entered || newState === GroupCallState.Ended) {
|
394
|
-
// We either entered, left, or ended the call
|
395
|
-
this.updateParticipants();
|
396
|
-
this.updateMemberState().catch(e => logger.error("GroupCall ".concat(this.groupCallId, " onStateChanged() failed to update member state devices\""), e));
|
397
|
-
}
|
398
|
-
});
|
399
|
-
_defineProperty(this, "onLocalFeedsChanged", () => {
|
400
|
-
if (this.state === GroupCallState.Entered) {
|
401
|
-
this.updateMemberState().catch(e => logger.error("GroupCall ".concat(this.groupCallId, " onLocalFeedsChanged() failed to update member state feeds"), e));
|
402
|
-
}
|
403
|
-
});
|
404
|
-
this.reEmitter = new ReEmitter(this);
|
405
|
-
this.groupCallId = groupCallId !== null && groupCallId !== void 0 ? groupCallId : genCallID();
|
406
|
-
this._livekitServiceURL = livekitServiceURL;
|
407
|
-
this.creationTs = (_room$currentState$ge = (_room$currentState$ge2 = room.currentState.getStateEvents(EventType.GroupCallPrefix, this.groupCallId)) === null || _room$currentState$ge2 === void 0 ? void 0 : _room$currentState$ge2.getTs()) !== null && _room$currentState$ge !== void 0 ? _room$currentState$ge : null;
|
408
|
-
this.updateParticipants();
|
409
|
-
room.on(RoomStateEvent.Update, this.onRoomState);
|
410
|
-
this.on(GroupCallEvent.ParticipantsChanged, this.onParticipantsChanged);
|
411
|
-
this.on(GroupCallEvent.GroupCallStateChanged, this.onStateChanged);
|
412
|
-
this.on(GroupCallEvent.LocalScreenshareStateChanged, this.onLocalFeedsChanged);
|
413
|
-
this.allowCallWithoutVideoAndAudio = !!isCallWithoutVideoAndAudio;
|
414
|
-
}
|
415
|
-
create() {
|
416
|
-
var _this = this;
|
417
|
-
return _asyncToGenerator(function* () {
|
418
|
-
_this.creationTs = Date.now();
|
419
|
-
_this.client.groupCallEventHandler.groupCalls.set(_this.room.roomId, _this);
|
420
|
-
_this.client.emit(GroupCallEventHandlerEvent.Outgoing, _this);
|
421
|
-
yield _this.sendCallStateEvent();
|
422
|
-
return _this;
|
423
|
-
})();
|
424
|
-
}
|
425
|
-
sendCallStateEvent() {
|
426
|
-
var _this2 = this;
|
427
|
-
return _asyncToGenerator(function* () {
|
428
|
-
var groupCallState = {
|
429
|
-
"m.intent": _this2.intent,
|
430
|
-
"m.type": _this2.type,
|
431
|
-
"io.element.ptt": _this2.isPtt,
|
432
|
-
// TODO: Specify data-channels better
|
433
|
-
"dataChannelsEnabled": _this2.dataChannelsEnabled,
|
434
|
-
"dataChannelOptions": _this2.dataChannelsEnabled ? _this2.dataChannelOptions : undefined
|
435
|
-
};
|
436
|
-
if (_this2.livekitServiceURL) {
|
437
|
-
groupCallState["io.element.livekit_service_url"] = _this2.livekitServiceURL;
|
438
|
-
}
|
439
|
-
yield _this2.client.sendStateEvent(_this2.room.roomId, EventType.GroupCallPrefix, groupCallState, _this2.groupCallId);
|
440
|
-
})();
|
441
|
-
}
|
442
|
-
get livekitServiceURL() {
|
443
|
-
return this._livekitServiceURL;
|
444
|
-
}
|
445
|
-
updateLivekitServiceURL(newURL) {
|
446
|
-
this._livekitServiceURL = newURL;
|
447
|
-
return this.sendCallStateEvent();
|
448
|
-
}
|
449
|
-
/**
|
450
|
-
* The group call's state.
|
451
|
-
*/
|
452
|
-
get state() {
|
453
|
-
return this._state;
|
454
|
-
}
|
455
|
-
set state(value) {
|
456
|
-
var prevValue = this._state;
|
457
|
-
if (value !== prevValue) {
|
458
|
-
this._state = value;
|
459
|
-
this.emit(GroupCallEvent.GroupCallStateChanged, value, prevValue);
|
460
|
-
}
|
461
|
-
}
|
462
|
-
/**
|
463
|
-
* The current participants in the call, as a map from members to device IDs
|
464
|
-
* to participant info.
|
465
|
-
*/
|
466
|
-
get participants() {
|
467
|
-
return this._participants;
|
468
|
-
}
|
469
|
-
set participants(value) {
|
470
|
-
var prevValue = this._participants;
|
471
|
-
var participantStateEqual = (x, y) => x.sessionId === y.sessionId && x.screensharing === y.screensharing;
|
472
|
-
var deviceMapsEqual = (x, y) => mapsEqual(x, y, participantStateEqual);
|
473
|
-
|
474
|
-
// Only update if the map actually changed
|
475
|
-
if (!mapsEqual(value, prevValue, deviceMapsEqual)) {
|
476
|
-
this._participants = value;
|
477
|
-
this.emit(GroupCallEvent.ParticipantsChanged, value);
|
478
|
-
}
|
479
|
-
}
|
480
|
-
/**
|
481
|
-
* The timestamp at which the call was created, or null if it has not yet
|
482
|
-
* been created.
|
483
|
-
*/
|
484
|
-
get creationTs() {
|
485
|
-
return this._creationTs;
|
486
|
-
}
|
487
|
-
set creationTs(value) {
|
488
|
-
this._creationTs = value;
|
489
|
-
}
|
490
|
-
/**
|
491
|
-
* Whether the local device has entered this call via another session, such
|
492
|
-
* as a widget.
|
493
|
-
*/
|
494
|
-
get enteredViaAnotherSession() {
|
495
|
-
return this._enteredViaAnotherSession;
|
496
|
-
}
|
497
|
-
set enteredViaAnotherSession(value) {
|
498
|
-
this._enteredViaAnotherSession = value;
|
499
|
-
this.updateParticipants();
|
500
|
-
}
|
501
|
-
|
502
|
-
/**
|
503
|
-
* Executes the given callback on all calls in this group call.
|
504
|
-
* @param f - The callback.
|
505
|
-
*/
|
506
|
-
forEachCall(f) {
|
507
|
-
for (var deviceMap of this.calls.values()) {
|
508
|
-
for (var call of deviceMap.values()) f(call);
|
509
|
-
}
|
510
|
-
}
|
511
|
-
getLocalFeeds() {
|
512
|
-
var feeds = [];
|
513
|
-
if (this.localCallFeed) feeds.push(this.localCallFeed);
|
514
|
-
if (this.localScreenshareFeed) feeds.push(this.localScreenshareFeed);
|
515
|
-
return feeds;
|
516
|
-
}
|
517
|
-
hasLocalParticipant() {
|
518
|
-
var _this$participants$ge, _this$participants$ge2;
|
519
|
-
return (_this$participants$ge = (_this$participants$ge2 = this.participants.get(this.room.getMember(this.client.getUserId()))) === null || _this$participants$ge2 === void 0 ? void 0 : _this$participants$ge2.has(this.client.getDeviceId())) !== null && _this$participants$ge !== void 0 ? _this$participants$ge : false;
|
520
|
-
}
|
521
|
-
|
522
|
-
/**
|
523
|
-
* Determines whether the given call is one that we were expecting to exist
|
524
|
-
* given our knowledge of who is participating in the group call.
|
525
|
-
*/
|
526
|
-
callExpected(call) {
|
527
|
-
var _this$participants$ge3;
|
528
|
-
var userId = getCallUserId(call);
|
529
|
-
var member = userId === null ? null : this.room.getMember(userId);
|
530
|
-
var deviceId = call.getOpponentDeviceId();
|
531
|
-
return member !== null && deviceId !== undefined && ((_this$participants$ge3 = this.participants.get(member)) === null || _this$participants$ge3 === void 0 ? void 0 : _this$participants$ge3.get(deviceId)) !== undefined;
|
532
|
-
}
|
533
|
-
initLocalCallFeed() {
|
534
|
-
var _this3 = this;
|
535
|
-
return _asyncToGenerator(function* () {
|
536
|
-
if (_this3.useLivekit) {
|
537
|
-
logger.info("Livekit group call: not starting local call feed.");
|
538
|
-
return;
|
539
|
-
}
|
540
|
-
if (_this3.state !== GroupCallState.LocalCallFeedUninitialized) {
|
541
|
-
throw new Error("Cannot initialize local call feed in the \"".concat(_this3.state, "\" state."));
|
542
|
-
}
|
543
|
-
_this3.state = GroupCallState.InitializingLocalCallFeed;
|
544
|
-
|
545
|
-
// wraps the real method to serialise calls, because we don't want to try starting
|
546
|
-
// multiple call feeds at once
|
547
|
-
if (_this3.initCallFeedPromise) return _this3.initCallFeedPromise;
|
548
|
-
try {
|
549
|
-
_this3.initCallFeedPromise = _this3.initLocalCallFeedInternal();
|
550
|
-
yield _this3.initCallFeedPromise;
|
551
|
-
} finally {
|
552
|
-
_this3.initCallFeedPromise = undefined;
|
553
|
-
}
|
554
|
-
})();
|
555
|
-
}
|
556
|
-
initLocalCallFeedInternal() {
|
557
|
-
var _this4 = this;
|
558
|
-
return _asyncToGenerator(function* () {
|
559
|
-
logger.log("GroupCall ".concat(_this4.groupCallId, " initLocalCallFeedInternal() running"));
|
560
|
-
var stream;
|
561
|
-
try {
|
562
|
-
stream = yield _this4.client.getMediaHandler().getUserMediaStream(true, _this4.type === GroupCallType.Video);
|
563
|
-
} catch (error) {
|
564
|
-
// If is allowed to join a call without a media stream, then we
|
565
|
-
// don't throw an error here. But we need an empty Local Feed to establish
|
566
|
-
// a connection later.
|
567
|
-
if (_this4.allowCallWithoutVideoAndAudio) {
|
568
|
-
stream = new MediaStream();
|
569
|
-
} else {
|
570
|
-
_this4.state = GroupCallState.LocalCallFeedUninitialized;
|
571
|
-
throw error;
|
572
|
-
}
|
573
|
-
}
|
574
|
-
|
575
|
-
// The call could've been disposed while we were waiting, and could
|
576
|
-
// also have been started back up again (hello, React 18) so if we're
|
577
|
-
// still in this 'initializing' state, carry on, otherwise bail.
|
578
|
-
if (_this4._state !== GroupCallState.InitializingLocalCallFeed) {
|
579
|
-
_this4.client.getMediaHandler().stopUserMediaStream(stream);
|
580
|
-
throw new Error("Group call disposed while gathering media stream");
|
581
|
-
}
|
582
|
-
var callFeed = new CallFeed({
|
583
|
-
client: _this4.client,
|
584
|
-
roomId: _this4.room.roomId,
|
585
|
-
userId: _this4.client.getUserId(),
|
586
|
-
deviceId: _this4.client.getDeviceId(),
|
587
|
-
stream,
|
588
|
-
purpose: SDPStreamMetadataPurpose.Usermedia,
|
589
|
-
audioMuted: _this4.initWithAudioMuted || stream.getAudioTracks().length === 0 || _this4.isPtt,
|
590
|
-
videoMuted: _this4.initWithVideoMuted || stream.getVideoTracks().length === 0
|
591
|
-
});
|
592
|
-
setTracksEnabled(stream.getAudioTracks(), !callFeed.isAudioMuted());
|
593
|
-
setTracksEnabled(stream.getVideoTracks(), !callFeed.isVideoMuted());
|
594
|
-
_this4.localCallFeed = callFeed;
|
595
|
-
_this4.addUserMediaFeed(callFeed);
|
596
|
-
_this4.state = GroupCallState.LocalCallFeedInitialized;
|
597
|
-
})();
|
598
|
-
}
|
599
|
-
updateLocalUsermediaStream(stream) {
|
600
|
-
var _this5 = this;
|
601
|
-
return _asyncToGenerator(function* () {
|
602
|
-
if (_this5.localCallFeed) {
|
603
|
-
var oldStream = _this5.localCallFeed.stream;
|
604
|
-
_this5.localCallFeed.setNewStream(stream);
|
605
|
-
var micShouldBeMuted = _this5.localCallFeed.isAudioMuted();
|
606
|
-
var vidShouldBeMuted = _this5.localCallFeed.isVideoMuted();
|
607
|
-
logger.log("GroupCall ".concat(_this5.groupCallId, " updateLocalUsermediaStream() (oldStreamId=").concat(oldStream.id, ", newStreamId=").concat(stream.id, ", micShouldBeMuted=").concat(micShouldBeMuted, ", vidShouldBeMuted=").concat(vidShouldBeMuted, ")"));
|
608
|
-
setTracksEnabled(stream.getAudioTracks(), !micShouldBeMuted);
|
609
|
-
setTracksEnabled(stream.getVideoTracks(), !vidShouldBeMuted);
|
610
|
-
_this5.client.getMediaHandler().stopUserMediaStream(oldStream);
|
611
|
-
}
|
612
|
-
})();
|
613
|
-
}
|
614
|
-
enter() {
|
615
|
-
var _this6 = this;
|
616
|
-
return _asyncToGenerator(function* () {
|
617
|
-
if (_this6.state === GroupCallState.LocalCallFeedUninitialized) {
|
618
|
-
yield _this6.initLocalCallFeed();
|
619
|
-
} else if (_this6.state !== GroupCallState.LocalCallFeedInitialized) {
|
620
|
-
throw new Error("Cannot enter call in the \"".concat(_this6.state, "\" state"));
|
621
|
-
}
|
622
|
-
logger.log("GroupCall ".concat(_this6.groupCallId, " enter() running"));
|
623
|
-
_this6.state = GroupCallState.Entered;
|
624
|
-
_this6.client.on(CallEventHandlerEvent.Incoming, _this6.onIncomingCall);
|
625
|
-
for (var call of _this6.client.callEventHandler.calls.values()) {
|
626
|
-
_this6.onIncomingCall(call);
|
627
|
-
}
|
628
|
-
if (!_this6.useLivekit) {
|
629
|
-
_this6.retryCallLoopInterval = setInterval(_this6.onRetryCallLoop, _this6.retryCallInterval);
|
630
|
-
_this6.activeSpeaker = undefined;
|
631
|
-
_this6.onActiveSpeakerLoop();
|
632
|
-
_this6.activeSpeakerLoopInterval = setInterval(_this6.onActiveSpeakerLoop, _this6.activeSpeakerInterval);
|
633
|
-
}
|
634
|
-
})();
|
635
|
-
}
|
636
|
-
dispose() {
|
637
|
-
var _this$stats;
|
638
|
-
if (this.localCallFeed) {
|
639
|
-
this.removeUserMediaFeed(this.localCallFeed);
|
640
|
-
this.localCallFeed = undefined;
|
641
|
-
}
|
642
|
-
if (this.localScreenshareFeed) {
|
643
|
-
this.client.getMediaHandler().stopScreensharingStream(this.localScreenshareFeed.stream);
|
644
|
-
this.removeScreenshareFeed(this.localScreenshareFeed);
|
645
|
-
this.localScreenshareFeed = undefined;
|
646
|
-
this.localDesktopCapturerSourceId = undefined;
|
647
|
-
}
|
648
|
-
this.client.getMediaHandler().stopAllStreams();
|
649
|
-
if (this.transmitTimer !== null) {
|
650
|
-
clearTimeout(this.transmitTimer);
|
651
|
-
this.transmitTimer = null;
|
652
|
-
}
|
653
|
-
if (this.retryCallLoopInterval !== undefined) {
|
654
|
-
clearInterval(this.retryCallLoopInterval);
|
655
|
-
this.retryCallLoopInterval = undefined;
|
656
|
-
}
|
657
|
-
if (this.participantsExpirationTimer !== null) {
|
658
|
-
clearTimeout(this.participantsExpirationTimer);
|
659
|
-
this.participantsExpirationTimer = null;
|
660
|
-
}
|
661
|
-
if (this.state !== GroupCallState.Entered) {
|
662
|
-
return;
|
663
|
-
}
|
664
|
-
this.forEachCall(call => call.hangup(CallErrorCode.UserHangup, false));
|
665
|
-
this.activeSpeaker = undefined;
|
666
|
-
clearInterval(this.activeSpeakerLoopInterval);
|
667
|
-
this.retryCallCounts.clear();
|
668
|
-
clearInterval(this.retryCallLoopInterval);
|
669
|
-
this.client.removeListener(CallEventHandlerEvent.Incoming, this.onIncomingCall);
|
670
|
-
(_this$stats = this.stats) === null || _this$stats === void 0 || _this$stats.stop();
|
671
|
-
}
|
672
|
-
leave() {
|
673
|
-
this.dispose();
|
674
|
-
this.state = GroupCallState.LocalCallFeedUninitialized;
|
675
|
-
}
|
676
|
-
terminate() {
|
677
|
-
var _arguments = arguments,
|
678
|
-
_this7 = this;
|
679
|
-
return _asyncToGenerator(function* () {
|
680
|
-
var emitStateEvent = _arguments.length > 0 && _arguments[0] !== undefined ? _arguments[0] : true;
|
681
|
-
_this7.dispose();
|
682
|
-
_this7.room.off(RoomStateEvent.Update, _this7.onRoomState);
|
683
|
-
_this7.client.groupCallEventHandler.groupCalls.delete(_this7.room.roomId);
|
684
|
-
_this7.client.emit(GroupCallEventHandlerEvent.Ended, _this7);
|
685
|
-
_this7.state = GroupCallState.Ended;
|
686
|
-
if (emitStateEvent) {
|
687
|
-
var existingStateEvent = _this7.room.currentState.getStateEvents(EventType.GroupCallPrefix, _this7.groupCallId);
|
688
|
-
yield _this7.client.sendStateEvent(_this7.room.roomId, EventType.GroupCallPrefix, _objectSpread(_objectSpread({}, existingStateEvent.getContent()), {}, {
|
689
|
-
"m.terminated": GroupCallTerminationReason.CallEnded
|
690
|
-
}), _this7.groupCallId);
|
691
|
-
}
|
692
|
-
})();
|
693
|
-
}
|
694
|
-
|
695
|
-
/*
|
696
|
-
* Local Usermedia
|
697
|
-
*/
|
698
|
-
|
699
|
-
isLocalVideoMuted() {
|
700
|
-
if (this.localCallFeed) {
|
701
|
-
return this.localCallFeed.isVideoMuted();
|
702
|
-
}
|
703
|
-
return true;
|
704
|
-
}
|
705
|
-
isMicrophoneMuted() {
|
706
|
-
if (this.localCallFeed) {
|
707
|
-
return this.localCallFeed.isAudioMuted();
|
708
|
-
}
|
709
|
-
return true;
|
710
|
-
}
|
711
|
-
|
712
|
-
/**
|
713
|
-
* Sets the mute state of the local participants's microphone.
|
714
|
-
* @param muted - Whether to mute the microphone
|
715
|
-
* @returns Whether muting/unmuting was successful
|
716
|
-
*/
|
717
|
-
setMicrophoneMuted(muted) {
|
718
|
-
var _this8 = this;
|
719
|
-
return _asyncToGenerator(function* () {
|
720
|
-
// hasAudioDevice can block indefinitely if the window has lost focus,
|
721
|
-
// and it doesn't make much sense to keep a device from being muted, so
|
722
|
-
// we always allow muted = true changes to go through
|
723
|
-
if (!muted && !(yield _this8.client.getMediaHandler().hasAudioDevice())) {
|
724
|
-
return false;
|
725
|
-
}
|
726
|
-
var sendUpdatesBefore = !muted && _this8.isPtt;
|
727
|
-
|
728
|
-
// set a timer for the maximum transmit time on PTT calls
|
729
|
-
if (_this8.isPtt) {
|
730
|
-
// Set or clear the max transmit timer
|
731
|
-
if (!muted && _this8.isMicrophoneMuted()) {
|
732
|
-
_this8.transmitTimer = setTimeout(() => {
|
733
|
-
_this8.setMicrophoneMuted(true);
|
734
|
-
}, _this8.pttMaxTransmitTime);
|
735
|
-
} else if (muted && !_this8.isMicrophoneMuted()) {
|
736
|
-
if (_this8.transmitTimer !== null) clearTimeout(_this8.transmitTimer);
|
737
|
-
_this8.transmitTimer = null;
|
738
|
-
}
|
739
|
-
}
|
740
|
-
_this8.forEachCall(call => {
|
741
|
-
var _call$localUsermediaF;
|
742
|
-
return (_call$localUsermediaF = call.localUsermediaFeed) === null || _call$localUsermediaF === void 0 ? void 0 : _call$localUsermediaF.setAudioVideoMuted(muted, null);
|
743
|
-
});
|
744
|
-
var sendUpdates = /*#__PURE__*/function () {
|
745
|
-
var _ref = _asyncToGenerator(function* () {
|
746
|
-
var updates = [];
|
747
|
-
_this8.forEachCall(call => updates.push(call.sendMetadataUpdate()));
|
748
|
-
yield Promise.all(updates).catch(e => logger.info("GroupCall ".concat(_this8.groupCallId, " setMicrophoneMuted() failed to send some metadata updates"), e));
|
749
|
-
});
|
750
|
-
return function sendUpdates() {
|
751
|
-
return _ref.apply(this, arguments);
|
752
|
-
};
|
753
|
-
}();
|
754
|
-
if (sendUpdatesBefore) yield sendUpdates();
|
755
|
-
if (_this8.localCallFeed) {
|
756
|
-
logger.log("GroupCall ".concat(_this8.groupCallId, " setMicrophoneMuted() (streamId=").concat(_this8.localCallFeed.stream.id, ", muted=").concat(muted, ")"));
|
757
|
-
var hasPermission = yield _this8.checkAudioPermissionIfNecessary(muted);
|
758
|
-
if (!hasPermission) {
|
759
|
-
return false;
|
760
|
-
}
|
761
|
-
_this8.localCallFeed.setAudioVideoMuted(muted, null);
|
762
|
-
// I don't believe its actually necessary to enable these tracks: they
|
763
|
-
// are the one on the GroupCall's own CallFeed and are cloned before being
|
764
|
-
// given to any of the actual calls, so these tracks don't actually go
|
765
|
-
// anywhere. Let's do it anyway to avoid confusion.
|
766
|
-
setTracksEnabled(_this8.localCallFeed.stream.getAudioTracks(), !muted);
|
767
|
-
} else {
|
768
|
-
logger.log("GroupCall ".concat(_this8.groupCallId, " setMicrophoneMuted() no stream muted (muted=").concat(muted, ")"));
|
769
|
-
_this8.initWithAudioMuted = muted;
|
770
|
-
}
|
771
|
-
_this8.forEachCall(call => setTracksEnabled(call.localUsermediaFeed.stream.getAudioTracks(), !muted && _this8.callExpected(call)));
|
772
|
-
_this8.emit(GroupCallEvent.LocalMuteStateChanged, muted, _this8.isLocalVideoMuted());
|
773
|
-
if (!sendUpdatesBefore) yield sendUpdates();
|
774
|
-
return true;
|
775
|
-
})();
|
776
|
-
}
|
777
|
-
|
778
|
-
/**
|
779
|
-
* If we allow entering a call without a camera and without video, it can happen that the access rights to the
|
780
|
-
* devices have not yet been queried. If a stream does not yet have an audio track, we assume that the rights have
|
781
|
-
* not yet been checked.
|
782
|
-
*
|
783
|
-
* `this.client.getMediaHandler().getUserMediaStream` clones the current stream, so it only wanted to be called when
|
784
|
-
* not Audio Track exists.
|
785
|
-
* As such, this is a compromise, because, the access rights should always be queried before the call.
|
786
|
-
*/
|
787
|
-
checkAudioPermissionIfNecessary(muted) {
|
788
|
-
var _this9 = this;
|
789
|
-
return _asyncToGenerator(function* () {
|
790
|
-
// We needed this here to avoid an error in case user join a call without a device.
|
791
|
-
try {
|
792
|
-
if (!muted && _this9.localCallFeed && !_this9.localCallFeed.hasAudioTrack) {
|
793
|
-
var stream = yield _this9.client.getMediaHandler().getUserMediaStream(true, !_this9.localCallFeed.isVideoMuted());
|
794
|
-
if ((stream === null || stream === void 0 ? void 0 : stream.getTracks().length) === 0) {
|
795
|
-
// if case permission denied to get a stream stop this here
|
796
|
-
/* istanbul ignore next */
|
797
|
-
logger.log("GroupCall ".concat(_this9.groupCallId, " setMicrophoneMuted() no device to receive local stream, muted=").concat(muted));
|
798
|
-
return false;
|
799
|
-
}
|
800
|
-
}
|
801
|
-
} catch (_unused) {
|
802
|
-
/* istanbul ignore next */
|
803
|
-
logger.log("GroupCall ".concat(_this9.groupCallId, " setMicrophoneMuted() no device or permission to receive local stream, muted=").concat(muted));
|
804
|
-
return false;
|
805
|
-
}
|
806
|
-
return true;
|
807
|
-
})();
|
808
|
-
}
|
809
|
-
|
810
|
-
/**
|
811
|
-
* Sets the mute state of the local participants's video.
|
812
|
-
* @param muted - Whether to mute the video
|
813
|
-
* @returns Whether muting/unmuting was successful
|
814
|
-
*/
|
815
|
-
setLocalVideoMuted(muted) {
|
816
|
-
var _this10 = this;
|
817
|
-
return _asyncToGenerator(function* () {
|
818
|
-
// hasAudioDevice can block indefinitely if the window has lost focus,
|
819
|
-
// and it doesn't make much sense to keep a device from being muted, so
|
820
|
-
// we always allow muted = true changes to go through
|
821
|
-
if (!muted && !(yield _this10.client.getMediaHandler().hasVideoDevice())) {
|
822
|
-
return false;
|
823
|
-
}
|
824
|
-
if (_this10.localCallFeed) {
|
825
|
-
/* istanbul ignore next */
|
826
|
-
logger.log("GroupCall ".concat(_this10.groupCallId, " setLocalVideoMuted() (stream=").concat(_this10.localCallFeed.stream.id, ", muted=").concat(muted, ")"));
|
827
|
-
try {
|
828
|
-
var stream = yield _this10.client.getMediaHandler().getUserMediaStream(true, !muted);
|
829
|
-
yield _this10.updateLocalUsermediaStream(stream);
|
830
|
-
_this10.localCallFeed.setAudioVideoMuted(null, muted);
|
831
|
-
setTracksEnabled(_this10.localCallFeed.stream.getVideoTracks(), !muted);
|
832
|
-
} catch (_unused2) {
|
833
|
-
// No permission to video device
|
834
|
-
/* istanbul ignore next */
|
835
|
-
logger.log("GroupCall ".concat(_this10.groupCallId, " setLocalVideoMuted() no device or permission to receive local stream, muted=").concat(muted));
|
836
|
-
return false;
|
837
|
-
}
|
838
|
-
} else {
|
839
|
-
logger.log("GroupCall ".concat(_this10.groupCallId, " setLocalVideoMuted() no stream muted (muted=").concat(muted, ")"));
|
840
|
-
_this10.initWithVideoMuted = muted;
|
841
|
-
}
|
842
|
-
var updates = [];
|
843
|
-
_this10.forEachCall(call => updates.push(call.setLocalVideoMuted(muted)));
|
844
|
-
yield Promise.all(updates);
|
845
|
-
|
846
|
-
// We setTracksEnabled again, independently from the call doing it
|
847
|
-
// internally, since we might not be expecting the call
|
848
|
-
_this10.forEachCall(call => setTracksEnabled(call.localUsermediaFeed.stream.getVideoTracks(), !muted && _this10.callExpected(call)));
|
849
|
-
_this10.emit(GroupCallEvent.LocalMuteStateChanged, _this10.isMicrophoneMuted(), muted);
|
850
|
-
return true;
|
851
|
-
})();
|
852
|
-
}
|
853
|
-
setScreensharingEnabled(enabled) {
|
854
|
-
var _arguments2 = arguments,
|
855
|
-
_this11 = this;
|
856
|
-
return _asyncToGenerator(function* () {
|
857
|
-
var opts = _arguments2.length > 1 && _arguments2[1] !== undefined ? _arguments2[1] : {};
|
858
|
-
if (enabled === _this11.isScreensharing()) {
|
859
|
-
return enabled;
|
860
|
-
}
|
861
|
-
if (enabled) {
|
862
|
-
try {
|
863
|
-
logger.log("GroupCall ".concat(_this11.groupCallId, " setScreensharingEnabled() is asking for screensharing permissions"));
|
864
|
-
var stream = yield _this11.client.getMediaHandler().getScreensharingStream(opts);
|
865
|
-
var _loop = function* _loop(track) {
|
866
|
-
var onTrackEnded = () => {
|
867
|
-
_this11.setScreensharingEnabled(false);
|
868
|
-
track.removeEventListener("ended", onTrackEnded);
|
869
|
-
};
|
870
|
-
track.addEventListener("ended", onTrackEnded);
|
871
|
-
};
|
872
|
-
for (var track of stream.getTracks()) {
|
873
|
-
yield* _loop(track);
|
874
|
-
}
|
875
|
-
logger.log("GroupCall ".concat(_this11.groupCallId, " setScreensharingEnabled() granted screensharing permissions. Setting screensharing enabled on all calls"));
|
876
|
-
_this11.localDesktopCapturerSourceId = opts.desktopCapturerSourceId;
|
877
|
-
_this11.localScreenshareFeed = new CallFeed({
|
878
|
-
client: _this11.client,
|
879
|
-
roomId: _this11.room.roomId,
|
880
|
-
userId: _this11.client.getUserId(),
|
881
|
-
deviceId: _this11.client.getDeviceId(),
|
882
|
-
stream,
|
883
|
-
purpose: SDPStreamMetadataPurpose.Screenshare,
|
884
|
-
audioMuted: false,
|
885
|
-
videoMuted: false
|
886
|
-
});
|
887
|
-
_this11.addScreenshareFeed(_this11.localScreenshareFeed);
|
888
|
-
_this11.emit(GroupCallEvent.LocalScreenshareStateChanged, true, _this11.localScreenshareFeed, _this11.localDesktopCapturerSourceId);
|
889
|
-
|
890
|
-
// TODO: handle errors
|
891
|
-
_this11.forEachCall(call => call.pushLocalFeed(_this11.localScreenshareFeed.clone()));
|
892
|
-
return true;
|
893
|
-
} catch (error) {
|
894
|
-
if (opts.throwOnFail) throw error;
|
895
|
-
logger.error("GroupCall ".concat(_this11.groupCallId, " setScreensharingEnabled() enabling screensharing error"), error);
|
896
|
-
_this11.emit(GroupCallEvent.Error, new GroupCallError(GroupCallErrorCode.NoUserMedia, "Failed to get screen-sharing stream: ", error));
|
897
|
-
return false;
|
898
|
-
}
|
899
|
-
} else {
|
900
|
-
_this11.forEachCall(call => {
|
901
|
-
if (call.localScreensharingFeed) call.removeLocalFeed(call.localScreensharingFeed);
|
902
|
-
});
|
903
|
-
_this11.client.getMediaHandler().stopScreensharingStream(_this11.localScreenshareFeed.stream);
|
904
|
-
_this11.removeScreenshareFeed(_this11.localScreenshareFeed);
|
905
|
-
_this11.localScreenshareFeed = undefined;
|
906
|
-
_this11.localDesktopCapturerSourceId = undefined;
|
907
|
-
_this11.emit(GroupCallEvent.LocalScreenshareStateChanged, false, undefined, undefined);
|
908
|
-
return false;
|
909
|
-
}
|
910
|
-
})();
|
911
|
-
}
|
912
|
-
isScreensharing() {
|
913
|
-
return !!this.localScreenshareFeed;
|
914
|
-
}
|
915
|
-
/**
|
916
|
-
* Determines whether a given participant expects us to call them (versus
|
917
|
-
* them calling us).
|
918
|
-
* @param userId - The participant's user ID.
|
919
|
-
* @param deviceId - The participant's device ID.
|
920
|
-
* @returns Whether we need to place an outgoing call to the participant.
|
921
|
-
*/
|
922
|
-
wantsOutgoingCall(userId, deviceId) {
|
923
|
-
var localUserId = this.client.getUserId();
|
924
|
-
var localDeviceId = this.client.getDeviceId();
|
925
|
-
return (
|
926
|
-
// If a user's ID is less than our own, they'll call us
|
927
|
-
userId >= localUserId && (
|
928
|
-
// If this is another one of our devices, compare device IDs to tell whether it'll call us
|
929
|
-
userId !== localUserId || deviceId > localDeviceId)
|
930
|
-
);
|
931
|
-
}
|
932
|
-
|
933
|
-
/**
|
934
|
-
* Places calls to all participants that we're responsible for calling.
|
935
|
-
*/
|
936
|
-
placeOutgoingCalls() {
|
937
|
-
var _this12 = this;
|
938
|
-
var callsChanged = false;
|
939
|
-
var _loop2 = function _loop2(_userId2) {
|
940
|
-
var _this12$calls$get;
|
941
|
-
var callMap = (_this12$calls$get = _this12.calls.get(_userId2)) !== null && _this12$calls$get !== void 0 ? _this12$calls$get : new Map();
|
942
|
-
var _loop3 = function _loop3(deviceId) {
|
943
|
-
var prevCall = callMap.get(deviceId);
|
944
|
-
if ((prevCall === null || prevCall === void 0 ? void 0 : prevCall.getOpponentSessionId()) !== participant.sessionId && _this12.wantsOutgoingCall(_userId2, deviceId)) {
|
945
|
-
callsChanged = true;
|
946
|
-
if (prevCall !== undefined) {
|
947
|
-
logger.debug("GroupCall ".concat(_this12.groupCallId, " placeOutgoingCalls() replacing call (userId=").concat(_userId2, ", deviceId=").concat(deviceId, ", callId=").concat(prevCall.callId, ")"));
|
948
|
-
prevCall.hangup(CallErrorCode.NewSession, false);
|
949
|
-
}
|
950
|
-
var newCall = createNewMatrixCall(_this12.client, _this12.room.roomId, {
|
951
|
-
invitee: _userId2,
|
952
|
-
opponentDeviceId: deviceId,
|
953
|
-
opponentSessionId: participant.sessionId,
|
954
|
-
groupCallId: _this12.groupCallId
|
955
|
-
});
|
956
|
-
if (newCall === null) {
|
957
|
-
logger.error("GroupCall ".concat(_this12.groupCallId, " placeOutgoingCalls() failed to create call (userId=").concat(_userId2, ", device=").concat(deviceId, ")"));
|
958
|
-
callMap.delete(deviceId);
|
959
|
-
} else {
|
960
|
-
_this12.initCall(newCall);
|
961
|
-
callMap.set(deviceId, newCall);
|
962
|
-
logger.debug("GroupCall ".concat(_this12.groupCallId, " placeOutgoingCalls() placing call (userId=").concat(_userId2, ", deviceId=").concat(deviceId, ", sessionId=").concat(participant.sessionId, ")"));
|
963
|
-
newCall.placeCallWithCallFeeds(_this12.getLocalFeeds().map(feed => feed.clone()), participant.screensharing).then(() => {
|
964
|
-
if (_this12.dataChannelsEnabled) {
|
965
|
-
newCall.createDataChannel("datachannel", _this12.dataChannelOptions);
|
966
|
-
}
|
967
|
-
}).catch(e => {
|
968
|
-
logger.warn("GroupCall ".concat(_this12.groupCallId, " placeOutgoingCalls() failed to place call (userId=").concat(_userId2, ")"), e);
|
969
|
-
if (e instanceof CallError && e.code === GroupCallErrorCode.UnknownDevice) {
|
970
|
-
_this12.emit(GroupCallEvent.Error, e);
|
971
|
-
} else {
|
972
|
-
_this12.emit(GroupCallEvent.Error, new GroupCallError(GroupCallErrorCode.PlaceCallFailed, "Failed to place call to ".concat(_userId2)));
|
973
|
-
}
|
974
|
-
newCall.hangup(CallErrorCode.SignallingFailed, false);
|
975
|
-
if (callMap.get(deviceId) === newCall) callMap.delete(deviceId);
|
976
|
-
});
|
977
|
-
}
|
978
|
-
}
|
979
|
-
};
|
980
|
-
for (var [deviceId, participant] of participantMap) {
|
981
|
-
_loop3(deviceId);
|
982
|
-
}
|
983
|
-
if (callMap.size > 0) {
|
984
|
-
_this12.calls.set(_userId2, callMap);
|
985
|
-
} else {
|
986
|
-
_this12.calls.delete(_userId2);
|
987
|
-
}
|
988
|
-
};
|
989
|
-
for (var [{
|
990
|
-
userId: _userId2
|
991
|
-
}, participantMap] of this.participants) {
|
992
|
-
_loop2(_userId2);
|
993
|
-
}
|
994
|
-
if (callsChanged) this.emit(GroupCallEvent.CallsChanged, this.calls);
|
995
|
-
}
|
996
|
-
|
997
|
-
/*
|
998
|
-
* Room Member State
|
999
|
-
*/
|
1000
|
-
|
1001
|
-
getMemberStateEvents(userId) {
|
1002
|
-
return userId === undefined ? this.room.currentState.getStateEvents(EventType.GroupCallMemberPrefix) : this.room.currentState.getStateEvents(EventType.GroupCallMemberPrefix, userId);
|
1003
|
-
}
|
1004
|
-
initCall(call) {
|
1005
|
-
var opponentMemberId = getCallUserId(call);
|
1006
|
-
if (!opponentMemberId) {
|
1007
|
-
throw new Error("Cannot init call without user id");
|
1008
|
-
}
|
1009
|
-
var onCallFeedsChanged = () => this.onCallFeedsChanged(call);
|
1010
|
-
var onCallStateChanged = (state, oldState) => this.onCallStateChanged(call, state, oldState);
|
1011
|
-
var onCallHangup = this.onCallHangup;
|
1012
|
-
var onCallReplaced = newCall => this.onCallReplaced(call, newCall);
|
1013
|
-
var deviceMap = this.callHandlers.get(opponentMemberId);
|
1014
|
-
if (deviceMap === undefined) {
|
1015
|
-
deviceMap = new Map();
|
1016
|
-
this.callHandlers.set(opponentMemberId, deviceMap);
|
1017
|
-
}
|
1018
|
-
deviceMap.set(call.getOpponentDeviceId(), {
|
1019
|
-
onCallFeedsChanged,
|
1020
|
-
onCallStateChanged,
|
1021
|
-
onCallHangup,
|
1022
|
-
onCallReplaced
|
1023
|
-
});
|
1024
|
-
call.on(CallEvent.FeedsChanged, onCallFeedsChanged);
|
1025
|
-
call.on(CallEvent.State, onCallStateChanged);
|
1026
|
-
call.on(CallEvent.Hangup, onCallHangup);
|
1027
|
-
call.on(CallEvent.Replaced, onCallReplaced);
|
1028
|
-
call.isPtt = this.isPtt;
|
1029
|
-
this.reEmitter.reEmit(call, Object.values(CallEvent));
|
1030
|
-
call.initStats(this.getGroupCallStats());
|
1031
|
-
onCallFeedsChanged();
|
1032
|
-
}
|
1033
|
-
disposeCall(call, hangupReason) {
|
1034
|
-
var opponentMemberId = getCallUserId(call);
|
1035
|
-
var opponentDeviceId = call.getOpponentDeviceId();
|
1036
|
-
if (!opponentMemberId) {
|
1037
|
-
throw new Error("Cannot dispose call without user id");
|
1038
|
-
}
|
1039
|
-
var deviceMap = this.callHandlers.get(opponentMemberId);
|
1040
|
-
var {
|
1041
|
-
onCallFeedsChanged,
|
1042
|
-
onCallStateChanged,
|
1043
|
-
onCallHangup,
|
1044
|
-
onCallReplaced
|
1045
|
-
} = deviceMap.get(opponentDeviceId);
|
1046
|
-
call.removeListener(CallEvent.FeedsChanged, onCallFeedsChanged);
|
1047
|
-
call.removeListener(CallEvent.State, onCallStateChanged);
|
1048
|
-
call.removeListener(CallEvent.Hangup, onCallHangup);
|
1049
|
-
call.removeListener(CallEvent.Replaced, onCallReplaced);
|
1050
|
-
deviceMap.delete(opponentMemberId);
|
1051
|
-
if (deviceMap.size === 0) this.callHandlers.delete(opponentMemberId);
|
1052
|
-
if (call.hangupReason === CallErrorCode.Replaced) {
|
1053
|
-
return;
|
1054
|
-
}
|
1055
|
-
var usermediaFeed = this.getUserMediaFeed(opponentMemberId, opponentDeviceId);
|
1056
|
-
if (usermediaFeed) {
|
1057
|
-
this.removeUserMediaFeed(usermediaFeed);
|
1058
|
-
}
|
1059
|
-
var screenshareFeed = this.getScreenshareFeed(opponentMemberId, opponentDeviceId);
|
1060
|
-
if (screenshareFeed) {
|
1061
|
-
this.removeScreenshareFeed(screenshareFeed);
|
1062
|
-
}
|
1063
|
-
}
|
1064
|
-
/*
|
1065
|
-
* UserMedia CallFeed Event Handlers
|
1066
|
-
*/
|
1067
|
-
|
1068
|
-
getUserMediaFeed(userId, deviceId) {
|
1069
|
-
return this.userMediaFeeds.find(f => f.userId === userId && f.deviceId === deviceId);
|
1070
|
-
}
|
1071
|
-
addUserMediaFeed(callFeed) {
|
1072
|
-
this.userMediaFeeds.push(callFeed);
|
1073
|
-
callFeed.measureVolumeActivity(true);
|
1074
|
-
this.emit(GroupCallEvent.UserMediaFeedsChanged, this.userMediaFeeds);
|
1075
|
-
}
|
1076
|
-
replaceUserMediaFeed(existingFeed, replacementFeed) {
|
1077
|
-
var feedIndex = this.userMediaFeeds.findIndex(f => f.userId === existingFeed.userId && f.deviceId === existingFeed.deviceId);
|
1078
|
-
if (feedIndex === -1) {
|
1079
|
-
throw new Error("Couldn't find user media feed to replace");
|
1080
|
-
}
|
1081
|
-
this.userMediaFeeds.splice(feedIndex, 1, replacementFeed);
|
1082
|
-
existingFeed.dispose();
|
1083
|
-
replacementFeed.measureVolumeActivity(true);
|
1084
|
-
this.emit(GroupCallEvent.UserMediaFeedsChanged, this.userMediaFeeds);
|
1085
|
-
}
|
1086
|
-
removeUserMediaFeed(callFeed) {
|
1087
|
-
var feedIndex = this.userMediaFeeds.findIndex(f => f.userId === callFeed.userId && f.deviceId === callFeed.deviceId);
|
1088
|
-
if (feedIndex === -1) {
|
1089
|
-
throw new Error("Couldn't find user media feed to remove");
|
1090
|
-
}
|
1091
|
-
this.userMediaFeeds.splice(feedIndex, 1);
|
1092
|
-
callFeed.dispose();
|
1093
|
-
this.emit(GroupCallEvent.UserMediaFeedsChanged, this.userMediaFeeds);
|
1094
|
-
if (this.activeSpeaker === callFeed) {
|
1095
|
-
this.activeSpeaker = this.userMediaFeeds[0];
|
1096
|
-
this.emit(GroupCallEvent.ActiveSpeakerChanged, this.activeSpeaker);
|
1097
|
-
}
|
1098
|
-
}
|
1099
|
-
/*
|
1100
|
-
* Screenshare Call Feed Event Handlers
|
1101
|
-
*/
|
1102
|
-
|
1103
|
-
getScreenshareFeed(userId, deviceId) {
|
1104
|
-
return this.screenshareFeeds.find(f => f.userId === userId && f.deviceId === deviceId);
|
1105
|
-
}
|
1106
|
-
addScreenshareFeed(callFeed) {
|
1107
|
-
this.screenshareFeeds.push(callFeed);
|
1108
|
-
this.emit(GroupCallEvent.ScreenshareFeedsChanged, this.screenshareFeeds);
|
1109
|
-
}
|
1110
|
-
replaceScreenshareFeed(existingFeed, replacementFeed) {
|
1111
|
-
var feedIndex = this.screenshareFeeds.findIndex(f => f.userId === existingFeed.userId && f.deviceId === existingFeed.deviceId);
|
1112
|
-
if (feedIndex === -1) {
|
1113
|
-
throw new Error("Couldn't find screenshare feed to replace");
|
1114
|
-
}
|
1115
|
-
this.screenshareFeeds.splice(feedIndex, 1, replacementFeed);
|
1116
|
-
existingFeed.dispose();
|
1117
|
-
this.emit(GroupCallEvent.ScreenshareFeedsChanged, this.screenshareFeeds);
|
1118
|
-
}
|
1119
|
-
removeScreenshareFeed(callFeed) {
|
1120
|
-
var feedIndex = this.screenshareFeeds.findIndex(f => f.userId === callFeed.userId && f.deviceId === callFeed.deviceId);
|
1121
|
-
if (feedIndex === -1) {
|
1122
|
-
throw new Error("Couldn't find screenshare feed to remove");
|
1123
|
-
}
|
1124
|
-
this.screenshareFeeds.splice(feedIndex, 1);
|
1125
|
-
callFeed.dispose();
|
1126
|
-
this.emit(GroupCallEvent.ScreenshareFeedsChanged, this.screenshareFeeds);
|
1127
|
-
}
|
1128
|
-
|
1129
|
-
/**
|
1130
|
-
* Recalculates and updates the participant map to match the room state.
|
1131
|
-
*/
|
1132
|
-
updateParticipants() {
|
1133
|
-
var localMember = this.room.getMember(this.client.getUserId());
|
1134
|
-
if (!localMember) {
|
1135
|
-
// The client hasn't fetched enough of the room state to get our own member
|
1136
|
-
// event. This probably shouldn't happen, but sanity check & exit for now.
|
1137
|
-
logger.warn("GroupCall ".concat(this.groupCallId, " updateParticipants() tried to update participants before local room member is available"));
|
1138
|
-
return;
|
1139
|
-
}
|
1140
|
-
if (this.participantsExpirationTimer !== null) {
|
1141
|
-
clearTimeout(this.participantsExpirationTimer);
|
1142
|
-
this.participantsExpirationTimer = null;
|
1143
|
-
}
|
1144
|
-
if (this.state === GroupCallState.Ended) {
|
1145
|
-
this.participants = new Map();
|
1146
|
-
return;
|
1147
|
-
}
|
1148
|
-
var participants = new Map();
|
1149
|
-
var now = Date.now();
|
1150
|
-
var entered = this.state === GroupCallState.Entered || this.enteredViaAnotherSession;
|
1151
|
-
var nextExpiration = Infinity;
|
1152
|
-
for (var e of this.getMemberStateEvents()) {
|
1153
|
-
var member = this.room.getMember(e.getStateKey());
|
1154
|
-
var content = e.getContent();
|
1155
|
-
var calls = Array.isArray(content["m.calls"]) ? content["m.calls"] : [];
|
1156
|
-
var call = calls.find(call => call["m.call_id"] === this.groupCallId);
|
1157
|
-
var devices = Array.isArray(call === null || call === void 0 ? void 0 : call["m.devices"]) ? call["m.devices"] : [];
|
1158
|
-
|
1159
|
-
// Filter out invalid and expired devices
|
1160
|
-
var validDevices = devices.filter(d => typeof d.device_id === "string" && typeof d.session_id === "string" && typeof d.expires_ts === "number" && d.expires_ts > now && Array.isArray(d.feeds));
|
1161
|
-
|
1162
|
-
// Apply local echo for the unentered case
|
1163
|
-
if (!entered && (member === null || member === void 0 ? void 0 : member.userId) === this.client.getUserId()) {
|
1164
|
-
validDevices = validDevices.filter(d => d.device_id !== this.client.getDeviceId());
|
1165
|
-
}
|
1166
|
-
|
1167
|
-
// Must have a connected device and be joined to the room
|
1168
|
-
if (validDevices.length > 0 && (member === null || member === void 0 ? void 0 : member.membership) === KnownMembership.Join) {
|
1169
|
-
var deviceMap = new Map();
|
1170
|
-
participants.set(member, deviceMap);
|
1171
|
-
for (var d of validDevices) {
|
1172
|
-
deviceMap.set(d.device_id, {
|
1173
|
-
sessionId: d.session_id,
|
1174
|
-
screensharing: d.feeds.some(f => f.purpose === SDPStreamMetadataPurpose.Screenshare)
|
1175
|
-
});
|
1176
|
-
if (d.expires_ts < nextExpiration) nextExpiration = d.expires_ts;
|
1177
|
-
}
|
1178
|
-
}
|
1179
|
-
}
|
1180
|
-
|
1181
|
-
// Apply local echo for the entered case
|
1182
|
-
if (entered) {
|
1183
|
-
var _deviceMap = participants.get(localMember);
|
1184
|
-
if (_deviceMap === undefined) {
|
1185
|
-
_deviceMap = new Map();
|
1186
|
-
participants.set(localMember, _deviceMap);
|
1187
|
-
}
|
1188
|
-
if (!_deviceMap.has(this.client.getDeviceId())) {
|
1189
|
-
_deviceMap.set(this.client.getDeviceId(), {
|
1190
|
-
sessionId: this.client.getSessionId(),
|
1191
|
-
screensharing: this.getLocalFeeds().some(f => f.purpose === SDPStreamMetadataPurpose.Screenshare)
|
1192
|
-
});
|
1193
|
-
}
|
1194
|
-
}
|
1195
|
-
this.participants = participants;
|
1196
|
-
if (nextExpiration < Infinity) {
|
1197
|
-
this.participantsExpirationTimer = setTimeout(() => this.updateParticipants(), nextExpiration - now);
|
1198
|
-
}
|
1199
|
-
}
|
1200
|
-
|
1201
|
-
/**
|
1202
|
-
* Updates the local user's member state with the devices returned by the given function.
|
1203
|
-
* @param fn - A function from the current devices to the new devices. If it
|
1204
|
-
* returns null, the update will be skipped.
|
1205
|
-
* @param keepAlive - Whether the request should outlive the window.
|
1206
|
-
*/
|
1207
|
-
updateDevices(fn) {
|
1208
|
-
var _arguments3 = arguments,
|
1209
|
-
_this13 = this;
|
1210
|
-
return _asyncToGenerator(function* () {
|
1211
|
-
var _event$getContent;
|
1212
|
-
var keepAlive = _arguments3.length > 1 && _arguments3[1] !== undefined ? _arguments3[1] : false;
|
1213
|
-
var now = Date.now();
|
1214
|
-
var localUserId = _this13.client.getUserId();
|
1215
|
-
var event = _this13.getMemberStateEvents(localUserId);
|
1216
|
-
var content = (_event$getContent = event === null || event === void 0 ? void 0 : event.getContent()) !== null && _event$getContent !== void 0 ? _event$getContent : {};
|
1217
|
-
var calls = Array.isArray(content["m.calls"]) ? content["m.calls"] : [];
|
1218
|
-
var call = null;
|
1219
|
-
var otherCalls = [];
|
1220
|
-
for (var c of calls) {
|
1221
|
-
if (c["m.call_id"] === _this13.groupCallId) {
|
1222
|
-
call = c;
|
1223
|
-
} else {
|
1224
|
-
otherCalls.push(c);
|
1225
|
-
}
|
1226
|
-
}
|
1227
|
-
if (call === null) call = {};
|
1228
|
-
var devices = Array.isArray(call["m.devices"]) ? call["m.devices"] : [];
|
1229
|
-
|
1230
|
-
// Filter out invalid and expired devices
|
1231
|
-
var validDevices = devices.filter(d => typeof d.device_id === "string" && typeof d.session_id === "string" && typeof d.expires_ts === "number" && d.expires_ts > now && Array.isArray(d.feeds));
|
1232
|
-
var newDevices = fn(validDevices);
|
1233
|
-
if (newDevices === null) return;
|
1234
|
-
var newCalls = [...otherCalls];
|
1235
|
-
if (newDevices.length > 0) {
|
1236
|
-
newCalls.push(_objectSpread(_objectSpread({}, call), {}, {
|
1237
|
-
"m.call_id": _this13.groupCallId,
|
1238
|
-
"m.devices": newDevices
|
1239
|
-
}));
|
1240
|
-
}
|
1241
|
-
var newContent = {
|
1242
|
-
"m.calls": newCalls
|
1243
|
-
};
|
1244
|
-
yield _this13.client.sendStateEvent(_this13.room.roomId, EventType.GroupCallMemberPrefix, newContent, localUserId, {
|
1245
|
-
keepAlive
|
1246
|
-
});
|
1247
|
-
})();
|
1248
|
-
}
|
1249
|
-
addDeviceToMemberState() {
|
1250
|
-
var _this14 = this;
|
1251
|
-
return _asyncToGenerator(function* () {
|
1252
|
-
yield _this14.updateDevices(devices => [...devices.filter(d => d.device_id !== _this14.client.getDeviceId()), {
|
1253
|
-
device_id: _this14.client.getDeviceId(),
|
1254
|
-
session_id: _this14.client.getSessionId(),
|
1255
|
-
expires_ts: Date.now() + DEVICE_TIMEOUT,
|
1256
|
-
feeds: _this14.getLocalFeeds().map(feed => ({
|
1257
|
-
purpose: feed.purpose
|
1258
|
-
}))
|
1259
|
-
// TODO: Add data channels
|
1260
|
-
}]);
|
1261
|
-
})();
|
1262
|
-
}
|
1263
|
-
updateMemberState() {
|
1264
|
-
var _this15 = this;
|
1265
|
-
return _asyncToGenerator(function* () {
|
1266
|
-
// Clear the old update interval before proceeding
|
1267
|
-
if (_this15.resendMemberStateTimer !== null) {
|
1268
|
-
clearInterval(_this15.resendMemberStateTimer);
|
1269
|
-
_this15.resendMemberStateTimer = null;
|
1270
|
-
}
|
1271
|
-
if (_this15.state === GroupCallState.Entered) {
|
1272
|
-
// Add the local device
|
1273
|
-
yield _this15.addDeviceToMemberState();
|
1274
|
-
|
1275
|
-
// Resend the state event every so often so it doesn't become stale
|
1276
|
-
_this15.resendMemberStateTimer = setInterval(/*#__PURE__*/_asyncToGenerator(function* () {
|
1277
|
-
logger.log("GroupCall ".concat(_this15.groupCallId, " updateMemberState() resending call member state\""));
|
1278
|
-
try {
|
1279
|
-
yield _this15.addDeviceToMemberState();
|
1280
|
-
} catch (e) {
|
1281
|
-
logger.error("GroupCall ".concat(_this15.groupCallId, " updateMemberState() failed to resend call member state"), e);
|
1282
|
-
}
|
1283
|
-
}), DEVICE_TIMEOUT * 3 / 4);
|
1284
|
-
} else {
|
1285
|
-
// Remove the local device
|
1286
|
-
yield _this15.updateDevices(devices => devices.filter(d => d.device_id !== _this15.client.getDeviceId()), true);
|
1287
|
-
}
|
1288
|
-
})();
|
1289
|
-
}
|
1290
|
-
|
1291
|
-
/**
|
1292
|
-
* Cleans up our member state by filtering out logged out devices, inactive
|
1293
|
-
* devices, and our own device (if we know we haven't entered).
|
1294
|
-
*/
|
1295
|
-
cleanMemberState() {
|
1296
|
-
var _this16 = this;
|
1297
|
-
return _asyncToGenerator(function* () {
|
1298
|
-
var {
|
1299
|
-
devices: myDevices
|
1300
|
-
} = yield _this16.client.getDevices();
|
1301
|
-
var deviceMap = new Map(myDevices.map(d => [d.device_id, d]));
|
1302
|
-
|
1303
|
-
// updateDevices takes care of filtering out inactive devices for us
|
1304
|
-
yield _this16.updateDevices(devices => {
|
1305
|
-
var newDevices = devices.filter(d => {
|
1306
|
-
var device = deviceMap.get(d.device_id);
|
1307
|
-
return (device === null || device === void 0 ? void 0 : device.last_seen_ts) !== undefined && !(d.device_id === _this16.client.getDeviceId() && _this16.state !== GroupCallState.Entered && !_this16.enteredViaAnotherSession);
|
1308
|
-
});
|
1309
|
-
|
1310
|
-
// Skip the update if the devices are unchanged
|
1311
|
-
return newDevices.length === devices.length ? null : newDevices;
|
1312
|
-
});
|
1313
|
-
})();
|
1314
|
-
}
|
1315
|
-
getGroupCallStats() {
|
1316
|
-
if (this.stats === undefined) {
|
1317
|
-
var userID = this.client.getUserId() || "unknown";
|
1318
|
-
this.stats = new GroupCallStats(this.groupCallId, userID, this.statsCollectIntervalTime);
|
1319
|
-
this.stats.reports.on(StatsReport.CONNECTION_STATS, this.onConnectionStats);
|
1320
|
-
this.stats.reports.on(StatsReport.BYTE_SENT_STATS, this.onByteSentStats);
|
1321
|
-
this.stats.reports.on(StatsReport.SUMMARY_STATS, this.onSummaryStats);
|
1322
|
-
this.stats.reports.on(StatsReport.CALL_FEED_REPORT, this.onCallFeedReport);
|
1323
|
-
}
|
1324
|
-
return this.stats;
|
1325
|
-
}
|
1326
|
-
setGroupCallStatsInterval(interval) {
|
1327
|
-
this.statsCollectIntervalTime = interval;
|
1328
|
-
if (this.stats !== undefined) {
|
1329
|
-
this.stats.stop();
|
1330
|
-
this.stats.setInterval(interval);
|
1331
|
-
if (interval > 0) {
|
1332
|
-
this.stats.start();
|
1333
|
-
}
|
1334
|
-
}
|
1335
|
-
}
|
1336
|
-
}
|
1337
|
-
//# sourceMappingURL=groupCall.js.map
|