@webex/plugin-meetings 2.59.1 → 2.59.3-next.1
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/.eslintrc.js +6 -6
- package/LICENSE +1 -1
- package/README.md +1204 -1204
- package/UPGRADING.md +287 -287
- package/babel.config.js +3 -3
- package/browsers.js +108 -108
- package/dist/common/browser-detection.d.ts +9 -9
- package/dist/common/browser-detection.js.map +1 -1
- package/dist/common/collection.d.ts +48 -48
- package/dist/common/collection.js +43 -43
- package/dist/common/collection.js.map +1 -1
- package/dist/common/config.d.ts +2 -2
- package/dist/common/config.js.map +1 -1
- package/dist/common/errors/captcha-error.d.ts +15 -15
- package/dist/common/errors/captcha-error.js +7 -7
- package/dist/common/errors/captcha-error.js.map +1 -1
- package/dist/common/errors/intent-to-join.d.ts +16 -16
- package/dist/common/errors/intent-to-join.js +7 -7
- package/dist/common/errors/intent-to-join.js.map +1 -1
- package/dist/common/errors/join-meeting.d.ts +17 -17
- package/dist/common/errors/join-meeting.js +8 -8
- package/dist/common/errors/join-meeting.js.map +1 -1
- package/dist/common/errors/media.d.ts +15 -15
- package/dist/common/errors/media.js +7 -7
- package/dist/common/errors/media.js.map +1 -1
- package/dist/common/errors/parameter.d.ts +15 -15
- package/dist/common/errors/parameter.js +7 -7
- package/dist/common/errors/parameter.js.map +1 -1
- package/dist/common/errors/password-error.d.ts +15 -15
- package/dist/common/errors/password-error.js +7 -7
- package/dist/common/errors/password-error.js.map +1 -1
- package/dist/common/errors/permission.d.ts +14 -14
- package/dist/common/errors/permission.js +6 -6
- package/dist/common/errors/permission.js.map +1 -1
- package/dist/common/errors/reconnection-in-progress.d.ts +9 -9
- package/dist/common/errors/reconnection-in-progress.js +6 -6
- package/dist/common/errors/reconnection-in-progress.js.map +1 -1
- package/dist/common/errors/reconnection.d.ts +15 -15
- package/dist/common/errors/reconnection.js +7 -7
- package/dist/common/errors/reconnection.js.map +1 -1
- package/dist/common/errors/stats.d.ts +15 -15
- package/dist/common/errors/stats.js +7 -7
- package/dist/common/errors/stats.js.map +1 -1
- package/dist/common/errors/webex-errors.d.ts +81 -81
- package/dist/common/errors/webex-errors.js +42 -42
- package/dist/common/errors/webex-errors.js.map +1 -1
- package/dist/common/errors/webex-meetings-error.d.ts +20 -20
- package/dist/common/errors/webex-meetings-error.js +12 -12
- package/dist/common/errors/webex-meetings-error.js.map +1 -1
- package/dist/common/events/events-scope.d.ts +17 -17
- package/dist/common/events/events-scope.js +10 -10
- package/dist/common/events/events-scope.js.map +1 -1
- package/dist/common/events/events.d.ts +12 -12
- package/dist/common/events/events.js +4 -4
- package/dist/common/events/events.js.map +1 -1
- package/dist/common/events/trigger-proxy.d.ts +2 -2
- package/dist/common/events/trigger-proxy.js.map +1 -1
- package/dist/common/events/util.d.ts +2 -2
- package/dist/common/events/util.js.map +1 -1
- package/dist/common/logs/logger-config.d.ts +2 -2
- package/dist/common/logs/logger-config.js.map +1 -1
- package/dist/common/logs/logger-proxy.d.ts +2 -2
- package/dist/common/logs/logger-proxy.js.map +1 -1
- package/dist/common/logs/request.d.ts +34 -34
- package/dist/common/logs/request.js +18 -18
- package/dist/common/logs/request.js.map +1 -1
- package/dist/common/queue.d.ts +32 -32
- package/dist/common/queue.js +18 -18
- package/dist/common/queue.js.map +1 -1
- package/dist/config.d.ts +73 -73
- package/dist/config.js.map +1 -1
- package/dist/constants.d.ts +924 -924
- package/dist/constants.js +9 -9
- package/dist/constants.js.map +1 -1
- package/dist/controls-options-manager/constants.d.ts +4 -4
- package/dist/controls-options-manager/constants.js.map +1 -1
- package/dist/controls-options-manager/enums.d.ts +5 -5
- package/dist/controls-options-manager/enums.js.map +1 -1
- package/dist/controls-options-manager/index.d.ts +120 -120
- package/dist/controls-options-manager/index.js +81 -81
- package/dist/controls-options-manager/index.js.map +1 -1
- package/dist/controls-options-manager/util.d.ts +7 -7
- package/dist/controls-options-manager/util.js.map +1 -1
- package/dist/index.d.ts +4 -4
- package/dist/index.js.map +1 -1
- package/dist/locus-info/controlsUtils.d.ts +2 -2
- package/dist/locus-info/controlsUtils.js +21 -21
- package/dist/locus-info/controlsUtils.js.map +1 -1
- package/dist/locus-info/embeddedAppsUtils.d.ts +2 -2
- package/dist/locus-info/embeddedAppsUtils.js +14 -14
- package/dist/locus-info/embeddedAppsUtils.js.map +1 -1
- package/dist/locus-info/fullState.d.ts +2 -2
- package/dist/locus-info/fullState.js.map +1 -1
- package/dist/locus-info/hostUtils.d.ts +2 -2
- package/dist/locus-info/hostUtils.js +19 -19
- package/dist/locus-info/hostUtils.js.map +1 -1
- package/dist/locus-info/index.d.ts +269 -269
- package/dist/locus-info/index.js +180 -180
- package/dist/locus-info/index.js.map +1 -1
- package/dist/locus-info/infoUtils.d.ts +2 -2
- package/dist/locus-info/infoUtils.js.map +1 -1
- package/dist/locus-info/mediaSharesUtils.d.ts +2 -2
- package/dist/locus-info/mediaSharesUtils.js +50 -50
- package/dist/locus-info/mediaSharesUtils.js.map +1 -1
- package/dist/locus-info/parser.d.ts +212 -212
- package/dist/locus-info/parser.js +136 -136
- package/dist/locus-info/parser.js.map +1 -1
- package/dist/locus-info/selfUtils.d.ts +2 -2
- package/dist/locus-info/selfUtils.js +52 -52
- package/dist/locus-info/selfUtils.js.map +1 -1
- package/dist/media/index.d.ts +32 -32
- package/dist/media/index.js +145 -145
- package/dist/media/index.js.map +1 -1
- package/dist/media/properties.d.ts +108 -108
- package/dist/media/properties.js +49 -49
- package/dist/media/properties.js.map +1 -1
- package/dist/media/util.d.ts +2 -2
- package/dist/media/util.js.map +1 -1
- package/dist/mediaQualityMetrics/config.d.ts +233 -233
- package/dist/mediaQualityMetrics/config.js.map +1 -1
- package/dist/meeting/effectsState.d.ts +42 -42
- package/dist/meeting/effectsState.js +24 -24
- package/dist/meeting/effectsState.js.map +1 -1
- package/dist/meeting/in-meeting-actions.d.ts +79 -79
- package/dist/meeting/in-meeting-actions.js +11 -11
- package/dist/meeting/in-meeting-actions.js.map +1 -1
- package/dist/meeting/index.d.ts +1621 -1621
- package/dist/meeting/index.js +1502 -1505
- package/dist/meeting/index.js.map +1 -1
- package/dist/meeting/muteState.d.ts +116 -116
- package/dist/meeting/muteState.js +85 -85
- package/dist/meeting/muteState.js.map +1 -1
- package/dist/meeting/request.d.ts +255 -255
- package/dist/meeting/request.js +141 -141
- package/dist/meeting/request.js.map +1 -1
- package/dist/meeting/state.d.ts +9 -9
- package/dist/meeting/state.js +30 -30
- package/dist/meeting/state.js.map +1 -1
- package/dist/meeting/util.d.ts +2 -2
- package/dist/meeting/util.js.map +1 -1
- package/dist/meeting-info/collection.d.ts +20 -20
- package/dist/meeting-info/collection.js +11 -11
- package/dist/meeting-info/collection.js.map +1 -1
- package/dist/meeting-info/index.d.ts +57 -57
- package/dist/meeting-info/index.js +50 -50
- package/dist/meeting-info/index.js.map +1 -1
- package/dist/meeting-info/meeting-info-v2.d.ts +93 -93
- package/dist/meeting-info/meeting-info-v2.js +52 -52
- package/dist/meeting-info/meeting-info-v2.js.map +1 -1
- package/dist/meeting-info/request.d.ts +22 -22
- package/dist/meeting-info/request.js +14 -14
- package/dist/meeting-info/request.js.map +1 -1
- package/dist/meeting-info/util.d.ts +2 -2
- package/dist/meeting-info/util.js +9 -9
- package/dist/meeting-info/util.js.map +1 -1
- package/dist/meeting-info/utilv2.d.ts +2 -2
- package/dist/meeting-info/utilv2.js +20 -20
- package/dist/meeting-info/utilv2.js.map +1 -1
- package/dist/meetings/collection.d.ts +23 -23
- package/dist/meetings/collection.js +14 -14
- package/dist/meetings/collection.js.map +1 -1
- package/dist/meetings/index.d.ts +296 -296
- package/dist/meetings/index.js +259 -259
- package/dist/meetings/index.js.map +1 -1
- package/dist/meetings/request.d.ts +27 -27
- package/dist/meetings/request.js +15 -15
- package/dist/meetings/request.js.map +1 -1
- package/dist/meetings/util.d.ts +18 -18
- package/dist/meetings/util.js +29 -29
- package/dist/meetings/util.js.map +1 -1
- package/dist/member/index.d.ts +147 -147
- package/dist/member/index.js +214 -214
- package/dist/member/index.js.map +1 -1
- package/dist/member/member.types.d.ts +11 -11
- package/dist/member/member.types.js.map +1 -1
- package/dist/member/util.d.ts +2 -2
- package/dist/member/util.js +60 -60
- package/dist/member/util.js.map +1 -1
- package/dist/members/collection.d.ts +24 -24
- package/dist/members/collection.js +11 -11
- package/dist/members/collection.js.map +1 -1
- package/dist/members/index.d.ts +298 -298
- package/dist/members/index.js +275 -275
- package/dist/members/index.js.map +1 -1
- package/dist/members/request.d.ts +50 -50
- package/dist/members/request.js +27 -27
- package/dist/members/request.js.map +1 -1
- package/dist/members/util.d.ts +2 -2
- package/dist/members/util.js +21 -21
- package/dist/members/util.js.map +1 -1
- package/dist/metrics/config.d.ts +169 -169
- package/dist/metrics/config.js.map +1 -1
- package/dist/metrics/constants.d.ts +57 -57
- package/dist/metrics/constants.js.map +1 -1
- package/dist/metrics/index.d.ts +152 -152
- package/dist/metrics/index.js +90 -90
- package/dist/metrics/index.js.map +1 -1
- package/dist/networkQualityMonitor/index.d.ts +70 -70
- package/dist/networkQualityMonitor/index.js +65 -65
- package/dist/networkQualityMonitor/index.js.map +1 -1
- package/dist/peer-connection-manager/index.d.ts +6 -6
- package/dist/peer-connection-manager/index.js +87 -87
- package/dist/peer-connection-manager/index.js.map +1 -1
- package/dist/peer-connection-manager/util.d.ts +6 -6
- package/dist/peer-connection-manager/util.js +9 -9
- package/dist/peer-connection-manager/util.js.map +1 -1
- package/dist/personal-meeting-room/index.d.ts +47 -47
- package/dist/personal-meeting-room/index.js +67 -67
- package/dist/personal-meeting-room/index.js.map +1 -1
- package/dist/personal-meeting-room/request.d.ts +14 -14
- package/dist/personal-meeting-room/request.js +7 -7
- package/dist/personal-meeting-room/request.js.map +1 -1
- package/dist/personal-meeting-room/util.d.ts +2 -2
- package/dist/personal-meeting-room/util.js.map +1 -1
- package/dist/reachability/index.d.ts +139 -139
- package/dist/reachability/index.js +110 -110
- package/dist/reachability/index.js.map +1 -1
- package/dist/reachability/request.d.ts +35 -35
- package/dist/reachability/request.js +15 -15
- package/dist/reachability/request.js.map +1 -1
- package/dist/reactions/reactions.d.ts +4 -4
- package/dist/reactions/reactions.js.map +1 -1
- package/dist/reactions/reactions.type.d.ts +32 -32
- package/dist/reactions/reactions.type.js.map +1 -1
- package/dist/reconnection-manager/index.d.ts +112 -112
- package/dist/reconnection-manager/index.js +112 -112
- package/dist/reconnection-manager/index.js.map +1 -1
- package/dist/recording-controller/enums.d.ts +7 -7
- package/dist/recording-controller/enums.js.map +1 -1
- package/dist/recording-controller/index.d.ts +193 -193
- package/dist/recording-controller/index.js +127 -127
- package/dist/recording-controller/index.js.map +1 -1
- package/dist/recording-controller/util.d.ts +13 -13
- package/dist/recording-controller/util.js.map +1 -1
- package/dist/roap/collection.d.ts +10 -10
- package/dist/roap/collection.js.map +1 -1
- package/dist/roap/handler.d.ts +47 -47
- package/dist/roap/handler.js +27 -27
- package/dist/roap/handler.js.map +1 -1
- package/dist/roap/index.d.ts +116 -116
- package/dist/roap/index.js +111 -111
- package/dist/roap/index.js.map +1 -1
- package/dist/roap/request.d.ts +35 -35
- package/dist/roap/request.js +17 -17
- package/dist/roap/request.js.map +1 -1
- package/dist/roap/state.d.ts +9 -9
- package/dist/roap/state.js +14 -14
- package/dist/roap/state.js.map +1 -1
- package/dist/roap/turnDiscovery.d.ts +67 -67
- package/dist/roap/turnDiscovery.js +46 -46
- package/dist/roap/turnDiscovery.js.map +1 -1
- package/dist/roap/util.d.ts +2 -2
- package/dist/roap/util.js.map +1 -1
- package/dist/statsAnalyzer/global.d.ts +126 -126
- package/dist/statsAnalyzer/global.js.map +1 -1
- package/dist/statsAnalyzer/index.d.ts +190 -190
- package/dist/statsAnalyzer/index.js +128 -128
- package/dist/statsAnalyzer/index.js.map +1 -1
- package/dist/statsAnalyzer/mqaUtil.d.ts +22 -22
- package/dist/statsAnalyzer/mqaUtil.js.map +1 -1
- package/dist/transcription/index.d.ts +64 -64
- package/dist/transcription/index.js +42 -42
- package/dist/transcription/index.js.map +1 -1
- package/internal-README.md +172 -172
- package/jest.config.js +3 -3
- package/package.json +22 -21
- package/process +1 -1
- package/src/common/browser-detection.ts +39 -39
- package/src/common/collection.ts +94 -94
- package/src/common/config.ts +9 -9
- package/src/common/errors/captcha-error.ts +25 -25
- package/src/common/errors/intent-to-join.ts +27 -27
- package/src/common/errors/join-meeting.ts +32 -32
- package/src/common/errors/media.ts +25 -25
- package/src/common/errors/parameter.ts +33 -33
- package/src/common/errors/password-error.ts +25 -25
- package/src/common/errors/permission.ts +24 -24
- package/src/common/errors/reconnection-in-progress.ts +8 -8
- package/src/common/errors/reconnection.ts +25 -25
- package/src/common/errors/stats.ts +25 -25
- package/src/common/errors/webex-errors.ts +140 -140
- package/src/common/errors/webex-meetings-error.ts +35 -35
- package/src/common/events/events-scope.ts +30 -30
- package/src/common/events/events.ts +25 -25
- package/src/common/events/trigger-proxy.ts +25 -25
- package/src/common/events/util.ts +39 -39
- package/src/common/logs/logger-config.ts +8 -8
- package/src/common/logs/logger-proxy.ts +44 -44
- package/src/common/logs/request.ts +65 -65
- package/src/common/queue.ts +50 -50
- package/src/config.ts +96 -96
- package/src/constants.ts +1121 -1121
- package/src/controls-options-manager/constants.ts +5 -5
- package/src/controls-options-manager/enums.ts +6 -6
- package/src/controls-options-manager/index.ts +183 -183
- package/src/controls-options-manager/util.ts +20 -20
- package/src/index.js +15 -15
- package/src/locus-info/controlsUtils.ts +112 -112
- package/src/locus-info/embeddedAppsUtils.ts +57 -57
- package/src/locus-info/fullState.ts +69 -69
- package/src/locus-info/hostUtils.ts +60 -60
- package/src/locus-info/index.ts +1303 -1303
- package/src/locus-info/infoUtils.ts +101 -101
- package/src/locus-info/mediaSharesUtils.ts +173 -173
- package/src/locus-info/parser.ts +680 -680
- package/src/locus-info/selfUtils.ts +428 -428
- package/src/media/index.ts +675 -675
- package/src/media/properties.ts +313 -313
- package/src/media/util.ts +37 -37
- package/src/mediaQualityMetrics/config.ts +382 -382
- package/src/meeting/effectsState.ts +209 -209
- package/src/meeting/in-meeting-actions.ts +153 -153
- package/src/meeting/index.ts +6537 -6543
- package/src/meeting/muteState.ts +365 -365
- package/src/meeting/request.ts +810 -810
- package/src/meeting/state.ts +194 -194
- package/src/meeting/util.ts +530 -530
- package/src/meeting-info/collection.ts +41 -41
- package/src/meeting-info/index.ts +137 -137
- package/src/meeting-info/meeting-info-v2.ts +273 -273
- package/src/meeting-info/request.ts +46 -46
- package/src/meeting-info/util.ts +314 -314
- package/src/meeting-info/utilv2.ts +324 -324
- package/src/meetings/collection.ts +43 -43
- package/src/meetings/index.ts +1128 -1128
- package/src/meetings/request.ts +81 -81
- package/src/meetings/util.ts +181 -181
- package/src/member/index.ts +446 -446
- package/src/member/member.types.ts +13 -13
- package/src/member/util.ts +286 -286
- package/src/members/collection.ts +40 -40
- package/src/members/index.ts +900 -900
- package/src/members/request.ts +175 -175
- package/src/members/util.ts +260 -260
- package/src/metrics/config.ts +485 -485
- package/src/metrics/constants.ts +61 -61
- package/src/metrics/index.ts +543 -543
- package/src/networkQualityMonitor/index.ts +211 -211
- package/src/peer-connection-manager/index.ts +847 -847
- package/src/peer-connection-manager/util.ts +119 -119
- package/src/personal-meeting-room/index.ts +157 -157
- package/src/personal-meeting-room/request.ts +48 -48
- package/src/personal-meeting-room/util.ts +49 -49
- package/src/reachability/index.ts +478 -478
- package/src/reachability/request.ts +81 -81
- package/src/reactions/reactions.ts +104 -104
- package/src/reactions/reactions.type.ts +36 -36
- package/src/reconnection-manager/index.ts +622 -622
- package/src/recording-controller/enums.ts +8 -8
- package/src/recording-controller/index.ts +315 -315
- package/src/recording-controller/util.ts +58 -58
- package/src/roap/collection.ts +62 -62
- package/src/roap/handler.ts +294 -294
- package/src/roap/index.ts +413 -413
- package/src/roap/request.ts +229 -229
- package/src/roap/state.ts +156 -156
- package/src/roap/turnDiscovery.ts +283 -283
- package/src/roap/util.ts +100 -100
- package/src/statsAnalyzer/global.ts +128 -128
- package/src/statsAnalyzer/index.ts +1266 -1266
- package/src/statsAnalyzer/mqaUtil.ts +290 -290
- package/src/transcription/index.ts +154 -154
- package/test/integration/spec/journey.js +941 -941
- package/test/integration/spec/space-meeting.js +457 -457
- package/test/integration/spec/transcription.js +55 -55
- package/test/unit/spec/common/browser-detection.js +119 -119
- package/test/unit/spec/common/queue.js +69 -69
- package/test/unit/spec/controls-options-manager/index.js +123 -123
- package/test/unit/spec/controls-options-manager/util.js +65 -65
- package/test/unit/spec/fixture/locus.js +406 -406
- package/test/unit/spec/locus-info/controlsUtils.js +82 -82
- package/test/unit/spec/locus-info/embeddedAppsUtils.js +104 -104
- package/test/unit/spec/locus-info/index.js +1272 -1272
- package/test/unit/spec/locus-info/infoUtils.js +138 -138
- package/test/unit/spec/locus-info/lib/BasicSeqCmp.json +975 -975
- package/test/unit/spec/locus-info/lib/SeqCmp.json +522 -522
- package/test/unit/spec/locus-info/lib/selfConstant.js +286 -286
- package/test/unit/spec/locus-info/parser.js +298 -298
- package/test/unit/spec/locus-info/selfUtils.js +185 -185
- package/test/unit/spec/media/properties.ts +305 -305
- package/test/unit/spec/meeting/effectsState.js +281 -281
- package/test/unit/spec/meeting/in-meeting-actions.ts +90 -90
- package/test/unit/spec/meeting/index.js +5227 -5227
- package/test/unit/spec/meeting/muteState.js +430 -430
- package/test/unit/spec/meeting/request.js +317 -317
- package/test/unit/spec/meeting/utils.js +319 -319
- package/test/unit/spec/meeting-info/meetinginfov2.js +376 -376
- package/test/unit/spec/meeting-info/request.js +64 -64
- package/test/unit/spec/meeting-info/util.js +37 -37
- package/test/unit/spec/meeting-info/utilv2.js +330 -330
- package/test/unit/spec/meetings/collection.js +52 -52
- package/test/unit/spec/meetings/index.js +1375 -1375
- package/test/unit/spec/meetings/utils.js +66 -66
- package/test/unit/spec/member/index.js +47 -47
- package/test/unit/spec/member/util.js +80 -80
- package/test/unit/spec/members/index.js +364 -364
- package/test/unit/spec/members/request.js +200 -200
- package/test/unit/spec/members/utils.js +42 -42
- package/test/unit/spec/metrics/index.js +111 -111
- package/test/unit/spec/networkQualityMonitor/index.js +99 -99
- package/test/unit/spec/peerconnection-manager/index.js +218 -218
- package/test/unit/spec/peerconnection-manager/utils.js +49 -49
- package/test/unit/spec/peerconnection-manager/utils.test-fixtures.ts +388 -388
- package/test/unit/spec/personal-meeting-room/personal-meeting-room.js +29 -29
- package/test/unit/spec/reachability/index.ts +50 -50
- package/test/unit/spec/reconnection-manager/index.js +206 -206
- package/test/unit/spec/recording-controller/index.js +230 -230
- package/test/unit/spec/recording-controller/util.js +101 -101
- package/test/unit/spec/roap/index.ts +128 -128
- package/test/unit/spec/roap/turnDiscovery.ts +372 -372
- package/test/unit/spec/roap/util.js +30 -30
- package/test/unit/spec/stats-analyzer/index.js +287 -287
- package/test/utils/cmr.js +104 -104
- package/test/utils/testUtils.js +287 -287
- package/test/utils/webex-config.js +77 -77
- package/test/utils/webex-test-users.js +82 -82
- package/tsconfig.json +5 -5
package/src/media/index.ts
CHANGED
|
@@ -1,675 +1,675 @@
|
|
|
1
|
-
/*!
|
|
2
|
-
* Copyright (c) 2015-2020 Cisco Systems, Inc. See LICENSE file.
|
|
3
|
-
*/
|
|
4
|
-
/* globals navigator */
|
|
5
|
-
|
|
6
|
-
import LoggerProxy from '../common/logs/logger-proxy';
|
|
7
|
-
import {
|
|
8
|
-
AUDIO_INPUT,
|
|
9
|
-
VIDEO_INPUT,
|
|
10
|
-
PEER_CONNECTION_STATE,
|
|
11
|
-
MEDIA_TRACK_CONSTRAINT,
|
|
12
|
-
} from '../constants';
|
|
13
|
-
import Config from '../config';
|
|
14
|
-
import PeerConnectionManager from '../peer-connection-manager';
|
|
15
|
-
import ReconnectionError from '../common/errors/reconnection';
|
|
16
|
-
import MediaError from '../common/errors/media';
|
|
17
|
-
import BrowserDetection from '../common/browser-detection';
|
|
18
|
-
|
|
19
|
-
const {isBrowser} = BrowserDetection();
|
|
20
|
-
|
|
21
|
-
/**
|
|
22
|
-
* MediaDirection
|
|
23
|
-
* @typedef {Object} MediaDirection
|
|
24
|
-
* @property {boolean} sendAudio
|
|
25
|
-
* @property {boolean} receiveAudio
|
|
26
|
-
* @property {boolean} sendVideo
|
|
27
|
-
* @property {boolean} receiveVideo
|
|
28
|
-
* @property {boolean} sendShare
|
|
29
|
-
* @property {boolean} receiveShare
|
|
30
|
-
*/
|
|
31
|
-
|
|
32
|
-
/**
|
|
33
|
-
* SendOptions
|
|
34
|
-
* @typedef {Object} SendOptions
|
|
35
|
-
* @property sendAudio
|
|
36
|
-
* @property sendVideo
|
|
37
|
-
* @property sendShare
|
|
38
|
-
* @property isSharing
|
|
39
|
-
* @property {Object} sharePreferences
|
|
40
|
-
*/
|
|
41
|
-
/**
|
|
42
|
-
*
|
|
43
|
-
* @public
|
|
44
|
-
* @export
|
|
45
|
-
* Mimic browser APIs as "the ultimate browser".
|
|
46
|
-
* Handles the quirks of each browser.
|
|
47
|
-
* Extends and enhances adapter.js, i.e., the "media" file from the web client.
|
|
48
|
-
*/
|
|
49
|
-
const Media: any = {};
|
|
50
|
-
|
|
51
|
-
/**
|
|
52
|
-
* @param {boolean} enabled
|
|
53
|
-
* @param {MediaStreamTrack} track
|
|
54
|
-
* @returns {Boolean}
|
|
55
|
-
* @public
|
|
56
|
-
*/
|
|
57
|
-
Media.setLocalTrack = (enabled: boolean, track: MediaStreamTrack) => {
|
|
58
|
-
if (track) {
|
|
59
|
-
track.enabled = enabled;
|
|
60
|
-
|
|
61
|
-
return true;
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
return false;
|
|
65
|
-
};
|
|
66
|
-
|
|
67
|
-
/**
|
|
68
|
-
* @param {RTCPeerConnection} peerConnection
|
|
69
|
-
* @param {Object} meetingProperties
|
|
70
|
-
* @param {string} meetingProperties.meetingId
|
|
71
|
-
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
72
|
-
* @returns {Promise}
|
|
73
|
-
*/
|
|
74
|
-
Media.reconnectMedia = (
|
|
75
|
-
peerConnection: RTCPeerConnection,
|
|
76
|
-
{
|
|
77
|
-
meetingId,
|
|
78
|
-
remoteQualityLevel,
|
|
79
|
-
enableRtx,
|
|
80
|
-
enableExtmap,
|
|
81
|
-
}: {
|
|
82
|
-
meetingId: string;
|
|
83
|
-
remoteQualityLevel: string;
|
|
84
|
-
enableRtx: any;
|
|
85
|
-
enableExtmap: any;
|
|
86
|
-
}
|
|
87
|
-
) => {
|
|
88
|
-
if (
|
|
89
|
-
peerConnection.connectionState === PEER_CONNECTION_STATE.CLOSED ||
|
|
90
|
-
peerConnection.connectionState === PEER_CONNECTION_STATE.FAILED
|
|
91
|
-
) {
|
|
92
|
-
return Promise.reject(new ReconnectionError('Reinitiate peerconnection'));
|
|
93
|
-
}
|
|
94
|
-
|
|
95
|
-
return PeerConnectionManager.createOffer(peerConnection, {
|
|
96
|
-
meetingId,
|
|
97
|
-
remoteQualityLevel,
|
|
98
|
-
enableRtx,
|
|
99
|
-
enableExtmap,
|
|
100
|
-
});
|
|
101
|
-
};
|
|
102
|
-
|
|
103
|
-
/**
|
|
104
|
-
* format the media array for send
|
|
105
|
-
* @param {String} mediaId
|
|
106
|
-
* @param {Boolean} audioMuted
|
|
107
|
-
* @param {Boolean} videoMuted
|
|
108
|
-
* @returns {Array} medias
|
|
109
|
-
*/
|
|
110
|
-
Media.generateLocalMedias = (mediaId: string, audioMuted: boolean, videoMuted: boolean) => {
|
|
111
|
-
if (mediaId) {
|
|
112
|
-
return [
|
|
113
|
-
{
|
|
114
|
-
localSdp: JSON.stringify({
|
|
115
|
-
audioMuted,
|
|
116
|
-
videoMuted,
|
|
117
|
-
}),
|
|
118
|
-
mediaId,
|
|
119
|
-
},
|
|
120
|
-
];
|
|
121
|
-
}
|
|
122
|
-
|
|
123
|
-
return [];
|
|
124
|
-
};
|
|
125
|
-
|
|
126
|
-
/**
|
|
127
|
-
* make a browser call to get the media
|
|
128
|
-
* @param {SendOptions} options
|
|
129
|
-
* @param {Object} config SDK Configuration for meetings plugin
|
|
130
|
-
* @returns {Promise}
|
|
131
|
-
*/
|
|
132
|
-
Media.getLocalMedia = (options: any, config: object) => {
|
|
133
|
-
const {sendAudio, sendVideo, sendShare, sharePreferences, isSharing} = options;
|
|
134
|
-
|
|
135
|
-
if (sendAudio || sendVideo) {
|
|
136
|
-
return Media.getMedia(sendAudio, sendVideo, config);
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
if (sendShare && !isSharing) {
|
|
140
|
-
return Media.getDisplayMedia(
|
|
141
|
-
{
|
|
142
|
-
sendAudio: false,
|
|
143
|
-
sendShare: true,
|
|
144
|
-
sharePreferences,
|
|
145
|
-
},
|
|
146
|
-
config
|
|
147
|
-
);
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
return Promise.resolve(undefined);
|
|
151
|
-
};
|
|
152
|
-
/**
|
|
153
|
-
* Returns the direction and tracks
|
|
154
|
-
* @param {string} trackType type of track (audio/video)
|
|
155
|
-
* @param {object} track tracks passed
|
|
156
|
-
* @param {boolean} receiveTracks do you want to receive tracks from the remote side
|
|
157
|
-
* @returns {Object} returns direction tracks to be added in transceiver
|
|
158
|
-
*/
|
|
159
|
-
Media.checkTracks = (trackType: string, track: object, receiveTracks: boolean) => {
|
|
160
|
-
// eslint-disable-next-line @typescript-eslint/no-shadow
|
|
161
|
-
const getDirection = (sendTracks, receiveTracks) => {
|
|
162
|
-
if (sendTracks && receiveTracks) {
|
|
163
|
-
return 'sendrecv';
|
|
164
|
-
}
|
|
165
|
-
if (sendTracks && !receiveTracks) {
|
|
166
|
-
return 'sendonly';
|
|
167
|
-
}
|
|
168
|
-
if (!sendTracks && receiveTracks) {
|
|
169
|
-
return 'recvonly';
|
|
170
|
-
}
|
|
171
|
-
|
|
172
|
-
return 'inactive';
|
|
173
|
-
};
|
|
174
|
-
|
|
175
|
-
if (track) {
|
|
176
|
-
return {track, direction: getDirection(!!track, receiveTracks)};
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
return {track: trackType, direction: getDirection(!!track, receiveTracks)};
|
|
180
|
-
};
|
|
181
|
-
/**
|
|
182
|
-
* creates peerconnection and attaches streams
|
|
183
|
-
* @param {MediaDirection} mediaProperties
|
|
184
|
-
* @param {Object} meetingProperties
|
|
185
|
-
* @param {string} meetingProperties.meetingId
|
|
186
|
-
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
187
|
-
* @returns {Array} [peerConnection, ]
|
|
188
|
-
*/
|
|
189
|
-
Media.attachMedia = (
|
|
190
|
-
mediaProperties: any,
|
|
191
|
-
{
|
|
192
|
-
meetingId,
|
|
193
|
-
remoteQualityLevel,
|
|
194
|
-
enableRtx,
|
|
195
|
-
enableExtmap,
|
|
196
|
-
}: {
|
|
197
|
-
meetingId: string;
|
|
198
|
-
remoteQualityLevel: string;
|
|
199
|
-
enableRtx: any;
|
|
200
|
-
enableExtmap: any;
|
|
201
|
-
}
|
|
202
|
-
) => {
|
|
203
|
-
const {mediaDirection, audioTrack, videoTrack, shareTrack, peerConnection} = mediaProperties;
|
|
204
|
-
|
|
205
|
-
let result = null;
|
|
206
|
-
|
|
207
|
-
// Add Transceiver for audio
|
|
208
|
-
result = Media.checkTracks(
|
|
209
|
-
'audio',
|
|
210
|
-
mediaDirection.sendAudio && audioTrack,
|
|
211
|
-
mediaDirection.receiveAudio
|
|
212
|
-
);
|
|
213
|
-
peerConnection.audioTransceiver = peerConnection.addTransceiver(result.track, {
|
|
214
|
-
direction: result.direction,
|
|
215
|
-
});
|
|
216
|
-
|
|
217
|
-
// Add Transceiver for video
|
|
218
|
-
result = Media.checkTracks(
|
|
219
|
-
'video',
|
|
220
|
-
mediaDirection.sendVideo && videoTrack,
|
|
221
|
-
mediaDirection.receiveVideo
|
|
222
|
-
);
|
|
223
|
-
peerConnection.videoTransceiver = peerConnection.addTransceiver(result.track, {
|
|
224
|
-
direction: result.direction,
|
|
225
|
-
});
|
|
226
|
-
|
|
227
|
-
// Add Transceiver for share
|
|
228
|
-
result = Media.checkTracks(
|
|
229
|
-
'video',
|
|
230
|
-
mediaDirection.sendShare && shareTrack,
|
|
231
|
-
mediaDirection.receiveShare
|
|
232
|
-
);
|
|
233
|
-
peerConnection.shareTransceiver = peerConnection.addTransceiver(result.track, {
|
|
234
|
-
direction: result.direction,
|
|
235
|
-
});
|
|
236
|
-
|
|
237
|
-
peerConnection.onnegotiationneeded = (event) => {
|
|
238
|
-
LoggerProxy.logger.info(
|
|
239
|
-
`Media:index#attachMedia --> onnegotiationneeded#PeerConnection: ${event}`
|
|
240
|
-
);
|
|
241
|
-
};
|
|
242
|
-
|
|
243
|
-
return PeerConnectionManager.createOffer(peerConnection, {
|
|
244
|
-
meetingId,
|
|
245
|
-
remoteQualityLevel,
|
|
246
|
-
enableRtx,
|
|
247
|
-
enableExtmap,
|
|
248
|
-
});
|
|
249
|
-
};
|
|
250
|
-
|
|
251
|
-
/**
|
|
252
|
-
* updates all the media streams and creates a new media offer
|
|
253
|
-
* @param {MediaDirection} mediaProperties
|
|
254
|
-
* @param {Object} meetingProperties
|
|
255
|
-
* @param {string} meetingProperties.meetingId
|
|
256
|
-
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
257
|
-
* @returns {Promise}
|
|
258
|
-
*/
|
|
259
|
-
Media.updateMedia = (
|
|
260
|
-
mediaProperties: any,
|
|
261
|
-
{
|
|
262
|
-
meetingId,
|
|
263
|
-
remoteQualityLevel,
|
|
264
|
-
enableRtx,
|
|
265
|
-
enableExtmap,
|
|
266
|
-
}: {
|
|
267
|
-
meetingId: string;
|
|
268
|
-
remoteQualityLevel: string;
|
|
269
|
-
enableRtx: any;
|
|
270
|
-
enableExtmap: any;
|
|
271
|
-
}
|
|
272
|
-
) => {
|
|
273
|
-
const {mediaDirection, audioTrack, videoTrack, shareTrack, peerConnection} = mediaProperties;
|
|
274
|
-
|
|
275
|
-
// update audio transceiver
|
|
276
|
-
Media.setTrackOnTransceiver(peerConnection.audioTransceiver, {
|
|
277
|
-
type: 'audio',
|
|
278
|
-
track: audioTrack,
|
|
279
|
-
sendTrack: mediaDirection.sendAudio && audioTrack,
|
|
280
|
-
receiveTrack: mediaDirection.receiveAudio,
|
|
281
|
-
});
|
|
282
|
-
|
|
283
|
-
// update video transceiver
|
|
284
|
-
Media.setTrackOnTransceiver(peerConnection.videoTransceiver, {
|
|
285
|
-
type: 'video',
|
|
286
|
-
track: videoTrack,
|
|
287
|
-
sendTrack: mediaDirection.sendVideo && videoTrack,
|
|
288
|
-
receiveTrack: mediaDirection.receiveVideo,
|
|
289
|
-
});
|
|
290
|
-
|
|
291
|
-
// update content transceiver
|
|
292
|
-
Media.setTrackOnTransceiver(peerConnection.shareTransceiver, {
|
|
293
|
-
type: 'video',
|
|
294
|
-
track: shareTrack,
|
|
295
|
-
sendTrack: mediaDirection.sendShare && shareTrack,
|
|
296
|
-
receiveTrack: mediaDirection.receiveShare,
|
|
297
|
-
});
|
|
298
|
-
peerConnection.onnegotiationneeded = (event) => {
|
|
299
|
-
LoggerProxy.logger.info(
|
|
300
|
-
`Media:index#updateMedia --> onnegotiationneeded#PeerConnection: ${event}`
|
|
301
|
-
);
|
|
302
|
-
};
|
|
303
|
-
|
|
304
|
-
return PeerConnectionManager.createOffer(peerConnection, {
|
|
305
|
-
meetingId,
|
|
306
|
-
remoteQualityLevel,
|
|
307
|
-
enableRtx,
|
|
308
|
-
enableExtmap,
|
|
309
|
-
});
|
|
310
|
-
};
|
|
311
|
-
|
|
312
|
-
/**
|
|
313
|
-
* @param {RTCRtpTransceiver} transceiver
|
|
314
|
-
* @param {Object} options
|
|
315
|
-
* @param {MediaStreamTrack} options.track
|
|
316
|
-
* @returns {undefined}
|
|
317
|
-
*/
|
|
318
|
-
Media.setTrackOnTransceiver = (
|
|
319
|
-
transceiver: RTCRtpTransceiver,
|
|
320
|
-
options: {
|
|
321
|
-
track: MediaStreamTrack;
|
|
322
|
-
type: any;
|
|
323
|
-
sendTrack: any;
|
|
324
|
-
receiveTrack: any;
|
|
325
|
-
}
|
|
326
|
-
) => {
|
|
327
|
-
const {type, track, sendTrack, receiveTrack} = options;
|
|
328
|
-
|
|
329
|
-
try {
|
|
330
|
-
const result = Media.checkTracks(type, sendTrack && track, receiveTrack);
|
|
331
|
-
|
|
332
|
-
transceiver.direction = result.direction;
|
|
333
|
-
if (options.track) {
|
|
334
|
-
transceiver.sender.replaceTrack(track);
|
|
335
|
-
}
|
|
336
|
-
} catch (e) {
|
|
337
|
-
LoggerProxy.logger.error(`Media:index#setTrackOnTransceiver --> ${e}`);
|
|
338
|
-
throw e;
|
|
339
|
-
}
|
|
340
|
-
};
|
|
341
|
-
|
|
342
|
-
/**
|
|
343
|
-
* creates a new offer
|
|
344
|
-
* @param {Object} meetingProperties
|
|
345
|
-
* @param {string} meetingProperties.meetingId
|
|
346
|
-
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
347
|
-
* @param {RTCPeerConnection} peerConnection
|
|
348
|
-
* @param {RTCRtpTransceiver} transceiver
|
|
349
|
-
* @param {Object} options see #Media.setTrackOnTransceiver
|
|
350
|
-
* @returns {Promise}
|
|
351
|
-
*/
|
|
352
|
-
Media.updateTransceiver = (
|
|
353
|
-
{
|
|
354
|
-
meetingId,
|
|
355
|
-
remoteQualityLevel,
|
|
356
|
-
enableRtx,
|
|
357
|
-
enableExtmap,
|
|
358
|
-
}: {
|
|
359
|
-
meetingId: string;
|
|
360
|
-
remoteQualityLevel: string;
|
|
361
|
-
enableRtx: any;
|
|
362
|
-
enableExtmap: any;
|
|
363
|
-
},
|
|
364
|
-
peerConnection: RTCPeerConnection,
|
|
365
|
-
transceiver: RTCRtpTransceiver,
|
|
366
|
-
options: object
|
|
367
|
-
) => {
|
|
368
|
-
Media.setTrackOnTransceiver(transceiver, options);
|
|
369
|
-
|
|
370
|
-
return PeerConnectionManager.createOffer(peerConnection, {
|
|
371
|
-
meetingId,
|
|
372
|
-
remoteQualityLevel,
|
|
373
|
-
enableRtx,
|
|
374
|
-
enableExtmap,
|
|
375
|
-
});
|
|
376
|
-
};
|
|
377
|
-
|
|
378
|
-
/**
|
|
379
|
-
* generates share streams
|
|
380
|
-
* @param {Object} options parameter
|
|
381
|
-
* @param {Boolean} options.sendAudio send audio from the display share
|
|
382
|
-
* @param {Boolean} options.sendShare send video from the display share
|
|
383
|
-
* @param {Object} options.sharePreferences
|
|
384
|
-
* @param {MediaTrackConstraints} options.sharePreferences.shareConstraints constraints to apply to video
|
|
385
|
-
* @see {@link https://developer.mozilla.org/en-US/docs/Web/API/MediaTrackConstraints}
|
|
386
|
-
* @param {Boolean} options.sharePreferences.highFrameRate if shareConstraints isn't provided, set default values based off of this boolean
|
|
387
|
-
* @param {Object} config SDK Configuration for meetings plugin
|
|
388
|
-
* @returns {Promise.<MediaStream>}
|
|
389
|
-
*/
|
|
390
|
-
Media.getDisplayMedia = (
|
|
391
|
-
options: {
|
|
392
|
-
sendAudio: boolean;
|
|
393
|
-
sendShare: boolean;
|
|
394
|
-
sharePreferences: {
|
|
395
|
-
shareConstraints: MediaTrackConstraints;
|
|
396
|
-
highFrameRate: any;
|
|
397
|
-
};
|
|
398
|
-
},
|
|
399
|
-
config: any = {}
|
|
400
|
-
) => {
|
|
401
|
-
// SDK screen share resolution settings from Webex.init
|
|
402
|
-
const customResolution = config.screenResolution || {};
|
|
403
|
-
// user defined screen share frame rate
|
|
404
|
-
const customShareFrameRate = config.screenFrameRate || null;
|
|
405
|
-
// user defined share preferences
|
|
406
|
-
const hasSharePreferences = options.sharePreferences;
|
|
407
|
-
const hasCustomConstraints = hasSharePreferences && hasSharePreferences.shareConstraints;
|
|
408
|
-
const hasHighFrameRate = hasSharePreferences && hasSharePreferences.highFrameRate;
|
|
409
|
-
const {screenResolution, resolution, videoShareFrameRate, screenFrameRate, aspectRatio} =
|
|
410
|
-
Config.meetings;
|
|
411
|
-
|
|
412
|
-
let shareConstraints: any = {
|
|
413
|
-
cursor: MEDIA_TRACK_CONSTRAINT.CURSOR.AWLAYS,
|
|
414
|
-
aspectRatio,
|
|
415
|
-
};
|
|
416
|
-
|
|
417
|
-
if (hasCustomConstraints) {
|
|
418
|
-
shareConstraints = hasSharePreferences.shareConstraints;
|
|
419
|
-
} else if (hasHighFrameRate) {
|
|
420
|
-
shareConstraints = {
|
|
421
|
-
...shareConstraints,
|
|
422
|
-
frameRate: videoShareFrameRate,
|
|
423
|
-
height: resolution.idealHeight,
|
|
424
|
-
width: resolution.idealWidth,
|
|
425
|
-
...config.resolution,
|
|
426
|
-
};
|
|
427
|
-
} else {
|
|
428
|
-
shareConstraints = {
|
|
429
|
-
...shareConstraints,
|
|
430
|
-
frameRate: customShareFrameRate || screenFrameRate,
|
|
431
|
-
height: customResolution.idealHeight || screenResolution.idealHeight,
|
|
432
|
-
width: customResolution.idealWidth || screenResolution.idealWidth,
|
|
433
|
-
...config.screenResolution,
|
|
434
|
-
};
|
|
435
|
-
}
|
|
436
|
-
|
|
437
|
-
// chrome and webkit based browsers (edge, safari) automatically adjust everything
|
|
438
|
-
// and we have noticed higher quality with those browser types
|
|
439
|
-
// firefox specifically has some issues with resolution and frame rate decision making
|
|
440
|
-
// so we are making it optional and configurable (with defaults) for firefox
|
|
441
|
-
// to have higher quality, and for developers to control the values
|
|
442
|
-
// eventually we may have to add the same functionality to chrome, OR conversely, get to with firefox
|
|
443
|
-
|
|
444
|
-
if (isBrowser('firefox')) {
|
|
445
|
-
const mediaConfig: any = {
|
|
446
|
-
audio: options.sendAudio,
|
|
447
|
-
video: options.sendShare,
|
|
448
|
-
};
|
|
449
|
-
|
|
450
|
-
return navigator.mediaDevices
|
|
451
|
-
.getDisplayMedia({audio: options.sendAudio, video: mediaConfig})
|
|
452
|
-
.then((stream) => {
|
|
453
|
-
if (options.sendShare && stream.getVideoTracks().length > 0) {
|
|
454
|
-
// Firefox has a bug with the spec where changing in the height and width only happens
|
|
455
|
-
// after we get the inital tracks
|
|
456
|
-
// https://bugzilla.mozilla.org/show_bug.cgi?id=1321221
|
|
457
|
-
stream.getVideoTracks()[0].applyConstraints(shareConstraints);
|
|
458
|
-
}
|
|
459
|
-
|
|
460
|
-
return stream;
|
|
461
|
-
});
|
|
462
|
-
}
|
|
463
|
-
|
|
464
|
-
const getDisplayMediaParams: any = {video: options.sendShare ? shareConstraints : false};
|
|
465
|
-
|
|
466
|
-
// safari doesn't support sending screen share audio
|
|
467
|
-
// https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getDisplayMedia
|
|
468
|
-
if (options.sendAudio && isBrowser('safari')) {
|
|
469
|
-
getDisplayMediaParams.audio = options.sendAudio;
|
|
470
|
-
}
|
|
471
|
-
|
|
472
|
-
return navigator.mediaDevices.getDisplayMedia(getDisplayMediaParams);
|
|
473
|
-
};
|
|
474
|
-
|
|
475
|
-
/**
|
|
476
|
-
* generates audio and video using constraints (often called after getSupportedDevices)
|
|
477
|
-
* @param {Object|Boolean} audio gum constraints
|
|
478
|
-
* @param {Object|Boolean} video gum constraints
|
|
479
|
-
* @param {Object} config SDK Configuration for meetings plugin
|
|
480
|
-
* @returns {Object} {streams}
|
|
481
|
-
*/
|
|
482
|
-
Media.getMedia = (audio: any | boolean, video: any | boolean, config: any) => {
|
|
483
|
-
const defaultWidth = {ideal: config.resolution.idealWidth, max: config.resolution.maxWidth};
|
|
484
|
-
const defaultHeight = {ideal: config.resolution.idealHeight, max: config.resolution.maxHeight};
|
|
485
|
-
const mediaConfig = {
|
|
486
|
-
audio,
|
|
487
|
-
// TODO: Remove temporary workaround once Firefox fixes low constraint issues
|
|
488
|
-
// eslint-disable-next-line no-nested-ternary
|
|
489
|
-
video: video
|
|
490
|
-
? isBrowser('firefox') && video.width && video.width.max === 320
|
|
491
|
-
? {
|
|
492
|
-
deviceId: video.deviceId ? video.deviceId : undefined,
|
|
493
|
-
width: 320,
|
|
494
|
-
height: 180,
|
|
495
|
-
frameRate: video.frameRate ? video.frameRate : undefined,
|
|
496
|
-
facingMode: video.facingMode ? video.facingMode : undefined,
|
|
497
|
-
}
|
|
498
|
-
: {
|
|
499
|
-
deviceId: video.deviceId ? video.deviceId : undefined,
|
|
500
|
-
width: video.width ? video.width : defaultWidth,
|
|
501
|
-
height: video.height ? video.height : defaultHeight,
|
|
502
|
-
frameRate: video.frameRate ? video.frameRate : undefined,
|
|
503
|
-
facingMode: video.facingMode ? video.facingMode : undefined,
|
|
504
|
-
}
|
|
505
|
-
: false,
|
|
506
|
-
fake: process.env.NODE_ENV === 'test', // Special case to get fake media for Firefox browser for testing
|
|
507
|
-
};
|
|
508
|
-
|
|
509
|
-
return navigator.mediaDevices.getUserMedia(mediaConfig).catch((err) => {
|
|
510
|
-
const logPath = 'Media:index#getMedia --> navigator.mediaDevices.getUserMedia';
|
|
511
|
-
|
|
512
|
-
LoggerProxy.logger.error(`${logPath} failed - ${err} (${err.constraint})`);
|
|
513
|
-
throw err;
|
|
514
|
-
});
|
|
515
|
-
};
|
|
516
|
-
|
|
517
|
-
/**
|
|
518
|
-
* Checks if the machine has at least one audio or video device (Dont use this for screen share)
|
|
519
|
-
* @param {object} [options]
|
|
520
|
-
* {
|
|
521
|
-
* sendAudio: true/false,
|
|
522
|
-
* sendVideo: true/false
|
|
523
|
-
* }
|
|
524
|
-
* @returns {Object} {
|
|
525
|
-
* sendAudio: true/false,
|
|
526
|
-
* sendVideo: true/false
|
|
527
|
-
*}
|
|
528
|
-
*/
|
|
529
|
-
Media.getSupportedDevice = ({sendAudio, sendVideo}: {sendAudio: boolean; sendVideo: boolean}) =>
|
|
530
|
-
Promise.resolve().then(() => {
|
|
531
|
-
if (!navigator.mediaDevices || navigator.mediaDevices.enumerateDevices === undefined) {
|
|
532
|
-
return {
|
|
533
|
-
sendAudio: false,
|
|
534
|
-
sendVideo: false,
|
|
535
|
-
};
|
|
536
|
-
}
|
|
537
|
-
|
|
538
|
-
return navigator.mediaDevices.enumerateDevices().then((devices) => {
|
|
539
|
-
const supported = {
|
|
540
|
-
audio: devices.filter((device) => device.kind === AUDIO_INPUT).length > 0,
|
|
541
|
-
video: devices.filter((device) => device.kind === VIDEO_INPUT).length > 0,
|
|
542
|
-
};
|
|
543
|
-
|
|
544
|
-
return {
|
|
545
|
-
sendAudio: supported.audio && sendAudio,
|
|
546
|
-
sendVideo: supported.video && sendVideo,
|
|
547
|
-
};
|
|
548
|
-
});
|
|
549
|
-
});
|
|
550
|
-
|
|
551
|
-
/**
|
|
552
|
-
* proxy to browser navigator.mediaDevices.enumerateDevices()
|
|
553
|
-
* @returns {Promise}
|
|
554
|
-
*/
|
|
555
|
-
Media.getDevices = () => {
|
|
556
|
-
if (navigator && navigator.mediaDevices && navigator.mediaDevices.enumerateDevices) {
|
|
557
|
-
return navigator.mediaDevices.enumerateDevices();
|
|
558
|
-
}
|
|
559
|
-
|
|
560
|
-
return Promise.reject(new MediaError('enumerateDevices not supported.'));
|
|
561
|
-
};
|
|
562
|
-
|
|
563
|
-
/**
|
|
564
|
-
*
|
|
565
|
-
* Toggle a specific stream
|
|
566
|
-
* noop as of now, does nothing
|
|
567
|
-
* @returns {null}
|
|
568
|
-
*/
|
|
569
|
-
Media.toggleStream = () => {};
|
|
570
|
-
|
|
571
|
-
/**
|
|
572
|
-
* Stop input stream
|
|
573
|
-
* @param {MediaTrack} track A media stream
|
|
574
|
-
* @returns {null}
|
|
575
|
-
*/
|
|
576
|
-
Media.stopTracks = (track: any) => {
|
|
577
|
-
if (!track) {
|
|
578
|
-
return Promise.resolve();
|
|
579
|
-
}
|
|
580
|
-
|
|
581
|
-
return Promise.resolve().then(() => {
|
|
582
|
-
if (track && track.stop) {
|
|
583
|
-
try {
|
|
584
|
-
track.stop();
|
|
585
|
-
} catch (e) {
|
|
586
|
-
LoggerProxy.logger.error(
|
|
587
|
-
`Media:index#stopTracks --> Unable to stop the track with state ${track.readyState}, error: ${e}`
|
|
588
|
-
);
|
|
589
|
-
}
|
|
590
|
-
}
|
|
591
|
-
});
|
|
592
|
-
};
|
|
593
|
-
|
|
594
|
-
/**
|
|
595
|
-
*
|
|
596
|
-
* Stop input stream
|
|
597
|
-
* @param {Stream} stream A media stream
|
|
598
|
-
* @returns {null}
|
|
599
|
-
* @deprecated after v1.89.3
|
|
600
|
-
*/
|
|
601
|
-
Media.stopStream = (stream: any) => {
|
|
602
|
-
LoggerProxy.logger.warn(
|
|
603
|
-
'Media:index#stopStream --> [DEPRECATION WARNING]: stopStream has been deprecated after v1.89.3'
|
|
604
|
-
);
|
|
605
|
-
if (!stream) {
|
|
606
|
-
return Promise.resolve();
|
|
607
|
-
}
|
|
608
|
-
|
|
609
|
-
/*
|
|
610
|
-
* To release local media
|
|
611
|
-
* 1) Chrome requires all tracks to be stopped (stream.stop got deprecated)
|
|
612
|
-
* 2) Firefox requires the stream to be stopped
|
|
613
|
-
*/
|
|
614
|
-
return Promise.resolve().then(() => {
|
|
615
|
-
if (stream.getTracks) {
|
|
616
|
-
stream.getTracks().forEach((track) => {
|
|
617
|
-
track.stop();
|
|
618
|
-
});
|
|
619
|
-
} else if (stream.stop) {
|
|
620
|
-
stream.stop();
|
|
621
|
-
}
|
|
622
|
-
});
|
|
623
|
-
};
|
|
624
|
-
|
|
625
|
-
/**
|
|
626
|
-
* generates streams for audio video and share
|
|
627
|
-
* @param {object} mediaSetting parameter
|
|
628
|
-
* @param {Object} mediaSetting.sendAudio sendAudio: {Boolean} sendAudio constraints
|
|
629
|
-
* @param {Object} mediaSetting.sendVideo sendVideo: {Boolean} sendVideo constraints
|
|
630
|
-
* @param {Object} mediaSetting.sendShare sendShare: {Boolean} sendShare constraints
|
|
631
|
-
* @param {Object} mediaSetting.isSharing isSharing: {Boolean} isSharing constraints
|
|
632
|
-
* @param {Object} audioVideo parameter
|
|
633
|
-
* @param {Object} audioVideo.audio {deviceId: {String}}
|
|
634
|
-
* @param {Object} audioVideo.video {deviceId: {String}}
|
|
635
|
-
* @param {Object} sharePreferences parameter
|
|
636
|
-
* @param {Object} sharePreferences.shareConstraints parameter
|
|
637
|
-
* @param {Boolean} sharePreferences.highFrameRate parameter
|
|
638
|
-
* @param {Object} config SDK Config
|
|
639
|
-
* @returns {Array} [localStream, shareStream]
|
|
640
|
-
*/
|
|
641
|
-
Media.getUserMedia = (
|
|
642
|
-
mediaSetting: {
|
|
643
|
-
sendAudio: object;
|
|
644
|
-
sendVideo: object;
|
|
645
|
-
sendShare: object;
|
|
646
|
-
isSharing: object;
|
|
647
|
-
},
|
|
648
|
-
audioVideo: {
|
|
649
|
-
audio: object;
|
|
650
|
-
video: object;
|
|
651
|
-
},
|
|
652
|
-
sharePreferences: {
|
|
653
|
-
shareConstraints: object;
|
|
654
|
-
highFrameRate: boolean;
|
|
655
|
-
},
|
|
656
|
-
config: object
|
|
657
|
-
) =>
|
|
658
|
-
Media.getLocalMedia(
|
|
659
|
-
{
|
|
660
|
-
sendAudio: mediaSetting.sendAudio ? audioVideo.audio || mediaSetting.sendAudio : false,
|
|
661
|
-
sendVideo: mediaSetting.sendVideo ? audioVideo.video || mediaSetting.sendVideo : false,
|
|
662
|
-
},
|
|
663
|
-
config
|
|
664
|
-
).then((localStream) =>
|
|
665
|
-
Media.getLocalMedia(
|
|
666
|
-
{
|
|
667
|
-
sendShare: mediaSetting.sendShare,
|
|
668
|
-
isSharing: mediaSetting.isSharing,
|
|
669
|
-
sharePreferences,
|
|
670
|
-
},
|
|
671
|
-
config
|
|
672
|
-
).then((shareStream) => [localStream, shareStream])
|
|
673
|
-
);
|
|
674
|
-
|
|
675
|
-
export default Media;
|
|
1
|
+
/*!
|
|
2
|
+
* Copyright (c) 2015-2020 Cisco Systems, Inc. See LICENSE file.
|
|
3
|
+
*/
|
|
4
|
+
/* globals navigator */
|
|
5
|
+
|
|
6
|
+
import LoggerProxy from '../common/logs/logger-proxy';
|
|
7
|
+
import {
|
|
8
|
+
AUDIO_INPUT,
|
|
9
|
+
VIDEO_INPUT,
|
|
10
|
+
PEER_CONNECTION_STATE,
|
|
11
|
+
MEDIA_TRACK_CONSTRAINT,
|
|
12
|
+
} from '../constants';
|
|
13
|
+
import Config from '../config';
|
|
14
|
+
import PeerConnectionManager from '../peer-connection-manager';
|
|
15
|
+
import ReconnectionError from '../common/errors/reconnection';
|
|
16
|
+
import MediaError from '../common/errors/media';
|
|
17
|
+
import BrowserDetection from '../common/browser-detection';
|
|
18
|
+
|
|
19
|
+
const {isBrowser} = BrowserDetection();
|
|
20
|
+
|
|
21
|
+
/**
|
|
22
|
+
* MediaDirection
|
|
23
|
+
* @typedef {Object} MediaDirection
|
|
24
|
+
* @property {boolean} sendAudio
|
|
25
|
+
* @property {boolean} receiveAudio
|
|
26
|
+
* @property {boolean} sendVideo
|
|
27
|
+
* @property {boolean} receiveVideo
|
|
28
|
+
* @property {boolean} sendShare
|
|
29
|
+
* @property {boolean} receiveShare
|
|
30
|
+
*/
|
|
31
|
+
|
|
32
|
+
/**
|
|
33
|
+
* SendOptions
|
|
34
|
+
* @typedef {Object} SendOptions
|
|
35
|
+
* @property sendAudio
|
|
36
|
+
* @property sendVideo
|
|
37
|
+
* @property sendShare
|
|
38
|
+
* @property isSharing
|
|
39
|
+
* @property {Object} sharePreferences
|
|
40
|
+
*/
|
|
41
|
+
/**
|
|
42
|
+
*
|
|
43
|
+
* @public
|
|
44
|
+
* @export
|
|
45
|
+
* Mimic browser APIs as "the ultimate browser".
|
|
46
|
+
* Handles the quirks of each browser.
|
|
47
|
+
* Extends and enhances adapter.js, i.e., the "media" file from the web client.
|
|
48
|
+
*/
|
|
49
|
+
const Media: any = {};
|
|
50
|
+
|
|
51
|
+
/**
|
|
52
|
+
* @param {boolean} enabled
|
|
53
|
+
* @param {MediaStreamTrack} track
|
|
54
|
+
* @returns {Boolean}
|
|
55
|
+
* @public
|
|
56
|
+
*/
|
|
57
|
+
Media.setLocalTrack = (enabled: boolean, track: MediaStreamTrack) => {
|
|
58
|
+
if (track) {
|
|
59
|
+
track.enabled = enabled;
|
|
60
|
+
|
|
61
|
+
return true;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
return false;
|
|
65
|
+
};
|
|
66
|
+
|
|
67
|
+
/**
|
|
68
|
+
* @param {RTCPeerConnection} peerConnection
|
|
69
|
+
* @param {Object} meetingProperties
|
|
70
|
+
* @param {string} meetingProperties.meetingId
|
|
71
|
+
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
72
|
+
* @returns {Promise}
|
|
73
|
+
*/
|
|
74
|
+
Media.reconnectMedia = (
|
|
75
|
+
peerConnection: RTCPeerConnection,
|
|
76
|
+
{
|
|
77
|
+
meetingId,
|
|
78
|
+
remoteQualityLevel,
|
|
79
|
+
enableRtx,
|
|
80
|
+
enableExtmap,
|
|
81
|
+
}: {
|
|
82
|
+
meetingId: string;
|
|
83
|
+
remoteQualityLevel: string;
|
|
84
|
+
enableRtx: any;
|
|
85
|
+
enableExtmap: any;
|
|
86
|
+
}
|
|
87
|
+
) => {
|
|
88
|
+
if (
|
|
89
|
+
peerConnection.connectionState === PEER_CONNECTION_STATE.CLOSED ||
|
|
90
|
+
peerConnection.connectionState === PEER_CONNECTION_STATE.FAILED
|
|
91
|
+
) {
|
|
92
|
+
return Promise.reject(new ReconnectionError('Reinitiate peerconnection'));
|
|
93
|
+
}
|
|
94
|
+
|
|
95
|
+
return PeerConnectionManager.createOffer(peerConnection, {
|
|
96
|
+
meetingId,
|
|
97
|
+
remoteQualityLevel,
|
|
98
|
+
enableRtx,
|
|
99
|
+
enableExtmap,
|
|
100
|
+
});
|
|
101
|
+
};
|
|
102
|
+
|
|
103
|
+
/**
|
|
104
|
+
* format the media array for send
|
|
105
|
+
* @param {String} mediaId
|
|
106
|
+
* @param {Boolean} audioMuted
|
|
107
|
+
* @param {Boolean} videoMuted
|
|
108
|
+
* @returns {Array} medias
|
|
109
|
+
*/
|
|
110
|
+
Media.generateLocalMedias = (mediaId: string, audioMuted: boolean, videoMuted: boolean) => {
|
|
111
|
+
if (mediaId) {
|
|
112
|
+
return [
|
|
113
|
+
{
|
|
114
|
+
localSdp: JSON.stringify({
|
|
115
|
+
audioMuted,
|
|
116
|
+
videoMuted,
|
|
117
|
+
}),
|
|
118
|
+
mediaId,
|
|
119
|
+
},
|
|
120
|
+
];
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
return [];
|
|
124
|
+
};
|
|
125
|
+
|
|
126
|
+
/**
|
|
127
|
+
* make a browser call to get the media
|
|
128
|
+
* @param {SendOptions} options
|
|
129
|
+
* @param {Object} config SDK Configuration for meetings plugin
|
|
130
|
+
* @returns {Promise}
|
|
131
|
+
*/
|
|
132
|
+
Media.getLocalMedia = (options: any, config: object) => {
|
|
133
|
+
const {sendAudio, sendVideo, sendShare, sharePreferences, isSharing} = options;
|
|
134
|
+
|
|
135
|
+
if (sendAudio || sendVideo) {
|
|
136
|
+
return Media.getMedia(sendAudio, sendVideo, config);
|
|
137
|
+
}
|
|
138
|
+
|
|
139
|
+
if (sendShare && !isSharing) {
|
|
140
|
+
return Media.getDisplayMedia(
|
|
141
|
+
{
|
|
142
|
+
sendAudio: false,
|
|
143
|
+
sendShare: true,
|
|
144
|
+
sharePreferences,
|
|
145
|
+
},
|
|
146
|
+
config
|
|
147
|
+
);
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
return Promise.resolve(undefined);
|
|
151
|
+
};
|
|
152
|
+
/**
|
|
153
|
+
* Returns the direction and tracks
|
|
154
|
+
* @param {string} trackType type of track (audio/video)
|
|
155
|
+
* @param {object} track tracks passed
|
|
156
|
+
* @param {boolean} receiveTracks do you want to receive tracks from the remote side
|
|
157
|
+
* @returns {Object} returns direction tracks to be added in transceiver
|
|
158
|
+
*/
|
|
159
|
+
Media.checkTracks = (trackType: string, track: object, receiveTracks: boolean) => {
|
|
160
|
+
// eslint-disable-next-line @typescript-eslint/no-shadow
|
|
161
|
+
const getDirection = (sendTracks, receiveTracks) => {
|
|
162
|
+
if (sendTracks && receiveTracks) {
|
|
163
|
+
return 'sendrecv';
|
|
164
|
+
}
|
|
165
|
+
if (sendTracks && !receiveTracks) {
|
|
166
|
+
return 'sendonly';
|
|
167
|
+
}
|
|
168
|
+
if (!sendTracks && receiveTracks) {
|
|
169
|
+
return 'recvonly';
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
return 'inactive';
|
|
173
|
+
};
|
|
174
|
+
|
|
175
|
+
if (track) {
|
|
176
|
+
return {track, direction: getDirection(!!track, receiveTracks)};
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
return {track: trackType, direction: getDirection(!!track, receiveTracks)};
|
|
180
|
+
};
|
|
181
|
+
/**
|
|
182
|
+
* creates peerconnection and attaches streams
|
|
183
|
+
* @param {MediaDirection} mediaProperties
|
|
184
|
+
* @param {Object} meetingProperties
|
|
185
|
+
* @param {string} meetingProperties.meetingId
|
|
186
|
+
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
187
|
+
* @returns {Array} [peerConnection, ]
|
|
188
|
+
*/
|
|
189
|
+
Media.attachMedia = (
|
|
190
|
+
mediaProperties: any,
|
|
191
|
+
{
|
|
192
|
+
meetingId,
|
|
193
|
+
remoteQualityLevel,
|
|
194
|
+
enableRtx,
|
|
195
|
+
enableExtmap,
|
|
196
|
+
}: {
|
|
197
|
+
meetingId: string;
|
|
198
|
+
remoteQualityLevel: string;
|
|
199
|
+
enableRtx: any;
|
|
200
|
+
enableExtmap: any;
|
|
201
|
+
}
|
|
202
|
+
) => {
|
|
203
|
+
const {mediaDirection, audioTrack, videoTrack, shareTrack, peerConnection} = mediaProperties;
|
|
204
|
+
|
|
205
|
+
let result = null;
|
|
206
|
+
|
|
207
|
+
// Add Transceiver for audio
|
|
208
|
+
result = Media.checkTracks(
|
|
209
|
+
'audio',
|
|
210
|
+
mediaDirection.sendAudio && audioTrack,
|
|
211
|
+
mediaDirection.receiveAudio
|
|
212
|
+
);
|
|
213
|
+
peerConnection.audioTransceiver = peerConnection.addTransceiver(result.track, {
|
|
214
|
+
direction: result.direction,
|
|
215
|
+
});
|
|
216
|
+
|
|
217
|
+
// Add Transceiver for video
|
|
218
|
+
result = Media.checkTracks(
|
|
219
|
+
'video',
|
|
220
|
+
mediaDirection.sendVideo && videoTrack,
|
|
221
|
+
mediaDirection.receiveVideo
|
|
222
|
+
);
|
|
223
|
+
peerConnection.videoTransceiver = peerConnection.addTransceiver(result.track, {
|
|
224
|
+
direction: result.direction,
|
|
225
|
+
});
|
|
226
|
+
|
|
227
|
+
// Add Transceiver for share
|
|
228
|
+
result = Media.checkTracks(
|
|
229
|
+
'video',
|
|
230
|
+
mediaDirection.sendShare && shareTrack,
|
|
231
|
+
mediaDirection.receiveShare
|
|
232
|
+
);
|
|
233
|
+
peerConnection.shareTransceiver = peerConnection.addTransceiver(result.track, {
|
|
234
|
+
direction: result.direction,
|
|
235
|
+
});
|
|
236
|
+
|
|
237
|
+
peerConnection.onnegotiationneeded = (event) => {
|
|
238
|
+
LoggerProxy.logger.info(
|
|
239
|
+
`Media:index#attachMedia --> onnegotiationneeded#PeerConnection: ${event}`
|
|
240
|
+
);
|
|
241
|
+
};
|
|
242
|
+
|
|
243
|
+
return PeerConnectionManager.createOffer(peerConnection, {
|
|
244
|
+
meetingId,
|
|
245
|
+
remoteQualityLevel,
|
|
246
|
+
enableRtx,
|
|
247
|
+
enableExtmap,
|
|
248
|
+
});
|
|
249
|
+
};
|
|
250
|
+
|
|
251
|
+
/**
|
|
252
|
+
* updates all the media streams and creates a new media offer
|
|
253
|
+
* @param {MediaDirection} mediaProperties
|
|
254
|
+
* @param {Object} meetingProperties
|
|
255
|
+
* @param {string} meetingProperties.meetingId
|
|
256
|
+
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
257
|
+
* @returns {Promise}
|
|
258
|
+
*/
|
|
259
|
+
Media.updateMedia = (
|
|
260
|
+
mediaProperties: any,
|
|
261
|
+
{
|
|
262
|
+
meetingId,
|
|
263
|
+
remoteQualityLevel,
|
|
264
|
+
enableRtx,
|
|
265
|
+
enableExtmap,
|
|
266
|
+
}: {
|
|
267
|
+
meetingId: string;
|
|
268
|
+
remoteQualityLevel: string;
|
|
269
|
+
enableRtx: any;
|
|
270
|
+
enableExtmap: any;
|
|
271
|
+
}
|
|
272
|
+
) => {
|
|
273
|
+
const {mediaDirection, audioTrack, videoTrack, shareTrack, peerConnection} = mediaProperties;
|
|
274
|
+
|
|
275
|
+
// update audio transceiver
|
|
276
|
+
Media.setTrackOnTransceiver(peerConnection.audioTransceiver, {
|
|
277
|
+
type: 'audio',
|
|
278
|
+
track: audioTrack,
|
|
279
|
+
sendTrack: mediaDirection.sendAudio && audioTrack,
|
|
280
|
+
receiveTrack: mediaDirection.receiveAudio,
|
|
281
|
+
});
|
|
282
|
+
|
|
283
|
+
// update video transceiver
|
|
284
|
+
Media.setTrackOnTransceiver(peerConnection.videoTransceiver, {
|
|
285
|
+
type: 'video',
|
|
286
|
+
track: videoTrack,
|
|
287
|
+
sendTrack: mediaDirection.sendVideo && videoTrack,
|
|
288
|
+
receiveTrack: mediaDirection.receiveVideo,
|
|
289
|
+
});
|
|
290
|
+
|
|
291
|
+
// update content transceiver
|
|
292
|
+
Media.setTrackOnTransceiver(peerConnection.shareTransceiver, {
|
|
293
|
+
type: 'video',
|
|
294
|
+
track: shareTrack,
|
|
295
|
+
sendTrack: mediaDirection.sendShare && shareTrack,
|
|
296
|
+
receiveTrack: mediaDirection.receiveShare,
|
|
297
|
+
});
|
|
298
|
+
peerConnection.onnegotiationneeded = (event) => {
|
|
299
|
+
LoggerProxy.logger.info(
|
|
300
|
+
`Media:index#updateMedia --> onnegotiationneeded#PeerConnection: ${event}`
|
|
301
|
+
);
|
|
302
|
+
};
|
|
303
|
+
|
|
304
|
+
return PeerConnectionManager.createOffer(peerConnection, {
|
|
305
|
+
meetingId,
|
|
306
|
+
remoteQualityLevel,
|
|
307
|
+
enableRtx,
|
|
308
|
+
enableExtmap,
|
|
309
|
+
});
|
|
310
|
+
};
|
|
311
|
+
|
|
312
|
+
/**
|
|
313
|
+
* @param {RTCRtpTransceiver} transceiver
|
|
314
|
+
* @param {Object} options
|
|
315
|
+
* @param {MediaStreamTrack} options.track
|
|
316
|
+
* @returns {undefined}
|
|
317
|
+
*/
|
|
318
|
+
Media.setTrackOnTransceiver = (
|
|
319
|
+
transceiver: RTCRtpTransceiver,
|
|
320
|
+
options: {
|
|
321
|
+
track: MediaStreamTrack;
|
|
322
|
+
type: any;
|
|
323
|
+
sendTrack: any;
|
|
324
|
+
receiveTrack: any;
|
|
325
|
+
}
|
|
326
|
+
) => {
|
|
327
|
+
const {type, track, sendTrack, receiveTrack} = options;
|
|
328
|
+
|
|
329
|
+
try {
|
|
330
|
+
const result = Media.checkTracks(type, sendTrack && track, receiveTrack);
|
|
331
|
+
|
|
332
|
+
transceiver.direction = result.direction;
|
|
333
|
+
if (options.track) {
|
|
334
|
+
transceiver.sender.replaceTrack(track);
|
|
335
|
+
}
|
|
336
|
+
} catch (e) {
|
|
337
|
+
LoggerProxy.logger.error(`Media:index#setTrackOnTransceiver --> ${e}`);
|
|
338
|
+
throw e;
|
|
339
|
+
}
|
|
340
|
+
};
|
|
341
|
+
|
|
342
|
+
/**
|
|
343
|
+
* creates a new offer
|
|
344
|
+
* @param {Object} meetingProperties
|
|
345
|
+
* @param {string} meetingProperties.meetingId
|
|
346
|
+
* @param {string} meetingProperties.remoteQualityLevel LOW|MEDIUM|HIGH
|
|
347
|
+
* @param {RTCPeerConnection} peerConnection
|
|
348
|
+
* @param {RTCRtpTransceiver} transceiver
|
|
349
|
+
* @param {Object} options see #Media.setTrackOnTransceiver
|
|
350
|
+
* @returns {Promise}
|
|
351
|
+
*/
|
|
352
|
+
Media.updateTransceiver = (
|
|
353
|
+
{
|
|
354
|
+
meetingId,
|
|
355
|
+
remoteQualityLevel,
|
|
356
|
+
enableRtx,
|
|
357
|
+
enableExtmap,
|
|
358
|
+
}: {
|
|
359
|
+
meetingId: string;
|
|
360
|
+
remoteQualityLevel: string;
|
|
361
|
+
enableRtx: any;
|
|
362
|
+
enableExtmap: any;
|
|
363
|
+
},
|
|
364
|
+
peerConnection: RTCPeerConnection,
|
|
365
|
+
transceiver: RTCRtpTransceiver,
|
|
366
|
+
options: object
|
|
367
|
+
) => {
|
|
368
|
+
Media.setTrackOnTransceiver(transceiver, options);
|
|
369
|
+
|
|
370
|
+
return PeerConnectionManager.createOffer(peerConnection, {
|
|
371
|
+
meetingId,
|
|
372
|
+
remoteQualityLevel,
|
|
373
|
+
enableRtx,
|
|
374
|
+
enableExtmap,
|
|
375
|
+
});
|
|
376
|
+
};
|
|
377
|
+
|
|
378
|
+
/**
|
|
379
|
+
* generates share streams
|
|
380
|
+
* @param {Object} options parameter
|
|
381
|
+
* @param {Boolean} options.sendAudio send audio from the display share
|
|
382
|
+
* @param {Boolean} options.sendShare send video from the display share
|
|
383
|
+
* @param {Object} options.sharePreferences
|
|
384
|
+
* @param {MediaTrackConstraints} options.sharePreferences.shareConstraints constraints to apply to video
|
|
385
|
+
* @see {@link https://developer.mozilla.org/en-US/docs/Web/API/MediaTrackConstraints}
|
|
386
|
+
* @param {Boolean} options.sharePreferences.highFrameRate if shareConstraints isn't provided, set default values based off of this boolean
|
|
387
|
+
* @param {Object} config SDK Configuration for meetings plugin
|
|
388
|
+
* @returns {Promise.<MediaStream>}
|
|
389
|
+
*/
|
|
390
|
+
Media.getDisplayMedia = (
|
|
391
|
+
options: {
|
|
392
|
+
sendAudio: boolean;
|
|
393
|
+
sendShare: boolean;
|
|
394
|
+
sharePreferences: {
|
|
395
|
+
shareConstraints: MediaTrackConstraints;
|
|
396
|
+
highFrameRate: any;
|
|
397
|
+
};
|
|
398
|
+
},
|
|
399
|
+
config: any = {}
|
|
400
|
+
) => {
|
|
401
|
+
// SDK screen share resolution settings from Webex.init
|
|
402
|
+
const customResolution = config.screenResolution || {};
|
|
403
|
+
// user defined screen share frame rate
|
|
404
|
+
const customShareFrameRate = config.screenFrameRate || null;
|
|
405
|
+
// user defined share preferences
|
|
406
|
+
const hasSharePreferences = options.sharePreferences;
|
|
407
|
+
const hasCustomConstraints = hasSharePreferences && hasSharePreferences.shareConstraints;
|
|
408
|
+
const hasHighFrameRate = hasSharePreferences && hasSharePreferences.highFrameRate;
|
|
409
|
+
const {screenResolution, resolution, videoShareFrameRate, screenFrameRate, aspectRatio} =
|
|
410
|
+
Config.meetings;
|
|
411
|
+
|
|
412
|
+
let shareConstraints: any = {
|
|
413
|
+
cursor: MEDIA_TRACK_CONSTRAINT.CURSOR.AWLAYS,
|
|
414
|
+
aspectRatio,
|
|
415
|
+
};
|
|
416
|
+
|
|
417
|
+
if (hasCustomConstraints) {
|
|
418
|
+
shareConstraints = hasSharePreferences.shareConstraints;
|
|
419
|
+
} else if (hasHighFrameRate) {
|
|
420
|
+
shareConstraints = {
|
|
421
|
+
...shareConstraints,
|
|
422
|
+
frameRate: videoShareFrameRate,
|
|
423
|
+
height: resolution.idealHeight,
|
|
424
|
+
width: resolution.idealWidth,
|
|
425
|
+
...config.resolution,
|
|
426
|
+
};
|
|
427
|
+
} else {
|
|
428
|
+
shareConstraints = {
|
|
429
|
+
...shareConstraints,
|
|
430
|
+
frameRate: customShareFrameRate || screenFrameRate,
|
|
431
|
+
height: customResolution.idealHeight || screenResolution.idealHeight,
|
|
432
|
+
width: customResolution.idealWidth || screenResolution.idealWidth,
|
|
433
|
+
...config.screenResolution,
|
|
434
|
+
};
|
|
435
|
+
}
|
|
436
|
+
|
|
437
|
+
// chrome and webkit based browsers (edge, safari) automatically adjust everything
|
|
438
|
+
// and we have noticed higher quality with those browser types
|
|
439
|
+
// firefox specifically has some issues with resolution and frame rate decision making
|
|
440
|
+
// so we are making it optional and configurable (with defaults) for firefox
|
|
441
|
+
// to have higher quality, and for developers to control the values
|
|
442
|
+
// eventually we may have to add the same functionality to chrome, OR conversely, get to with firefox
|
|
443
|
+
|
|
444
|
+
if (isBrowser('firefox')) {
|
|
445
|
+
const mediaConfig: any = {
|
|
446
|
+
audio: options.sendAudio,
|
|
447
|
+
video: options.sendShare,
|
|
448
|
+
};
|
|
449
|
+
|
|
450
|
+
return navigator.mediaDevices
|
|
451
|
+
.getDisplayMedia({audio: options.sendAudio, video: mediaConfig})
|
|
452
|
+
.then((stream) => {
|
|
453
|
+
if (options.sendShare && stream.getVideoTracks().length > 0) {
|
|
454
|
+
// Firefox has a bug with the spec where changing in the height and width only happens
|
|
455
|
+
// after we get the inital tracks
|
|
456
|
+
// https://bugzilla.mozilla.org/show_bug.cgi?id=1321221
|
|
457
|
+
stream.getVideoTracks()[0].applyConstraints(shareConstraints);
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
return stream;
|
|
461
|
+
});
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
const getDisplayMediaParams: any = {video: options.sendShare ? shareConstraints : false};
|
|
465
|
+
|
|
466
|
+
// safari doesn't support sending screen share audio
|
|
467
|
+
// https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getDisplayMedia
|
|
468
|
+
if (options.sendAudio && isBrowser('safari')) {
|
|
469
|
+
getDisplayMediaParams.audio = options.sendAudio;
|
|
470
|
+
}
|
|
471
|
+
|
|
472
|
+
return navigator.mediaDevices.getDisplayMedia(getDisplayMediaParams);
|
|
473
|
+
};
|
|
474
|
+
|
|
475
|
+
/**
|
|
476
|
+
* generates audio and video using constraints (often called after getSupportedDevices)
|
|
477
|
+
* @param {Object|Boolean} audio gum constraints
|
|
478
|
+
* @param {Object|Boolean} video gum constraints
|
|
479
|
+
* @param {Object} config SDK Configuration for meetings plugin
|
|
480
|
+
* @returns {Object} {streams}
|
|
481
|
+
*/
|
|
482
|
+
Media.getMedia = (audio: any | boolean, video: any | boolean, config: any) => {
|
|
483
|
+
const defaultWidth = {ideal: config.resolution.idealWidth, max: config.resolution.maxWidth};
|
|
484
|
+
const defaultHeight = {ideal: config.resolution.idealHeight, max: config.resolution.maxHeight};
|
|
485
|
+
const mediaConfig = {
|
|
486
|
+
audio,
|
|
487
|
+
// TODO: Remove temporary workaround once Firefox fixes low constraint issues
|
|
488
|
+
// eslint-disable-next-line no-nested-ternary
|
|
489
|
+
video: video
|
|
490
|
+
? isBrowser('firefox') && video.width && video.width.max === 320
|
|
491
|
+
? {
|
|
492
|
+
deviceId: video.deviceId ? video.deviceId : undefined,
|
|
493
|
+
width: 320,
|
|
494
|
+
height: 180,
|
|
495
|
+
frameRate: video.frameRate ? video.frameRate : undefined,
|
|
496
|
+
facingMode: video.facingMode ? video.facingMode : undefined,
|
|
497
|
+
}
|
|
498
|
+
: {
|
|
499
|
+
deviceId: video.deviceId ? video.deviceId : undefined,
|
|
500
|
+
width: video.width ? video.width : defaultWidth,
|
|
501
|
+
height: video.height ? video.height : defaultHeight,
|
|
502
|
+
frameRate: video.frameRate ? video.frameRate : undefined,
|
|
503
|
+
facingMode: video.facingMode ? video.facingMode : undefined,
|
|
504
|
+
}
|
|
505
|
+
: false,
|
|
506
|
+
fake: process.env.NODE_ENV === 'test', // Special case to get fake media for Firefox browser for testing
|
|
507
|
+
};
|
|
508
|
+
|
|
509
|
+
return navigator.mediaDevices.getUserMedia(mediaConfig).catch((err) => {
|
|
510
|
+
const logPath = 'Media:index#getMedia --> navigator.mediaDevices.getUserMedia';
|
|
511
|
+
|
|
512
|
+
LoggerProxy.logger.error(`${logPath} failed - ${err} (${err.constraint})`);
|
|
513
|
+
throw err;
|
|
514
|
+
});
|
|
515
|
+
};
|
|
516
|
+
|
|
517
|
+
/**
|
|
518
|
+
* Checks if the machine has at least one audio or video device (Dont use this for screen share)
|
|
519
|
+
* @param {object} [options]
|
|
520
|
+
* {
|
|
521
|
+
* sendAudio: true/false,
|
|
522
|
+
* sendVideo: true/false
|
|
523
|
+
* }
|
|
524
|
+
* @returns {Object} {
|
|
525
|
+
* sendAudio: true/false,
|
|
526
|
+
* sendVideo: true/false
|
|
527
|
+
*}
|
|
528
|
+
*/
|
|
529
|
+
Media.getSupportedDevice = ({sendAudio, sendVideo}: {sendAudio: boolean; sendVideo: boolean}) =>
|
|
530
|
+
Promise.resolve().then(() => {
|
|
531
|
+
if (!navigator.mediaDevices || navigator.mediaDevices.enumerateDevices === undefined) {
|
|
532
|
+
return {
|
|
533
|
+
sendAudio: false,
|
|
534
|
+
sendVideo: false,
|
|
535
|
+
};
|
|
536
|
+
}
|
|
537
|
+
|
|
538
|
+
return navigator.mediaDevices.enumerateDevices().then((devices) => {
|
|
539
|
+
const supported = {
|
|
540
|
+
audio: devices.filter((device) => device.kind === AUDIO_INPUT).length > 0,
|
|
541
|
+
video: devices.filter((device) => device.kind === VIDEO_INPUT).length > 0,
|
|
542
|
+
};
|
|
543
|
+
|
|
544
|
+
return {
|
|
545
|
+
sendAudio: supported.audio && sendAudio,
|
|
546
|
+
sendVideo: supported.video && sendVideo,
|
|
547
|
+
};
|
|
548
|
+
});
|
|
549
|
+
});
|
|
550
|
+
|
|
551
|
+
/**
|
|
552
|
+
* proxy to browser navigator.mediaDevices.enumerateDevices()
|
|
553
|
+
* @returns {Promise}
|
|
554
|
+
*/
|
|
555
|
+
Media.getDevices = () => {
|
|
556
|
+
if (navigator && navigator.mediaDevices && navigator.mediaDevices.enumerateDevices) {
|
|
557
|
+
return navigator.mediaDevices.enumerateDevices();
|
|
558
|
+
}
|
|
559
|
+
|
|
560
|
+
return Promise.reject(new MediaError('enumerateDevices not supported.'));
|
|
561
|
+
};
|
|
562
|
+
|
|
563
|
+
/**
|
|
564
|
+
*
|
|
565
|
+
* Toggle a specific stream
|
|
566
|
+
* noop as of now, does nothing
|
|
567
|
+
* @returns {null}
|
|
568
|
+
*/
|
|
569
|
+
Media.toggleStream = () => {};
|
|
570
|
+
|
|
571
|
+
/**
|
|
572
|
+
* Stop input stream
|
|
573
|
+
* @param {MediaTrack} track A media stream
|
|
574
|
+
* @returns {null}
|
|
575
|
+
*/
|
|
576
|
+
Media.stopTracks = (track: any) => {
|
|
577
|
+
if (!track) {
|
|
578
|
+
return Promise.resolve();
|
|
579
|
+
}
|
|
580
|
+
|
|
581
|
+
return Promise.resolve().then(() => {
|
|
582
|
+
if (track && track.stop) {
|
|
583
|
+
try {
|
|
584
|
+
track.stop();
|
|
585
|
+
} catch (e) {
|
|
586
|
+
LoggerProxy.logger.error(
|
|
587
|
+
`Media:index#stopTracks --> Unable to stop the track with state ${track.readyState}, error: ${e}`
|
|
588
|
+
);
|
|
589
|
+
}
|
|
590
|
+
}
|
|
591
|
+
});
|
|
592
|
+
};
|
|
593
|
+
|
|
594
|
+
/**
|
|
595
|
+
*
|
|
596
|
+
* Stop input stream
|
|
597
|
+
* @param {Stream} stream A media stream
|
|
598
|
+
* @returns {null}
|
|
599
|
+
* @deprecated after v1.89.3
|
|
600
|
+
*/
|
|
601
|
+
Media.stopStream = (stream: any) => {
|
|
602
|
+
LoggerProxy.logger.warn(
|
|
603
|
+
'Media:index#stopStream --> [DEPRECATION WARNING]: stopStream has been deprecated after v1.89.3'
|
|
604
|
+
);
|
|
605
|
+
if (!stream) {
|
|
606
|
+
return Promise.resolve();
|
|
607
|
+
}
|
|
608
|
+
|
|
609
|
+
/*
|
|
610
|
+
* To release local media
|
|
611
|
+
* 1) Chrome requires all tracks to be stopped (stream.stop got deprecated)
|
|
612
|
+
* 2) Firefox requires the stream to be stopped
|
|
613
|
+
*/
|
|
614
|
+
return Promise.resolve().then(() => {
|
|
615
|
+
if (stream.getTracks) {
|
|
616
|
+
stream.getTracks().forEach((track) => {
|
|
617
|
+
track.stop();
|
|
618
|
+
});
|
|
619
|
+
} else if (stream.stop) {
|
|
620
|
+
stream.stop();
|
|
621
|
+
}
|
|
622
|
+
});
|
|
623
|
+
};
|
|
624
|
+
|
|
625
|
+
/**
|
|
626
|
+
* generates streams for audio video and share
|
|
627
|
+
* @param {object} mediaSetting parameter
|
|
628
|
+
* @param {Object} mediaSetting.sendAudio sendAudio: {Boolean} sendAudio constraints
|
|
629
|
+
* @param {Object} mediaSetting.sendVideo sendVideo: {Boolean} sendVideo constraints
|
|
630
|
+
* @param {Object} mediaSetting.sendShare sendShare: {Boolean} sendShare constraints
|
|
631
|
+
* @param {Object} mediaSetting.isSharing isSharing: {Boolean} isSharing constraints
|
|
632
|
+
* @param {Object} audioVideo parameter
|
|
633
|
+
* @param {Object} audioVideo.audio {deviceId: {String}}
|
|
634
|
+
* @param {Object} audioVideo.video {deviceId: {String}}
|
|
635
|
+
* @param {Object} sharePreferences parameter
|
|
636
|
+
* @param {Object} sharePreferences.shareConstraints parameter
|
|
637
|
+
* @param {Boolean} sharePreferences.highFrameRate parameter
|
|
638
|
+
* @param {Object} config SDK Config
|
|
639
|
+
* @returns {Array} [localStream, shareStream]
|
|
640
|
+
*/
|
|
641
|
+
Media.getUserMedia = (
|
|
642
|
+
mediaSetting: {
|
|
643
|
+
sendAudio: object;
|
|
644
|
+
sendVideo: object;
|
|
645
|
+
sendShare: object;
|
|
646
|
+
isSharing: object;
|
|
647
|
+
},
|
|
648
|
+
audioVideo: {
|
|
649
|
+
audio: object;
|
|
650
|
+
video: object;
|
|
651
|
+
},
|
|
652
|
+
sharePreferences: {
|
|
653
|
+
shareConstraints: object;
|
|
654
|
+
highFrameRate: boolean;
|
|
655
|
+
},
|
|
656
|
+
config: object
|
|
657
|
+
) =>
|
|
658
|
+
Media.getLocalMedia(
|
|
659
|
+
{
|
|
660
|
+
sendAudio: mediaSetting.sendAudio ? audioVideo.audio || mediaSetting.sendAudio : false,
|
|
661
|
+
sendVideo: mediaSetting.sendVideo ? audioVideo.video || mediaSetting.sendVideo : false,
|
|
662
|
+
},
|
|
663
|
+
config
|
|
664
|
+
).then((localStream) =>
|
|
665
|
+
Media.getLocalMedia(
|
|
666
|
+
{
|
|
667
|
+
sendShare: mediaSetting.sendShare,
|
|
668
|
+
isSharing: mediaSetting.isSharing,
|
|
669
|
+
sharePreferences,
|
|
670
|
+
},
|
|
671
|
+
config
|
|
672
|
+
).then((shareStream) => [localStream, shareStream])
|
|
673
|
+
);
|
|
674
|
+
|
|
675
|
+
export default Media;
|