2020-06-18 23:15:49 +00:00
|
|
|
/* global APP, JitsiMeetJS, config, interfaceConfig */
|
2016-11-11 15:00:54 +00:00
|
|
|
|
2021-04-07 15:03:20 +00:00
|
|
|
import { jitsiLocalStorage } from '@jitsi/js-utils';
|
2020-05-20 10:57:03 +00:00
|
|
|
import EventEmitter from 'events';
|
|
|
|
import Logger from 'jitsi-meet-logger';
|
2016-10-12 00:08:24 +00:00
|
|
|
|
2020-05-20 10:57:03 +00:00
|
|
|
import { openConnection } from './connection';
|
2020-03-20 11:51:26 +00:00
|
|
|
import { ENDPOINT_TEXT_MESSAGE_NAME } from './modules/API/constants';
|
2021-04-12 07:37:39 +00:00
|
|
|
import { AUDIO_ONLY_SCREEN_SHARE_NO_TRACK } from './modules/UI/UIErrors';
|
2016-01-06 22:39:13 +00:00
|
|
|
import AuthHandler from './modules/UI/authentication/AuthHandler';
|
2016-09-09 01:19:18 +00:00
|
|
|
import UIUtil from './modules/UI/util/UIUtil';
|
2020-05-20 10:57:03 +00:00
|
|
|
import mediaDeviceHelper from './modules/devices/mediaDeviceHelper';
|
|
|
|
import Recorder from './modules/recorder/Recorder';
|
2018-08-31 20:02:04 +00:00
|
|
|
import { createTaskQueue } from './modules/util/helpers';
|
2017-12-11 18:48:32 +00:00
|
|
|
import {
|
2018-01-03 21:24:07 +00:00
|
|
|
createDeviceChangedEvent,
|
2019-06-14 11:16:08 +00:00
|
|
|
createStartSilentEvent,
|
2018-01-03 21:24:07 +00:00
|
|
|
createScreenSharingEvent,
|
|
|
|
createTrackMutedEvent,
|
|
|
|
sendAnalytics
|
2017-12-11 18:48:32 +00:00
|
|
|
} from './react/features/analytics';
|
2018-02-26 22:50:27 +00:00
|
|
|
import {
|
2019-07-01 12:02:25 +00:00
|
|
|
maybeRedirectToWelcomePage,
|
|
|
|
redirectToStaticPage,
|
2018-02-26 22:50:27 +00:00
|
|
|
reloadWithStoredParams
|
2020-06-04 14:09:13 +00:00
|
|
|
} from './react/features/app/actions';
|
2021-09-15 08:28:44 +00:00
|
|
|
import { showModeratedNotification } from './react/features/av-moderation/actions';
|
|
|
|
import { shouldShowModeratedNotification } from './react/features/av-moderation/functions';
|
2017-02-27 21:42:28 +00:00
|
|
|
import {
|
2017-02-28 23:12:02 +00:00
|
|
|
AVATAR_URL_COMMAND,
|
2018-08-16 15:11:06 +00:00
|
|
|
EMAIL_COMMAND,
|
2018-06-20 20:19:53 +00:00
|
|
|
authStatusChanged,
|
2018-08-16 15:11:06 +00:00
|
|
|
commonUserJoinedHandling,
|
|
|
|
commonUserLeftHandling,
|
2017-02-27 21:42:28 +00:00
|
|
|
conferenceFailed,
|
2017-02-28 23:12:02 +00:00
|
|
|
conferenceJoined,
|
|
|
|
conferenceLeft,
|
2019-03-12 17:45:53 +00:00
|
|
|
conferenceSubjectChanged,
|
2020-01-13 17:12:25 +00:00
|
|
|
conferenceTimestampChanged,
|
2021-02-03 10:28:39 +00:00
|
|
|
conferenceUniqueIdSet,
|
2017-11-21 22:45:14 +00:00
|
|
|
conferenceWillJoin,
|
2018-08-01 16:41:54 +00:00
|
|
|
conferenceWillLeave,
|
2017-08-09 19:40:03 +00:00
|
|
|
dataChannelOpened,
|
2021-08-04 10:56:07 +00:00
|
|
|
getConferenceOptions,
|
2019-06-26 21:53:48 +00:00
|
|
|
kickedOut,
|
2017-08-09 19:40:03 +00:00
|
|
|
lockStateChanged,
|
2017-11-21 02:21:35 +00:00
|
|
|
onStartMutedPolicyChanged,
|
2017-10-06 17:52:23 +00:00
|
|
|
p2pStatusChanged,
|
2021-09-14 21:57:05 +00:00
|
|
|
sendLocalParticipant,
|
|
|
|
_conferenceWillJoin
|
2017-02-27 21:42:28 +00:00
|
|
|
} from './react/features/base/conference';
|
2021-06-16 11:08:18 +00:00
|
|
|
import { getReplaceParticipant } from './react/features/base/config/functions';
|
2018-04-12 19:58:20 +00:00
|
|
|
import {
|
2019-05-03 17:25:33 +00:00
|
|
|
checkAndNotifyForNewDevice,
|
2018-08-06 15:24:59 +00:00
|
|
|
getAvailableDevices,
|
2020-06-03 21:49:08 +00:00
|
|
|
getDefaultDeviceId,
|
2019-05-29 21:17:07 +00:00
|
|
|
notifyCameraError,
|
|
|
|
notifyMicError,
|
2018-04-12 19:58:20 +00:00
|
|
|
setAudioOutputDeviceId,
|
|
|
|
updateDeviceList
|
|
|
|
} from './react/features/base/devices';
|
2021-08-31 15:06:31 +00:00
|
|
|
import { isIosMobileBrowser } from './react/features/base/environment/utils';
|
2017-02-19 00:42:11 +00:00
|
|
|
import {
|
2020-11-06 20:52:41 +00:00
|
|
|
browser,
|
2017-10-10 23:31:40 +00:00
|
|
|
isFatalJitsiConnectionError,
|
|
|
|
JitsiConferenceErrors,
|
|
|
|
JitsiConferenceEvents,
|
|
|
|
JitsiConnectionErrors,
|
|
|
|
JitsiConnectionEvents,
|
|
|
|
JitsiMediaDevicesEvents,
|
|
|
|
JitsiParticipantConnectionStatus,
|
|
|
|
JitsiTrackErrors,
|
|
|
|
JitsiTrackEvents
|
2017-02-19 00:42:11 +00:00
|
|
|
} from './react/features/base/lib-jitsi-meet';
|
2017-07-24 13:56:57 +00:00
|
|
|
import {
|
2020-10-21 16:57:50 +00:00
|
|
|
getStartWithAudioMuted,
|
|
|
|
getStartWithVideoMuted,
|
2017-08-18 11:30:30 +00:00
|
|
|
isVideoMutedByUser,
|
|
|
|
MEDIA_TYPE,
|
2017-07-24 13:56:57 +00:00
|
|
|
setAudioAvailable,
|
2017-08-18 11:30:30 +00:00
|
|
|
setAudioMuted,
|
|
|
|
setVideoAvailable,
|
|
|
|
setVideoMuted
|
2017-07-24 13:56:57 +00:00
|
|
|
} from './react/features/base/media';
|
2017-02-27 21:42:28 +00:00
|
|
|
import {
|
2017-09-18 18:35:52 +00:00
|
|
|
dominantSpeakerChanged,
|
2017-10-06 17:52:23 +00:00
|
|
|
getLocalParticipant,
|
2019-01-15 11:28:07 +00:00
|
|
|
getNormalizedDisplayName,
|
2017-10-06 17:52:23 +00:00
|
|
|
getParticipantById,
|
2017-07-13 00:26:10 +00:00
|
|
|
localParticipantConnectionStatusChanged,
|
2017-04-10 21:53:30 +00:00
|
|
|
localParticipantRoleChanged,
|
2017-07-13 00:26:10 +00:00
|
|
|
participantConnectionStatusChanged,
|
2019-06-17 14:00:09 +00:00
|
|
|
participantKicked,
|
|
|
|
participantMutedUs,
|
2017-07-31 23:33:22 +00:00
|
|
|
participantPresenceChanged,
|
2017-03-23 18:01:33 +00:00
|
|
|
participantRoleChanged,
|
2020-11-14 04:09:25 +00:00
|
|
|
participantUpdated,
|
|
|
|
updateRemoteParticipantFeatures
|
2017-02-27 21:42:28 +00:00
|
|
|
} from './react/features/base/participants';
|
2017-06-20 20:09:34 +00:00
|
|
|
import {
|
2019-11-26 10:57:03 +00:00
|
|
|
getUserSelectedCameraDeviceId,
|
|
|
|
updateSettings
|
|
|
|
} from './react/features/base/settings';
|
|
|
|
import {
|
|
|
|
createLocalPresenterTrack,
|
2017-08-14 13:25:37 +00:00
|
|
|
createLocalTracksF,
|
2019-01-01 21:19:34 +00:00
|
|
|
destroyLocalTracks,
|
2020-06-26 08:54:12 +00:00
|
|
|
getLocalJitsiAudioTrack,
|
|
|
|
getLocalJitsiVideoTrack,
|
2021-03-05 15:18:34 +00:00
|
|
|
getLocalTracks,
|
2020-10-26 18:05:22 +00:00
|
|
|
isLocalCameraTrackMuted,
|
2017-08-18 11:30:30 +00:00
|
|
|
isLocalTrackMuted,
|
2019-07-10 11:02:27 +00:00
|
|
|
isUserInteractionRequiredForUnmute,
|
2017-06-20 20:09:34 +00:00
|
|
|
replaceLocalTrack,
|
|
|
|
trackAdded,
|
|
|
|
trackRemoved
|
|
|
|
} from './react/features/base/tracks';
|
2020-10-02 13:20:24 +00:00
|
|
|
import { downloadJSON } from './react/features/base/util/downloadJSON';
|
2017-10-12 23:02:29 +00:00
|
|
|
import { showDesktopPicker } from './react/features/desktop-picker';
|
2017-12-05 03:27:17 +00:00
|
|
|
import { appendSuffix } from './react/features/display-name';
|
2018-01-19 22:19:55 +00:00
|
|
|
import {
|
|
|
|
maybeOpenFeedbackDialog,
|
|
|
|
submitFeedback
|
|
|
|
} from './react/features/feedback';
|
2021-09-15 08:28:44 +00:00
|
|
|
import { isModerationNotificationDisplayed, showNotification } from './react/features/notifications';
|
2020-11-18 12:38:00 +00:00
|
|
|
import { mediaPermissionPromptVisibilityChanged, toggleSlowGUMOverlay } from './react/features/overlay';
|
2019-07-12 13:08:34 +00:00
|
|
|
import { suspendDetected } from './react/features/power-monitor';
|
2020-05-20 10:57:03 +00:00
|
|
|
import {
|
|
|
|
initPrejoin,
|
2020-06-26 08:54:12 +00:00
|
|
|
isPrejoinPageEnabled,
|
2020-07-29 10:27:32 +00:00
|
|
|
isPrejoinPageVisible,
|
2021-09-28 22:50:57 +00:00
|
|
|
makePrecallTest,
|
|
|
|
setJoiningInProgress
|
2020-05-20 10:57:03 +00:00
|
|
|
} from './react/features/prejoin';
|
2020-11-14 04:09:25 +00:00
|
|
|
import { disableReceiver, stopReceiver } from './react/features/remote-control';
|
2021-04-12 07:37:39 +00:00
|
|
|
import { setScreenAudioShareState, isScreenAudioShared } from './react/features/screen-share/';
|
2021-07-26 11:38:56 +00:00
|
|
|
import { toggleScreenshotCaptureSummary } from './react/features/screenshot-capture';
|
2020-03-26 12:17:44 +00:00
|
|
|
import { AudioMixerEffect } from './react/features/stream-effects/audio-mixer/AudioMixerEffect';
|
2019-11-26 10:57:03 +00:00
|
|
|
import { createPresenterEffect } from './react/features/stream-effects/presenter';
|
2021-08-04 10:56:07 +00:00
|
|
|
import { createRnnoiseProcessor } from './react/features/stream-effects/rnnoise';
|
2018-07-17 17:31:12 +00:00
|
|
|
import { endpointMessageReceived } from './react/features/subtitles';
|
2020-05-20 10:57:03 +00:00
|
|
|
import UIEvents from './service/UI/UIEvents';
|
2017-02-27 21:42:28 +00:00
|
|
|
|
2020-05-20 10:57:03 +00:00
|
|
|
const logger = Logger.getLogger(__filename);
|
2017-10-16 20:37:13 +00:00
|
|
|
|
2017-01-23 18:07:08 +00:00
|
|
|
const eventEmitter = new EventEmitter();
|
|
|
|
|
2017-04-11 19:40:03 +00:00
|
|
|
let room;
|
|
|
|
let connection;
|
2016-07-22 18:42:41 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
/**
|
|
|
|
* The promise is used when the prejoin screen is shown.
|
|
|
|
* While the user configures the devices the connection can be made.
|
|
|
|
*
|
|
|
|
* @type {Promise<Object>}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
let _connectionPromise;
|
|
|
|
|
2021-09-23 19:54:27 +00:00
|
|
|
/**
|
|
|
|
* We are storing the resolve function of a Promise that waits for the _connectionPromise to be created. This is needed
|
|
|
|
* when the prejoin button was pressed before the conference object was initialized and the _connectionPromise has not
|
|
|
|
* been initialized when we tried to execute prejoinStart. In this case in prejoinStart we create a new Promise, assign
|
|
|
|
* the resolve function to this variable and wait for the promise to resolve before we continue. The
|
|
|
|
* _onConnectionPromiseCreated will be called once the _connectionPromise is created.
|
|
|
|
*/
|
|
|
|
let _onConnectionPromiseCreated;
|
|
|
|
|
2020-01-10 21:39:58 +00:00
|
|
|
/**
|
|
|
|
* This promise is used for chaining mutePresenterVideo calls in order to avoid calling GUM multiple times if it takes
|
|
|
|
* a while to finish.
|
|
|
|
*
|
|
|
|
* @type {Promise<void>}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
let _prevMutePresenterVideo = Promise.resolve();
|
|
|
|
|
2017-03-30 16:58:31 +00:00
|
|
|
/*
|
|
|
|
* Logic to open a desktop picker put on the window global for
|
|
|
|
* lib-jitsi-meet to detect and invoke
|
|
|
|
*/
|
|
|
|
window.JitsiMeetScreenObtainer = {
|
2017-07-09 21:34:08 +00:00
|
|
|
openDesktopPicker(options, onSourceChoose) {
|
|
|
|
APP.store.dispatch(showDesktopPicker(options, onSourceChoose));
|
2017-03-30 16:58:31 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-06-13 21:11:44 +00:00
|
|
|
/**
|
|
|
|
* Known custom conference commands.
|
|
|
|
*/
|
|
|
|
const commands = {
|
2017-02-28 23:12:02 +00:00
|
|
|
AVATAR_URL: AVATAR_URL_COMMAND,
|
2017-10-12 23:02:29 +00:00
|
|
|
CUSTOM_ROLE: 'custom-role',
|
2017-02-28 23:12:02 +00:00
|
|
|
EMAIL: EMAIL_COMMAND,
|
2021-04-16 09:43:34 +00:00
|
|
|
ETHERPAD: 'etherpad'
|
2016-06-13 21:11:44 +00:00
|
|
|
};
|
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
|
|
|
* Open Connection. When authentication failed it shows auth dialog.
|
2016-02-09 23:44:41 +00:00
|
|
|
* @param roomName the room name to use
|
2016-01-15 14:59:35 +00:00
|
|
|
* @returns Promise<JitsiConnection>
|
|
|
|
*/
|
2016-02-09 23:44:41 +00:00
|
|
|
function connect(roomName) {
|
2017-10-12 23:02:29 +00:00
|
|
|
return openConnection({
|
|
|
|
retry: true,
|
|
|
|
roomName
|
|
|
|
})
|
|
|
|
.catch(err => {
|
|
|
|
if (err === JitsiConnectionErrors.PASSWORD_REQUIRED) {
|
|
|
|
APP.UI.notifyTokenAuthFailed();
|
|
|
|
} else {
|
|
|
|
APP.UI.notifyConnectionFailed(err);
|
|
|
|
}
|
|
|
|
throw err;
|
|
|
|
});
|
2016-01-06 22:39:13 +00:00
|
|
|
}
|
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
2016-06-13 21:11:44 +00:00
|
|
|
* Share data to other users.
|
|
|
|
* @param command the command
|
|
|
|
* @param {string} value new value
|
2016-01-15 14:59:35 +00:00
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
function sendData(command, value) {
|
2017-08-18 11:30:30 +00:00
|
|
|
if (!room) {
|
2017-03-23 18:01:33 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2016-06-13 21:11:44 +00:00
|
|
|
room.removeCommand(command);
|
2017-10-12 23:02:29 +00:00
|
|
|
room.sendCommand(command, { value });
|
2016-01-15 14:59:35 +00:00
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
|
|
|
* Get user nickname by user id.
|
|
|
|
* @param {string} id user id
|
|
|
|
* @returns {string?} user nickname or undefined if user is unknown.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
function getDisplayName(id) {
|
2017-10-06 17:52:23 +00:00
|
|
|
const participant = getParticipantById(APP.store.getState(), id);
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-10-06 17:52:23 +00:00
|
|
|
return participant && participant.name;
|
2016-01-15 14:59:35 +00:00
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2016-02-03 14:44:10 +00:00
|
|
|
/**
|
|
|
|
* Mute or unmute local audio stream if it exists.
|
2016-10-03 16:12:04 +00:00
|
|
|
* @param {boolean} muted - if audio stream should be muted or unmuted.
|
2016-02-03 14:44:10 +00:00
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
function muteLocalAudio(muted) {
|
2017-08-18 11:30:30 +00:00
|
|
|
APP.store.dispatch(setAudioMuted(muted));
|
2016-02-03 14:44:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Mute or unmute local video stream if it exists.
|
|
|
|
* @param {boolean} muted if video stream should be muted or unmuted.
|
2017-07-24 09:20:32 +00:00
|
|
|
*
|
2016-02-03 14:44:10 +00:00
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
function muteLocalVideo(muted) {
|
2017-08-18 11:30:30 +00:00
|
|
|
APP.store.dispatch(setVideoMuted(muted));
|
2016-02-03 14:44:10 +00:00
|
|
|
}
|
|
|
|
|
2018-08-31 20:02:04 +00:00
|
|
|
/**
|
|
|
|
* A queue for the async replaceLocalTrack action so that multiple audio
|
|
|
|
* replacements cannot happen simultaneously. This solves the issue where
|
|
|
|
* replaceLocalTrack is called multiple times with an oldTrack of null, causing
|
|
|
|
* multiple local tracks of the same type to be used.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
const _replaceLocalAudioTrackQueue = createTaskQueue();
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A task queue for replacement local video tracks. This separate queue exists
|
|
|
|
* so video replacement is not blocked by audio replacement tasks in the queue
|
|
|
|
* {@link _replaceLocalAudioTrackQueue}.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
const _replaceLocalVideoTrackQueue = createTaskQueue();
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2016-01-06 22:39:13 +00:00
|
|
|
class ConferenceConnector {
|
2017-10-12 23:02:29 +00:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2017-04-13 00:23:43 +00:00
|
|
|
constructor(resolve, reject) {
|
2016-01-06 22:39:13 +00:00
|
|
|
this._resolve = resolve;
|
|
|
|
this._reject = reject;
|
|
|
|
this.reconnectTimeout = null;
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.CONFERENCE_JOINED,
|
2016-01-06 22:39:13 +00:00
|
|
|
this._handleConferenceJoined.bind(this));
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.CONFERENCE_FAILED,
|
2016-01-06 22:39:13 +00:00
|
|
|
this._onConferenceFailed.bind(this));
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2016-10-03 16:12:04 +00:00
|
|
|
_handleConferenceFailed(err) {
|
2016-01-06 22:39:13 +00:00
|
|
|
this._unsubscribe();
|
|
|
|
this._reject(err);
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2016-01-25 22:39:05 +00:00
|
|
|
_onConferenceFailed(err, ...params) {
|
2017-01-31 20:58:48 +00:00
|
|
|
APP.store.dispatch(conferenceFailed(room, err, ...params));
|
2016-11-11 15:00:54 +00:00
|
|
|
logger.error('CONFERENCE FAILED:', err, ...params);
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-06-05 18:19:25 +00:00
|
|
|
switch (err) {
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.NOT_ALLOWED_ERROR: {
|
2017-10-02 23:08:07 +00:00
|
|
|
// let's show some auth not allowed page
|
2019-07-01 12:02:25 +00:00
|
|
|
APP.store.dispatch(redirectToStaticPage('static/authError.html'));
|
2016-09-27 22:26:38 +00:00
|
|
|
break;
|
2017-10-02 23:08:07 +00:00
|
|
|
}
|
2016-09-27 22:26:38 +00:00
|
|
|
|
2017-10-02 23:08:07 +00:00
|
|
|
// not enough rights to create conference
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.AUTHENTICATION_REQUIRED: {
|
2021-06-11 08:58:45 +00:00
|
|
|
|
2021-06-16 11:08:18 +00:00
|
|
|
const replaceParticipant = getReplaceParticipant(APP.store.getState());
|
2021-06-11 08:58:45 +00:00
|
|
|
|
2017-10-02 23:08:07 +00:00
|
|
|
// Schedule reconnect to check if someone else created the room.
|
2019-01-19 23:53:05 +00:00
|
|
|
this.reconnectTimeout = setTimeout(() => {
|
|
|
|
APP.store.dispatch(conferenceWillJoin(room));
|
2021-06-11 08:58:45 +00:00
|
|
|
room.join(null, replaceParticipant);
|
2019-01-19 23:53:05 +00:00
|
|
|
}, 5000);
|
2017-04-13 00:23:43 +00:00
|
|
|
|
2021-04-22 15:05:14 +00:00
|
|
|
const { password }
|
|
|
|
= APP.store.getState()['features/base/conference'];
|
|
|
|
|
|
|
|
AuthHandler.requireAuth(room, password);
|
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
break;
|
2017-10-02 23:08:07 +00:00
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.RESERVATION_ERROR: {
|
2017-10-12 23:02:29 +00:00
|
|
|
const [ code, msg ] = params;
|
|
|
|
|
2017-10-02 23:08:07 +00:00
|
|
|
APP.UI.notifyReservationError(code, msg);
|
2016-01-25 22:39:05 +00:00
|
|
|
break;
|
2017-10-02 23:08:07 +00:00
|
|
|
}
|
2016-01-25 22:39:05 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.GRACEFUL_SHUTDOWN:
|
2016-02-05 15:04:48 +00:00
|
|
|
APP.UI.notifyGracefulShutdown();
|
2016-01-25 22:39:05 +00:00
|
|
|
break;
|
|
|
|
|
2017-10-02 23:08:07 +00:00
|
|
|
// FIXME FOCUS_DISCONNECTED is a confusing event name.
|
|
|
|
// What really happens there is that the library is not ready yet,
|
|
|
|
// because Jicofo is not available, but it is going to give it another
|
|
|
|
// try.
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.FOCUS_DISCONNECTED: {
|
2017-10-12 23:02:29 +00:00
|
|
|
const [ focus, retrySec ] = params;
|
|
|
|
|
2017-10-02 23:08:07 +00:00
|
|
|
APP.UI.notifyFocusDisconnected(focus, retrySec);
|
2016-01-25 22:39:05 +00:00
|
|
|
break;
|
2017-10-02 23:08:07 +00:00
|
|
|
}
|
2016-01-25 22:39:05 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.FOCUS_LEFT:
|
2020-05-07 11:59:37 +00:00
|
|
|
case JitsiConferenceErrors.ICE_FAILED:
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.VIDEOBRIDGE_NOT_AVAILABLE:
|
2019-06-17 10:35:47 +00:00
|
|
|
case JitsiConferenceErrors.OFFER_ANSWER_FAILED:
|
2018-08-01 20:37:15 +00:00
|
|
|
APP.store.dispatch(conferenceWillLeave(room));
|
|
|
|
|
2016-10-06 18:30:00 +00:00
|
|
|
// FIXME the conference should be stopped by the library and not by
|
|
|
|
// the app. Both the errors above are unrecoverable from the library
|
|
|
|
// perspective.
|
2016-02-22 14:57:36 +00:00
|
|
|
room.leave().then(() => connection.disconnect());
|
2016-02-05 15:04:48 +00:00
|
|
|
break;
|
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.CONFERENCE_MAX_USERS:
|
2016-03-15 19:08:01 +00:00
|
|
|
connection.disconnect();
|
|
|
|
APP.UI.notifyMaxUsersLimitReached();
|
|
|
|
break;
|
2017-10-02 23:08:07 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
case JitsiConferenceErrors.INCOMPATIBLE_SERVER_VERSIONS:
|
2018-02-26 22:50:27 +00:00
|
|
|
APP.store.dispatch(reloadWithStoredParams());
|
2016-07-08 01:44:04 +00:00
|
|
|
break;
|
2017-10-02 23:08:07 +00:00
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
default:
|
2016-01-25 22:39:05 +00:00
|
|
|
this._handleConferenceFailed(err, ...params);
|
|
|
|
}
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2016-01-06 22:39:13 +00:00
|
|
|
_unsubscribe() {
|
|
|
|
room.off(
|
2017-10-12 23:02:29 +00:00
|
|
|
JitsiConferenceEvents.CONFERENCE_JOINED,
|
|
|
|
this._handleConferenceJoined);
|
2016-01-06 22:39:13 +00:00
|
|
|
room.off(
|
2017-10-12 23:02:29 +00:00
|
|
|
JitsiConferenceEvents.CONFERENCE_FAILED,
|
|
|
|
this._onConferenceFailed);
|
2016-01-06 22:39:13 +00:00
|
|
|
if (this.reconnectTimeout !== null) {
|
|
|
|
clearTimeout(this.reconnectTimeout);
|
|
|
|
}
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2016-01-06 22:39:13 +00:00
|
|
|
_handleConferenceJoined() {
|
|
|
|
this._unsubscribe();
|
|
|
|
this._resolve();
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2016-01-06 22:39:13 +00:00
|
|
|
connect() {
|
2021-06-16 11:08:18 +00:00
|
|
|
const replaceParticipant = getReplaceParticipant(APP.store.getState());
|
2021-06-11 08:58:45 +00:00
|
|
|
|
2021-04-07 15:03:20 +00:00
|
|
|
// the local storage overrides here and in connection.js can be used by jibri
|
2021-06-11 08:58:45 +00:00
|
|
|
room.join(jitsiLocalStorage.getItem('xmpp_conference_password_override'), replaceParticipant);
|
2016-01-06 22:39:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-10-12 21:30:44 +00:00
|
|
|
/**
|
|
|
|
* Disconnects the connection.
|
|
|
|
* @returns resolved Promise. We need this in order to make the Promise.all
|
|
|
|
* call in hangup() to resolve when all operations are finished.
|
|
|
|
*/
|
|
|
|
function disconnect() {
|
2019-05-30 13:24:58 +00:00
|
|
|
const onDisconnected = () => {
|
|
|
|
APP.API.notifyConferenceLeft(APP.conference.roomName);
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2019-05-30 13:24:58 +00:00
|
|
|
return Promise.resolve();
|
|
|
|
};
|
|
|
|
|
2020-07-15 12:18:03 +00:00
|
|
|
if (!connection) {
|
|
|
|
return onDisconnected();
|
|
|
|
}
|
|
|
|
|
2019-05-30 13:24:58 +00:00
|
|
|
return connection.disconnect().then(onDisconnected, onDisconnected);
|
2016-10-12 21:30:44 +00:00
|
|
|
}
|
|
|
|
|
2017-01-31 20:58:48 +00:00
|
|
|
/**
|
|
|
|
* Handles CONNECTION_FAILED events from lib-jitsi-meet.
|
2017-02-19 00:42:11 +00:00
|
|
|
*
|
2017-10-10 23:31:40 +00:00
|
|
|
* @param {JitsiConnectionError} error - The reported error.
|
2017-01-31 20:58:48 +00:00
|
|
|
* @returns {void}
|
|
|
|
* @private
|
|
|
|
*/
|
2017-02-19 00:42:11 +00:00
|
|
|
function _connectionFailedHandler(error) {
|
|
|
|
if (isFatalJitsiConnectionError(error)) {
|
|
|
|
APP.connection.removeEventListener(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConnectionEvents.CONNECTION_FAILED,
|
2017-01-31 20:58:48 +00:00
|
|
|
_connectionFailedHandler);
|
2017-10-12 23:02:29 +00:00
|
|
|
if (room) {
|
2018-08-01 20:37:15 +00:00
|
|
|
APP.store.dispatch(conferenceWillLeave(room));
|
2017-01-31 20:58:48 +00:00
|
|
|
room.leave();
|
2017-10-12 23:02:29 +00:00
|
|
|
}
|
2017-01-31 20:58:48 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
export default {
|
2017-07-21 09:12:33 +00:00
|
|
|
/**
|
|
|
|
* Flag used to delay modification of the muted status of local media tracks
|
|
|
|
* until those are created (or not, but at that point it's certain that
|
|
|
|
* the tracks won't exist).
|
|
|
|
*/
|
|
|
|
_localTracksInitialized: false,
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2020-11-03 09:44:41 +00:00
|
|
|
isSharingScreen: false,
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2019-11-26 10:57:03 +00:00
|
|
|
/**
|
|
|
|
* The local presenter video track (if any).
|
2020-01-10 21:39:58 +00:00
|
|
|
* @type {JitsiLocalTrack|null}
|
2019-11-26 10:57:03 +00:00
|
|
|
*/
|
|
|
|
localPresenterVideo: null,
|
|
|
|
|
2017-07-17 11:38:46 +00:00
|
|
|
/**
|
2020-04-16 10:47:10 +00:00
|
|
|
* Returns an object containing a promise which resolves with the created tracks &
|
|
|
|
* the errors resulting from that process.
|
|
|
|
*
|
|
|
|
* @returns {Promise<JitsiLocalTrack[]>, Object}
|
2017-07-17 11:38:46 +00:00
|
|
|
*/
|
2020-04-16 10:47:10 +00:00
|
|
|
createInitialLocalTracks(options = {}) {
|
|
|
|
const errors = {};
|
2021-03-05 15:18:34 +00:00
|
|
|
|
|
|
|
// Always get a handle on the audio input device so that we have statistics (such as "No audio input" or
|
|
|
|
// "Are you trying to speak?" ) even if the user joins the conference muted.
|
2021-01-11 11:16:49 +00:00
|
|
|
const initialDevices = config.disableInitialGUM ? [] : [ 'audio' ];
|
|
|
|
const requestedAudio = !config.disableInitialGUM;
|
2017-07-24 16:10:31 +00:00
|
|
|
let requestedVideo = false;
|
|
|
|
|
2021-01-11 11:16:49 +00:00
|
|
|
if (!config.disableInitialGUM
|
|
|
|
&& !options.startWithVideoMuted
|
2017-07-24 16:10:31 +00:00
|
|
|
&& !options.startAudioOnly
|
|
|
|
&& !options.startScreenSharing) {
|
|
|
|
initialDevices.push('video');
|
|
|
|
requestedVideo = true;
|
|
|
|
}
|
2017-07-17 11:38:46 +00:00
|
|
|
|
2021-01-11 11:16:49 +00:00
|
|
|
if (!config.disableInitialGUM) {
|
|
|
|
JitsiMeetJS.mediaDevices.addEventListener(
|
|
|
|
JitsiMediaDevicesEvents.PERMISSION_PROMPT_IS_SHOWN,
|
|
|
|
browserName =>
|
|
|
|
APP.store.dispatch(
|
|
|
|
mediaPermissionPromptVisibilityChanged(true, browserName))
|
|
|
|
);
|
|
|
|
}
|
2017-07-17 11:38:46 +00:00
|
|
|
|
2020-11-18 12:38:00 +00:00
|
|
|
JitsiMeetJS.mediaDevices.addEventListener(
|
|
|
|
JitsiMediaDevicesEvents.SLOW_GET_USER_MEDIA,
|
|
|
|
() => APP.store.dispatch(toggleSlowGUMOverlay(true))
|
|
|
|
);
|
|
|
|
|
2017-06-29 17:43:35 +00:00
|
|
|
let tryCreateLocalTracks;
|
|
|
|
|
2021-02-08 21:10:24 +00:00
|
|
|
// On Electron there is no permission prompt for granting permissions. That's why we don't need to
|
2021-03-16 15:59:33 +00:00
|
|
|
// spend much time displaying the overlay screen. If GUM is not resolved within 15 seconds it will
|
2021-02-08 21:10:24 +00:00
|
|
|
// probably never resolve.
|
2021-02-02 00:20:39 +00:00
|
|
|
const timeout = browser.isElectron() ? 15000 : 60000;
|
|
|
|
|
2017-07-24 16:10:31 +00:00
|
|
|
// FIXME is there any simpler way to rewrite this spaghetti below ?
|
|
|
|
if (options.startScreenSharing) {
|
2017-06-29 17:43:35 +00:00
|
|
|
tryCreateLocalTracks = this._createDesktopTrack()
|
2020-04-16 10:47:10 +00:00
|
|
|
.then(([ desktopStream ]) => {
|
2017-07-24 16:10:31 +00:00
|
|
|
if (!requestedAudio) {
|
2017-10-12 23:02:29 +00:00
|
|
|
return [ desktopStream ];
|
2017-07-24 16:10:31 +00:00
|
|
|
}
|
|
|
|
|
2021-02-02 00:20:39 +00:00
|
|
|
return createLocalTracksF({
|
|
|
|
devices: [ 'audio' ],
|
2020-11-18 12:38:00 +00:00
|
|
|
timeout,
|
|
|
|
firePermissionPromptIsShownEvent: true,
|
|
|
|
fireSlowPromiseEvent: true
|
|
|
|
})
|
2017-10-12 23:02:29 +00:00
|
|
|
.then(([ audioStream ]) =>
|
|
|
|
[ desktopStream, audioStream ])
|
2017-06-29 17:43:35 +00:00
|
|
|
.catch(error => {
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.audioOnlyError = error;
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
return [ desktopStream ];
|
2017-06-29 17:43:35 +00:00
|
|
|
});
|
2017-10-12 23:02:29 +00:00
|
|
|
})
|
|
|
|
.catch(error => {
|
2017-06-29 17:43:35 +00:00
|
|
|
logger.error('Failed to obtain desktop stream', error);
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.screenSharingError = error;
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-07-24 16:10:31 +00:00
|
|
|
return requestedAudio
|
2021-02-02 00:20:39 +00:00
|
|
|
? createLocalTracksF({
|
|
|
|
devices: [ 'audio' ],
|
2020-11-18 12:38:00 +00:00
|
|
|
timeout,
|
|
|
|
firePermissionPromptIsShownEvent: true,
|
|
|
|
fireSlowPromiseEvent: true
|
|
|
|
})
|
2017-07-24 16:10:31 +00:00
|
|
|
: [];
|
2017-10-12 23:02:29 +00:00
|
|
|
})
|
|
|
|
.catch(error => {
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.audioOnlyError = error;
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-06-29 17:43:35 +00:00
|
|
|
return [];
|
|
|
|
});
|
2017-07-24 16:10:31 +00:00
|
|
|
} else if (!requestedAudio && !requestedVideo) {
|
|
|
|
// Resolve with no tracks
|
|
|
|
tryCreateLocalTracks = Promise.resolve([]);
|
2017-06-29 17:43:35 +00:00
|
|
|
} else {
|
2021-02-02 00:20:39 +00:00
|
|
|
tryCreateLocalTracks = createLocalTracksF({
|
|
|
|
devices: initialDevices,
|
2020-11-18 12:38:00 +00:00
|
|
|
timeout,
|
|
|
|
firePermissionPromptIsShownEvent: true,
|
|
|
|
fireSlowPromiseEvent: true
|
|
|
|
})
|
2019-07-08 10:59:23 +00:00
|
|
|
.catch(err => {
|
|
|
|
if (requestedAudio && requestedVideo) {
|
|
|
|
|
|
|
|
// Try audio only...
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.audioAndVideoError = err;
|
2019-07-08 10:59:23 +00:00
|
|
|
|
2021-02-02 00:20:39 +00:00
|
|
|
if (err.name === JitsiTrackErrors.TIMEOUT && !browser.isElectron()) {
|
|
|
|
// In this case we expect that the permission prompt is still visible. There is no point of
|
2021-03-16 15:59:33 +00:00
|
|
|
// executing GUM with different source. Also at the time of writing the following
|
2021-02-02 00:20:39 +00:00
|
|
|
// inconsistency have been noticed in some browsers - if the permissions prompt is visible
|
|
|
|
// and another GUM is executed the prompt does not change its content but if the user
|
|
|
|
// clicks allow the user action isassociated with the latest GUM call.
|
|
|
|
errors.audioOnlyError = err;
|
|
|
|
errors.videoOnlyError = err;
|
|
|
|
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
2019-07-08 10:59:23 +00:00
|
|
|
return (
|
2021-02-02 00:20:39 +00:00
|
|
|
createLocalTracksF({
|
|
|
|
devices: [ 'audio' ],
|
2020-11-18 12:38:00 +00:00
|
|
|
timeout,
|
|
|
|
firePermissionPromptIsShownEvent: true,
|
|
|
|
fireSlowPromiseEvent: true
|
|
|
|
}));
|
2019-07-08 10:59:23 +00:00
|
|
|
} else if (requestedAudio && !requestedVideo) {
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.audioOnlyError = err;
|
2017-07-24 16:10:31 +00:00
|
|
|
|
2019-07-08 10:59:23 +00:00
|
|
|
return [];
|
|
|
|
} else if (requestedVideo && !requestedAudio) {
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.videoOnlyError = err;
|
2017-07-24 16:10:31 +00:00
|
|
|
|
|
|
|
return [];
|
2019-07-08 10:59:23 +00:00
|
|
|
}
|
|
|
|
logger.error('Should never happen');
|
|
|
|
})
|
|
|
|
.catch(err => {
|
|
|
|
// Log this just in case...
|
|
|
|
if (!requestedAudio) {
|
|
|
|
logger.error('The impossible just happened', err);
|
|
|
|
}
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.audioOnlyError = err;
|
2019-07-08 10:59:23 +00:00
|
|
|
|
|
|
|
// Try video only...
|
|
|
|
return requestedVideo
|
2021-02-02 00:20:39 +00:00
|
|
|
? createLocalTracksF({
|
|
|
|
devices: [ 'video' ],
|
2020-11-18 12:38:00 +00:00
|
|
|
firePermissionPromptIsShownEvent: true,
|
|
|
|
fireSlowPromiseEvent: true
|
|
|
|
})
|
2019-07-08 10:59:23 +00:00
|
|
|
: [];
|
|
|
|
})
|
|
|
|
.catch(err => {
|
|
|
|
// Log this just in case...
|
|
|
|
if (!requestedVideo) {
|
|
|
|
logger.error('The impossible just happened', err);
|
|
|
|
}
|
2020-04-16 10:47:10 +00:00
|
|
|
errors.videoOnlyError = err;
|
2019-07-08 10:59:23 +00:00
|
|
|
|
|
|
|
return [];
|
|
|
|
});
|
2017-06-29 17:43:35 +00:00
|
|
|
}
|
2017-07-17 11:38:46 +00:00
|
|
|
|
2017-11-30 04:47:24 +00:00
|
|
|
// Hide the permissions prompt/overlay as soon as the tracks are
|
|
|
|
// created. Don't wait for the connection to be made, since in some
|
2021-03-16 15:59:33 +00:00
|
|
|
// cases, when auth is required, for instance, that won't happen until
|
2017-11-30 04:47:24 +00:00
|
|
|
// the user inputs their credentials, but the dialog would be
|
|
|
|
// overshadowed by the overlay.
|
2020-04-16 10:47:10 +00:00
|
|
|
tryCreateLocalTracks.then(tracks => {
|
2020-11-18 12:38:00 +00:00
|
|
|
APP.store.dispatch(toggleSlowGUMOverlay(false));
|
2020-04-16 10:47:10 +00:00
|
|
|
APP.store.dispatch(mediaPermissionPromptVisibilityChanged(false));
|
|
|
|
|
|
|
|
return tracks;
|
|
|
|
});
|
|
|
|
|
|
|
|
return {
|
|
|
|
tryCreateLocalTracks,
|
|
|
|
errors
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
2020-12-09 17:40:23 +00:00
|
|
|
/**
|
|
|
|
* Displays error notifications according to the state carried by {@code errors} object returned
|
|
|
|
* by {@link createInitialLocalTracks}.
|
|
|
|
* @param {Object} errors - the errors (if any) returned by {@link createInitialLocalTracks}.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_displayErrorsForCreateInitialLocalTracks(errors) {
|
|
|
|
const {
|
|
|
|
audioAndVideoError,
|
|
|
|
audioOnlyError,
|
|
|
|
screenSharingError,
|
|
|
|
videoOnlyError
|
|
|
|
} = errors;
|
|
|
|
|
|
|
|
// FIXME If there will be microphone error it will cover any screensharing dialog, but it's still better than in
|
|
|
|
// the reverse order where the screensharing dialog will sometimes be closing the microphone alert
|
|
|
|
// ($.prompt.close(); is called). Need to figure out dialogs chaining to fix that.
|
|
|
|
if (screenSharingError) {
|
|
|
|
this._handleScreenSharingError(screenSharingError);
|
|
|
|
}
|
|
|
|
if (audioAndVideoError || audioOnlyError) {
|
|
|
|
if (audioOnlyError || videoOnlyError) {
|
|
|
|
// If both requests for 'audio' + 'video' and 'audio' only failed, we assume that there are some
|
|
|
|
// problems with user's microphone and show corresponding dialog.
|
|
|
|
APP.store.dispatch(notifyMicError(audioOnlyError));
|
|
|
|
APP.store.dispatch(notifyCameraError(videoOnlyError));
|
|
|
|
} else {
|
|
|
|
// If request for 'audio' + 'video' failed, but request for 'audio' only was OK, we assume that we had
|
|
|
|
// problems with camera and show corresponding dialog.
|
|
|
|
APP.store.dispatch(notifyCameraError(audioAndVideoError));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
/**
|
|
|
|
* Creates local media tracks and connects to a room. Will show error
|
|
|
|
* dialogs in case accessing the local microphone and/or camera failed. Will
|
|
|
|
* show guidance overlay for users on how to give access to camera and/or
|
|
|
|
* microphone.
|
|
|
|
* @param {string} roomName
|
|
|
|
* @param {object} options
|
|
|
|
* @param {boolean} options.startAudioOnly=false - if <tt>true</tt> then
|
|
|
|
* only audio track will be created and the audio only mode will be turned
|
|
|
|
* on.
|
|
|
|
* @param {boolean} options.startScreenSharing=false - if <tt>true</tt>
|
|
|
|
* should start with screensharing instead of camera video.
|
|
|
|
* @param {boolean} options.startWithAudioMuted - will start the conference
|
|
|
|
* without any audio tracks.
|
|
|
|
* @param {boolean} options.startWithVideoMuted - will start the conference
|
|
|
|
* without any video tracks.
|
|
|
|
* @returns {Promise.<JitsiLocalTrack[], JitsiConnection>}
|
|
|
|
*/
|
|
|
|
createInitialLocalTracksAndConnect(roomName, options = {}) {
|
|
|
|
const { tryCreateLocalTracks, errors } = this.createInitialLocalTracks(options);
|
2017-11-28 11:02:51 +00:00
|
|
|
|
2017-07-17 11:38:46 +00:00
|
|
|
return Promise.all([ tryCreateLocalTracks, connect(roomName) ])
|
2017-10-12 23:02:29 +00:00
|
|
|
.then(([ tracks, con ]) => {
|
2020-12-09 17:40:23 +00:00
|
|
|
|
|
|
|
this._displayErrorsForCreateInitialLocalTracks(errors);
|
2017-07-17 11:38:46 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
return [ tracks, con ];
|
2017-07-17 11:38:46 +00:00
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
startConference(con, tracks) {
|
|
|
|
tracks.forEach(track => {
|
|
|
|
if ((track.isAudioTrack() && this.isLocalAudioMuted())
|
|
|
|
|| (track.isVideoTrack() && this.isLocalVideoMuted())) {
|
|
|
|
const mediaType = track.getType();
|
|
|
|
|
|
|
|
sendAnalytics(
|
|
|
|
createTrackMutedEvent(mediaType, 'initial mute'));
|
|
|
|
logger.log(`${mediaType} mute: initially muted.`);
|
|
|
|
track.mute();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
con.addEventListener(JitsiConnectionEvents.CONNECTION_FAILED, _connectionFailedHandler);
|
|
|
|
APP.connection = connection = con;
|
|
|
|
|
|
|
|
this._createRoom(tracks);
|
|
|
|
|
|
|
|
// if user didn't give access to mic or camera or doesn't have
|
|
|
|
// them at all, we mark corresponding toolbar buttons as muted,
|
|
|
|
// so that the user can try unmute later on and add audio/video
|
|
|
|
// to the conference
|
|
|
|
if (!tracks.find(t => t.isAudioTrack())) {
|
|
|
|
this.setAudioMuteStatus(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!tracks.find(t => t.isVideoTrack())) {
|
2021-05-04 12:57:34 +00:00
|
|
|
this.setVideoMuteStatus();
|
2020-04-16 10:47:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (config.iAmRecorder) {
|
|
|
|
this.recorder = new Recorder();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (config.startSilent) {
|
|
|
|
sendAnalytics(createStartSilentEvent());
|
|
|
|
APP.store.dispatch(showNotification({
|
|
|
|
descriptionKey: 'notify.startSilentDescription',
|
|
|
|
titleKey: 'notify.startSilentTitle'
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
// XXX The API will take care of disconnecting from the XMPP
|
|
|
|
// server (and, thus, leaving the room) on unload.
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
(new ConferenceConnector(resolve, reject)).connect();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
2020-04-16 10:47:10 +00:00
|
|
|
* Open new connection and join the conference when prejoin page is not enabled.
|
|
|
|
* If prejoin page is enabled open an new connection in the background
|
|
|
|
* and create local tracks.
|
|
|
|
*
|
|
|
|
* @param {{ roomName: string }} options
|
2016-01-15 14:59:35 +00:00
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
2020-04-16 10:47:10 +00:00
|
|
|
async init({ roomName }) {
|
|
|
|
const initialOptions = {
|
|
|
|
startAudioOnly: config.startAudioOnly,
|
|
|
|
startScreenSharing: config.startScreenSharing,
|
2020-10-21 16:57:50 +00:00
|
|
|
startWithAudioMuted: getStartWithAudioMuted(APP.store.getState())
|
2020-04-16 10:47:10 +00:00
|
|
|
|| config.startSilent
|
|
|
|
|| isUserInteractionRequiredForUnmute(APP.store.getState()),
|
2020-10-21 16:57:50 +00:00
|
|
|
startWithVideoMuted: getStartWithVideoMuted(APP.store.getState())
|
2020-04-16 10:47:10 +00:00
|
|
|
|| isUserInteractionRequiredForUnmute(APP.store.getState())
|
|
|
|
};
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
this.roomName = roomName;
|
2020-04-07 12:14:47 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
try {
|
2019-05-07 08:53:01 +00:00
|
|
|
// Initialize the device list first. This way, when creating tracks
|
|
|
|
// based on preferred devices, loose label matching can be done in
|
|
|
|
// cases where the exact ID match is no longer available, such as
|
|
|
|
// when the camera device has switched USB ports.
|
2019-06-14 11:16:08 +00:00
|
|
|
// when in startSilent mode we want to start with audio muted
|
2020-04-16 10:47:10 +00:00
|
|
|
await this._initDeviceList();
|
|
|
|
} catch (error) {
|
|
|
|
logger.warn('initial device list initialization failed', error);
|
|
|
|
}
|
2017-02-16 23:02:40 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
if (isPrejoinPageEnabled(APP.store.getState())) {
|
2020-08-13 18:12:56 +00:00
|
|
|
_connectionPromise = connect(roomName).then(c => {
|
|
|
|
// we want to initialize it early, in case of errors to be able
|
|
|
|
// to gather logs
|
|
|
|
APP.connection = c;
|
|
|
|
|
|
|
|
return c;
|
|
|
|
});
|
2017-02-16 23:02:40 +00:00
|
|
|
|
2021-09-23 19:54:27 +00:00
|
|
|
if (_onConnectionPromiseCreated) {
|
|
|
|
_onConnectionPromiseCreated();
|
|
|
|
}
|
|
|
|
|
2020-07-29 10:27:32 +00:00
|
|
|
APP.store.dispatch(makePrecallTest(this._getConferenceOptions()));
|
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
const { tryCreateLocalTracks, errors } = this.createInitialLocalTracks(initialOptions);
|
|
|
|
const tracks = await tryCreateLocalTracks;
|
2016-06-23 08:03:26 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
// Initialize device list a second time to ensure device labels
|
|
|
|
// get populated in case of an initial gUM acceptance; otherwise
|
|
|
|
// they may remain as empty strings.
|
|
|
|
this._initDeviceList(true);
|
2016-06-21 09:08:32 +00:00
|
|
|
|
2020-12-09 17:40:23 +00:00
|
|
|
if (isPrejoinPageVisible(APP.store.getState())) {
|
|
|
|
return APP.store.dispatch(initPrejoin(tracks, errors));
|
|
|
|
}
|
|
|
|
|
|
|
|
logger.debug('Prejoin screen no longer displayed at the time when tracks were created');
|
|
|
|
|
|
|
|
this._displayErrorsForCreateInitialLocalTracks(errors);
|
|
|
|
|
|
|
|
return this._setLocalAudioVideoStreams(tracks);
|
2020-04-16 10:47:10 +00:00
|
|
|
}
|
2016-05-26 08:53:02 +00:00
|
|
|
|
2021-03-05 15:18:34 +00:00
|
|
|
const [ tracks, con ] = await this.createInitialLocalTracksAndConnect(roomName, initialOptions);
|
|
|
|
let localTracks = tracks;
|
2016-05-17 15:58:25 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
this._initDeviceList(true);
|
2019-06-05 17:01:18 +00:00
|
|
|
|
2021-03-05 15:18:34 +00:00
|
|
|
if (initialOptions.startWithAudioMuted) {
|
2021-08-31 15:06:31 +00:00
|
|
|
// Always add the audio track to the peer connection and then mute the track on mobile Safari
|
|
|
|
// because of a known issue where audio playout doesn't happen if the user joins audio and video muted.
|
|
|
|
if (isIosMobileBrowser()) {
|
|
|
|
this.muteAudio(true, true);
|
|
|
|
} else {
|
|
|
|
localTracks = localTracks.filter(track => track.getType() !== MEDIA_TYPE.AUDIO);
|
|
|
|
}
|
2021-03-05 15:18:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return this.startConference(con, localTracks);
|
2020-04-16 10:47:10 +00:00
|
|
|
},
|
2016-05-17 15:58:25 +00:00
|
|
|
|
2020-04-16 10:47:10 +00:00
|
|
|
/**
|
|
|
|
* Joins conference after the tracks have been configured in the prejoin screen.
|
|
|
|
*
|
|
|
|
* @param {Object[]} tracks - An array with the configured tracks
|
2021-09-23 19:54:27 +00:00
|
|
|
* @returns {void}
|
2020-04-16 10:47:10 +00:00
|
|
|
*/
|
|
|
|
async prejoinStart(tracks) {
|
2021-09-23 19:54:27 +00:00
|
|
|
if (!_connectionPromise) {
|
|
|
|
// The conference object isn't initialized yet. Wait for the promise to initialise.
|
|
|
|
await new Promise(resolve => {
|
|
|
|
_onConnectionPromiseCreated = resolve;
|
|
|
|
});
|
|
|
|
_onConnectionPromiseCreated = undefined;
|
|
|
|
}
|
|
|
|
|
2021-09-28 22:50:57 +00:00
|
|
|
let con;
|
2020-04-16 10:47:10 +00:00
|
|
|
|
2021-09-28 22:50:57 +00:00
|
|
|
try {
|
|
|
|
con = await _connectionPromise;
|
|
|
|
this.startConference(con, tracks);
|
|
|
|
} catch (error) {
|
|
|
|
logger.error(`An error occurred while trying to join a meeting from the prejoin screen: ${error}`);
|
|
|
|
APP.store.dispatch(setJoiningInProgress(false));
|
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
|
|
|
* Check if id is id of the local user.
|
|
|
|
* @param {string} id id to check
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
isLocalId(id) {
|
2016-07-08 01:44:04 +00:00
|
|
|
return this.getMyUserId() === id;
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-08-18 11:30:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Tells whether the local video is muted or not.
|
|
|
|
* @return {boolean}
|
|
|
|
*/
|
|
|
|
isLocalVideoMuted() {
|
|
|
|
// If the tracks are not ready, read from base/media state
|
|
|
|
return this._localTracksInitialized
|
2020-10-26 18:05:22 +00:00
|
|
|
? isLocalCameraTrackMuted(
|
2019-11-26 10:57:03 +00:00
|
|
|
APP.store.getState()['features/base/tracks'])
|
2017-08-18 11:30:30 +00:00
|
|
|
: isVideoMutedByUser(APP.store);
|
|
|
|
},
|
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
/**
|
|
|
|
* Simulates toolbar button click for audio mute. Used by shortcuts and API.
|
2017-07-25 10:05:08 +00:00
|
|
|
* @param {boolean} mute true for mute and false for unmute.
|
|
|
|
* @param {boolean} [showUI] when set to false will not display any error
|
|
|
|
* dialogs in case of media permissions error.
|
2016-01-06 22:39:13 +00:00
|
|
|
*/
|
2017-07-25 10:05:08 +00:00
|
|
|
muteAudio(mute, showUI = true) {
|
2021-09-15 08:28:44 +00:00
|
|
|
const state = APP.store.getState();
|
|
|
|
|
2019-07-10 11:02:27 +00:00
|
|
|
if (!mute
|
2021-09-15 08:28:44 +00:00
|
|
|
&& isUserInteractionRequiredForUnmute(state)) {
|
2019-07-10 11:02:27 +00:00
|
|
|
logger.error('Unmuting audio requires user interaction');
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-15 08:28:44 +00:00
|
|
|
// check for A/V Moderation when trying to unmute
|
|
|
|
if (!mute && shouldShowModeratedNotification(MEDIA_TYPE.AUDIO, state)) {
|
|
|
|
if (!isModerationNotificationDisplayed(MEDIA_TYPE.AUDIO, state)) {
|
|
|
|
APP.store.dispatch(showModeratedNotification(MEDIA_TYPE.AUDIO));
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-07-25 10:05:08 +00:00
|
|
|
// Not ready to modify track's state yet
|
|
|
|
if (!this._localTracksInitialized) {
|
2017-08-18 11:30:30 +00:00
|
|
|
// This will only modify base/media.audio.muted which is then synced
|
|
|
|
// up with the track at the end of local tracks initialization.
|
|
|
|
muteLocalAudio(mute);
|
2017-08-04 08:15:11 +00:00
|
|
|
this.setAudioMuteStatus(mute);
|
2017-08-18 11:30:30 +00:00
|
|
|
|
2017-07-25 10:05:08 +00:00
|
|
|
return;
|
2017-08-18 11:30:30 +00:00
|
|
|
} else if (this.isLocalAudioMuted() === mute) {
|
2017-07-25 10:05:08 +00:00
|
|
|
// NO-OP
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
|
|
|
|
if (!localAudio && !mute) {
|
2017-08-14 13:25:37 +00:00
|
|
|
const maybeShowErrorDialog = error => {
|
2019-05-29 21:17:07 +00:00
|
|
|
showUI && APP.store.dispatch(notifyMicError(error));
|
2017-08-14 13:25:37 +00:00
|
|
|
};
|
|
|
|
|
2020-11-18 12:38:00 +00:00
|
|
|
createLocalTracksF({ devices: [ 'audio' ] })
|
2017-10-12 23:02:29 +00:00
|
|
|
.then(([ audioTrack ]) => audioTrack)
|
2017-07-25 10:05:08 +00:00
|
|
|
.catch(error => {
|
2017-08-14 13:25:37 +00:00
|
|
|
maybeShowErrorDialog(error);
|
2017-07-25 10:05:08 +00:00
|
|
|
|
|
|
|
// Rollback the audio muted status by using null track
|
|
|
|
return null;
|
|
|
|
})
|
|
|
|
.then(audioTrack => this.useAudioStream(audioTrack));
|
|
|
|
} else {
|
2017-08-18 11:30:30 +00:00
|
|
|
muteLocalAudio(mute);
|
2017-07-25 10:05:08 +00:00
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-02-09 16:29:50 +00:00
|
|
|
/**
|
|
|
|
* Returns whether local audio is muted or not.
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
isLocalAudioMuted() {
|
2017-08-18 11:30:30 +00:00
|
|
|
// If the tracks are not ready, read from base/media state
|
|
|
|
return this._localTracksInitialized
|
|
|
|
? isLocalTrackMuted(
|
|
|
|
APP.store.getState()['features/base/tracks'],
|
|
|
|
MEDIA_TYPE.AUDIO)
|
|
|
|
: Boolean(
|
|
|
|
APP.store.getState()['features/base/media'].audio.muted);
|
2016-02-09 16:29:50 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
/**
|
2017-04-04 20:55:03 +00:00
|
|
|
* Simulates toolbar button click for audio mute. Used by shortcuts
|
|
|
|
* and API.
|
2017-07-25 10:05:08 +00:00
|
|
|
* @param {boolean} [showUI] when set to false will not display any error
|
|
|
|
* dialogs in case of media permissions error.
|
2016-01-06 22:39:13 +00:00
|
|
|
*/
|
2017-07-25 10:05:08 +00:00
|
|
|
toggleAudioMuted(showUI = true) {
|
2017-08-18 11:30:30 +00:00
|
|
|
this.muteAudio(!this.isLocalAudioMuted(), showUI);
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2019-11-26 10:57:03 +00:00
|
|
|
/**
|
|
|
|
* Simulates toolbar button click for presenter video mute. Used by
|
|
|
|
* shortcuts and API.
|
|
|
|
* @param mute true for mute and false for unmute.
|
|
|
|
* @param {boolean} [showUI] when set to false will not display any error
|
|
|
|
* dialogs in case of media permissions error.
|
|
|
|
*/
|
2019-12-04 20:28:42 +00:00
|
|
|
async mutePresenter(mute, showUI = true) {
|
2019-11-26 10:57:03 +00:00
|
|
|
const maybeShowErrorDialog = error => {
|
|
|
|
showUI && APP.store.dispatch(notifyCameraError(error));
|
|
|
|
};
|
2021-09-13 17:33:04 +00:00
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
2019-11-26 10:57:03 +00:00
|
|
|
|
|
|
|
if (mute) {
|
|
|
|
try {
|
2021-09-13 17:33:04 +00:00
|
|
|
await localVideo.setEffect(undefined);
|
2019-11-26 10:57:03 +00:00
|
|
|
} catch (err) {
|
2019-12-04 20:28:42 +00:00
|
|
|
logger.error('Failed to remove the presenter effect', err);
|
|
|
|
maybeShowErrorDialog(err);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
try {
|
2021-09-13 17:33:04 +00:00
|
|
|
await localVideo.setEffect(await this._createPresenterStreamEffect());
|
2019-12-04 20:28:42 +00:00
|
|
|
} catch (err) {
|
|
|
|
logger.error('Failed to apply the presenter effect', err);
|
|
|
|
maybeShowErrorDialog(err);
|
2019-11-26 10:57:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
/**
|
|
|
|
* Simulates toolbar button click for video mute. Used by shortcuts and API.
|
|
|
|
* @param mute true for mute and false for unmute.
|
2017-07-20 12:29:15 +00:00
|
|
|
* @param {boolean} [showUI] when set to false will not display any error
|
|
|
|
* dialogs in case of media permissions error.
|
2016-01-06 22:39:13 +00:00
|
|
|
*/
|
2017-07-20 12:29:15 +00:00
|
|
|
muteVideo(mute, showUI = true) {
|
2019-07-10 11:02:27 +00:00
|
|
|
if (!mute
|
|
|
|
&& isUserInteractionRequiredForUnmute(APP.store.getState())) {
|
|
|
|
logger.error('Unmuting video requires user interaction');
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2019-11-26 10:57:03 +00:00
|
|
|
if (this.isSharingScreen) {
|
2020-01-10 21:39:58 +00:00
|
|
|
// Chain _mutePresenterVideo calls
|
|
|
|
_prevMutePresenterVideo = _prevMutePresenterVideo.then(() => this._mutePresenterVideo(mute));
|
|
|
|
|
|
|
|
return;
|
2019-11-26 10:57:03 +00:00
|
|
|
}
|
|
|
|
|
2017-08-18 11:30:30 +00:00
|
|
|
// If not ready to modify track's state yet adjust the base/media
|
2017-07-21 09:12:33 +00:00
|
|
|
if (!this._localTracksInitialized) {
|
2017-08-18 11:30:30 +00:00
|
|
|
// This will only modify base/media.video.muted which is then synced
|
|
|
|
// up with the track at the end of local tracks initialization.
|
|
|
|
muteLocalVideo(mute);
|
2021-05-04 12:57:34 +00:00
|
|
|
this.setVideoMuteStatus();
|
2017-07-21 09:12:33 +00:00
|
|
|
|
2017-07-24 15:36:19 +00:00
|
|
|
return;
|
2017-08-18 11:30:30 +00:00
|
|
|
} else if (this.isLocalVideoMuted() === mute) {
|
2017-07-24 15:36:19 +00:00
|
|
|
// NO-OP
|
2017-07-21 09:12:33 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
|
|
|
|
|
|
|
if (!localVideo && !mute) {
|
2017-08-14 13:25:37 +00:00
|
|
|
const maybeShowErrorDialog = error => {
|
2019-05-29 21:17:07 +00:00
|
|
|
showUI && APP.store.dispatch(notifyCameraError(error));
|
2017-08-14 13:25:37 +00:00
|
|
|
};
|
|
|
|
|
2017-07-20 12:29:15 +00:00
|
|
|
// Try to create local video if there wasn't any.
|
|
|
|
// This handles the case when user joined with no video
|
|
|
|
// (dismissed screen sharing screen or in audio only mode), but
|
|
|
|
// decided to add it later on by clicking on muted video icon or
|
|
|
|
// turning off the audio only mode.
|
|
|
|
//
|
|
|
|
// FIXME when local track creation is moved to react/redux
|
|
|
|
// it should take care of the use case described above
|
2020-11-18 12:38:00 +00:00
|
|
|
createLocalTracksF({ devices: [ 'video' ] })
|
2017-10-12 23:02:29 +00:00
|
|
|
.then(([ videoTrack ]) => videoTrack)
|
2017-07-20 12:29:15 +00:00
|
|
|
.catch(error => {
|
|
|
|
// FIXME should send some feedback to the API on error ?
|
2017-07-24 09:20:32 +00:00
|
|
|
maybeShowErrorDialog(error);
|
|
|
|
|
2017-07-20 12:29:15 +00:00
|
|
|
// Rollback the video muted status by using null track
|
|
|
|
return null;
|
|
|
|
})
|
2021-03-05 18:17:39 +00:00
|
|
|
.then(videoTrack => {
|
|
|
|
logger.debug(`muteVideo: calling useVideoStream for track: ${videoTrack}`);
|
|
|
|
|
|
|
|
return this.useVideoStream(videoTrack);
|
|
|
|
});
|
2017-07-20 12:29:15 +00:00
|
|
|
} else {
|
2017-08-18 11:30:30 +00:00
|
|
|
// FIXME show error dialog if it fails (should be handled by react)
|
|
|
|
muteLocalVideo(mute);
|
2017-07-20 12:29:15 +00:00
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
/**
|
|
|
|
* Simulates toolbar button click for video mute. Used by shortcuts and API.
|
2017-07-20 12:29:15 +00:00
|
|
|
* @param {boolean} [showUI] when set to false will not display any error
|
|
|
|
* dialogs in case of media permissions error.
|
2016-01-06 22:39:13 +00:00
|
|
|
*/
|
2017-07-20 12:29:15 +00:00
|
|
|
toggleVideoMuted(showUI = true) {
|
2017-08-18 11:30:30 +00:00
|
|
|
this.muteVideo(!this.isLocalVideoMuted(), showUI);
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
|
|
|
* Retrieve list of ids of conference participants (without local user).
|
|
|
|
* @returns {string[]}
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
listMembersIds() {
|
2016-01-06 22:39:13 +00:00
|
|
|
return room.getParticipants().map(p => p.getId());
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-03-24 20:25:26 +00:00
|
|
|
/**
|
|
|
|
* Checks whether the participant identified by id is a moderator.
|
|
|
|
* @id id to search for participant
|
|
|
|
* @return {boolean} whether the participant is moderator
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
isParticipantModerator(id) {
|
2017-10-12 23:02:29 +00:00
|
|
|
const user = room.getParticipantById(id);
|
|
|
|
|
2016-03-24 20:25:26 +00:00
|
|
|
return user && user.isModerator();
|
|
|
|
},
|
2017-10-16 20:37:13 +00:00
|
|
|
|
2020-02-21 09:17:11 +00:00
|
|
|
/**
|
|
|
|
* Retrieve list of conference participants (without local user).
|
|
|
|
* @returns {JitsiParticipant[]}
|
|
|
|
*
|
|
|
|
* NOTE: Used by jitsi-meet-torture!
|
|
|
|
*/
|
|
|
|
listMembers() {
|
|
|
|
return room.getParticipants();
|
|
|
|
},
|
|
|
|
|
2017-04-11 19:40:03 +00:00
|
|
|
get membersCount() {
|
2015-12-31 15:23:23 +00:00
|
|
|
return room.getParticipants().length + 1;
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-20 21:41:37 +00:00
|
|
|
/**
|
|
|
|
* Returns true if the callstats integration is enabled, otherwise returns
|
|
|
|
* false.
|
|
|
|
*
|
|
|
|
* @returns true if the callstats integration is enabled, otherwise returns
|
|
|
|
* false.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
isCallstatsEnabled() {
|
2017-02-06 13:32:05 +00:00
|
|
|
return room && room.isCallstatsEnabled();
|
2016-01-20 21:41:37 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-02-17 00:59:30 +00:00
|
|
|
/**
|
|
|
|
* Get speaker stats that track total dominant speaker time.
|
|
|
|
*
|
|
|
|
* @returns {object} A hash with keys being user ids and values being the
|
|
|
|
* library's SpeakerStats model used for calculating time as dominant
|
|
|
|
* speaker.
|
|
|
|
*/
|
|
|
|
getSpeakerStats() {
|
|
|
|
return room.getSpeakerStats();
|
|
|
|
},
|
|
|
|
|
2016-08-03 16:08:23 +00:00
|
|
|
/**
|
|
|
|
* Returns the connection times stored in the library.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
getConnectionTimes() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room.getConnectionTimes();
|
2016-08-03 16:08:23 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
// used by torture currently
|
2017-04-11 19:40:03 +00:00
|
|
|
isJoined() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room && room.isJoined();
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-04-11 19:40:03 +00:00
|
|
|
getConnectionState() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room && room.getConnectionState();
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-02-15 21:19:52 +00:00
|
|
|
/**
|
|
|
|
* Obtains current P2P ICE connection state.
|
|
|
|
* @return {string|null} ICE connection state or <tt>null</tt> if there's no
|
|
|
|
* P2P connection
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
getP2PConnectionState() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room && room.getP2PConnectionState();
|
2017-02-15 21:19:52 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-02-15 21:19:52 +00:00
|
|
|
/**
|
|
|
|
* Starts P2P (for tests only)
|
|
|
|
* @private
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
_startP2P() {
|
2017-02-15 21:19:52 +00:00
|
|
|
try {
|
2019-07-02 13:11:56 +00:00
|
|
|
room && room.startP2PSession();
|
2017-02-15 21:19:52 +00:00
|
|
|
} catch (error) {
|
2017-10-12 23:02:29 +00:00
|
|
|
logger.error('Start P2P failed', error);
|
2017-02-15 21:19:52 +00:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-02-15 21:19:52 +00:00
|
|
|
/**
|
|
|
|
* Stops P2P (for tests only)
|
|
|
|
* @private
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
_stopP2P() {
|
2017-02-15 21:19:52 +00:00
|
|
|
try {
|
2019-07-02 13:11:56 +00:00
|
|
|
room && room.stopP2PSession();
|
2017-02-15 21:19:52 +00:00
|
|
|
} catch (error) {
|
2017-10-12 23:02:29 +00:00
|
|
|
logger.error('Stop P2P failed', error);
|
2017-02-15 21:19:52 +00:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-07-28 16:09:22 +00:00
|
|
|
/**
|
|
|
|
* Checks whether or not our connection is currently in interrupted and
|
|
|
|
* reconnect attempts are in progress.
|
|
|
|
*
|
|
|
|
* @returns {boolean} true if the connection is in interrupted state or
|
|
|
|
* false otherwise.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
isConnectionInterrupted() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room.isConnectionInterrupted();
|
2016-07-28 16:09:22 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-10-06 17:52:23 +00:00
|
|
|
/**
|
|
|
|
* Obtains the local display name.
|
|
|
|
* @returns {string|undefined}
|
|
|
|
*/
|
|
|
|
getLocalDisplayName() {
|
|
|
|
return getDisplayName(this.getMyUserId());
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-09-16 20:40:24 +00:00
|
|
|
/**
|
|
|
|
* Finds JitsiParticipant for given id.
|
|
|
|
*
|
|
|
|
* @param {string} id participant's identifier(MUC nickname).
|
|
|
|
*
|
|
|
|
* @returns {JitsiParticipant|null} participant instance for given id or
|
|
|
|
* null if not found.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
getParticipantById(id) {
|
2016-09-16 20:40:24 +00:00
|
|
|
return room ? room.getParticipantById(id) : null;
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-09-24 16:46:19 +00:00
|
|
|
/**
|
|
|
|
* Gets the display name foe the <tt>JitsiParticipant</tt> identified by
|
|
|
|
* the given <tt>id</tt>.
|
|
|
|
*
|
|
|
|
* @param id {string} the participant's id(MUC nickname/JVB endpoint id)
|
|
|
|
*
|
|
|
|
* @return {string} the participant's display name or the default string if
|
|
|
|
* absent.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
getParticipantDisplayName(id) {
|
2017-10-12 23:02:29 +00:00
|
|
|
const displayName = getDisplayName(id);
|
|
|
|
|
2016-09-24 16:46:19 +00:00
|
|
|
if (displayName) {
|
|
|
|
return displayName;
|
2017-10-12 23:02:29 +00:00
|
|
|
}
|
|
|
|
if (APP.conference.isLocalId(id)) {
|
|
|
|
return APP.translation.generateTranslationHTML(
|
2016-09-24 16:46:19 +00:00
|
|
|
interfaceConfig.DEFAULT_LOCAL_DISPLAY_NAME);
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
return interfaceConfig.DEFAULT_REMOTE_DISPLAY_NAME;
|
2016-09-24 16:46:19 +00:00
|
|
|
},
|
2017-10-16 20:37:13 +00:00
|
|
|
|
2017-04-11 19:40:03 +00:00
|
|
|
getMyUserId() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room && room.myUserId();
|
2016-01-06 22:39:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-19 19:32:29 +00:00
|
|
|
/**
|
|
|
|
* Will be filled with values only when config.debug is enabled.
|
|
|
|
* Its used by torture to check audio levels.
|
|
|
|
*/
|
|
|
|
audioLevelsMap: {},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-02-02 20:54:15 +00:00
|
|
|
/**
|
|
|
|
* Returns the stored audio level (stored only if config.debug is enabled)
|
|
|
|
* @param id the id for the user audio level to return (the id value is
|
|
|
|
* returned for the participant using getMyUserId() method)
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
getPeerSSRCAudioLevel(id) {
|
2016-01-19 19:32:29 +00:00
|
|
|
return this.audioLevelsMap[id];
|
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-14 19:50:10 +00:00
|
|
|
/**
|
2016-04-21 22:48:30 +00:00
|
|
|
* @return {number} the number of participants in the conference with at
|
|
|
|
* least one track.
|
2016-01-14 19:50:10 +00:00
|
|
|
*/
|
2016-04-21 22:48:30 +00:00
|
|
|
getNumberOfParticipantsWithTracks() {
|
2019-07-02 13:11:56 +00:00
|
|
|
return room.getParticipants()
|
2017-10-12 23:02:29 +00:00
|
|
|
.filter(p => p.getTracks().length > 0)
|
2016-04-21 22:48:30 +00:00
|
|
|
.length;
|
2016-01-14 19:50:10 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-02-04 18:41:23 +00:00
|
|
|
/**
|
|
|
|
* Returns the stats.
|
|
|
|
*/
|
|
|
|
getStats() {
|
2016-10-26 19:29:40 +00:00
|
|
|
return room.connectionQuality.getStats();
|
2016-02-04 18:41:23 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-14 19:50:10 +00:00
|
|
|
// end used by torture
|
|
|
|
|
2016-09-21 20:46:10 +00:00
|
|
|
/**
|
|
|
|
* Download logs, a function that can be called from console while
|
|
|
|
* debugging.
|
|
|
|
* @param filename (optional) specify target filename
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
saveLogs(filename = 'meetlog.json') {
|
2016-09-21 20:46:10 +00:00
|
|
|
// this can be called from console and will not have reference to this
|
|
|
|
// that's why we reference the global var
|
2020-08-13 18:12:56 +00:00
|
|
|
const logs = APP.connection.getLogs();
|
2020-10-02 13:20:24 +00:00
|
|
|
|
|
|
|
downloadJSON(logs, filename);
|
2016-09-21 20:46:10 +00:00
|
|
|
},
|
|
|
|
|
2016-03-11 10:49:13 +00:00
|
|
|
/**
|
|
|
|
* Exposes a Command(s) API on this instance. It is necessitated by (1) the
|
|
|
|
* desire to keep room private to this instance and (2) the need of other
|
|
|
|
* modules to send and receive commands to and from participants.
|
|
|
|
* Eventually, this instance remains in control with respect to the
|
|
|
|
* decision whether the Command(s) API of room (i.e. lib-jitsi-meet's
|
|
|
|
* JitsiConference) is to be used in the implementation of the Command(s)
|
|
|
|
* API of this instance.
|
|
|
|
*/
|
|
|
|
commands: {
|
2016-04-07 17:08:00 +00:00
|
|
|
/**
|
|
|
|
* Known custom conference commands.
|
|
|
|
*/
|
2016-06-13 21:11:44 +00:00
|
|
|
defaults: commands,
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-03-11 10:49:13 +00:00
|
|
|
/**
|
|
|
|
* Receives notifications from other participants about commands aka
|
|
|
|
* custom events (sent by sendCommand or sendCommandOnce methods).
|
|
|
|
* @param command {String} the name of the command
|
|
|
|
* @param handler {Function} handler for the command
|
|
|
|
*/
|
2017-10-12 23:02:29 +00:00
|
|
|
addCommandListener() {
|
|
|
|
// eslint-disable-next-line prefer-rest-params
|
|
|
|
room.addCommandListener(...arguments);
|
2016-03-11 10:49:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-03-11 10:49:13 +00:00
|
|
|
/**
|
|
|
|
* Removes command.
|
|
|
|
* @param name {String} the name of the command.
|
|
|
|
*/
|
2017-10-12 23:02:29 +00:00
|
|
|
removeCommand() {
|
|
|
|
// eslint-disable-next-line prefer-rest-params
|
|
|
|
room.removeCommand(...arguments);
|
2016-03-11 10:49:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-03-11 10:49:13 +00:00
|
|
|
/**
|
|
|
|
* Sends command.
|
|
|
|
* @param name {String} the name of the command.
|
|
|
|
* @param values {Object} with keys and values that will be sent.
|
|
|
|
*/
|
2017-10-12 23:02:29 +00:00
|
|
|
sendCommand() {
|
|
|
|
// eslint-disable-next-line prefer-rest-params
|
|
|
|
room.sendCommand(...arguments);
|
2016-03-11 10:49:13 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-03-11 10:49:13 +00:00
|
|
|
/**
|
|
|
|
* Sends command one time.
|
|
|
|
* @param name {String} the name of the command.
|
|
|
|
* @param values {Object} with keys and values that will be sent.
|
|
|
|
*/
|
2017-10-12 23:02:29 +00:00
|
|
|
sendCommandOnce() {
|
|
|
|
// eslint-disable-next-line prefer-rest-params
|
|
|
|
room.sendCommandOnce(...arguments);
|
2016-04-07 17:08:00 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-10-13 19:31:05 +00:00
|
|
|
_createRoom(localTracks) {
|
|
|
|
room
|
|
|
|
= connection.initJitsiConference(
|
|
|
|
APP.conference.roomName,
|
|
|
|
this._getConferenceOptions());
|
2019-02-07 03:19:02 +00:00
|
|
|
|
2021-08-31 15:06:31 +00:00
|
|
|
// Filter out the tracks that are muted (except on mobile Safari).
|
|
|
|
const tracks = isIosMobileBrowser() ? localTracks : localTracks.filter(track => !track.isMuted());
|
2021-03-05 15:18:34 +00:00
|
|
|
|
|
|
|
this._setLocalAudioVideoStreams(tracks);
|
2016-04-07 17:08:00 +00:00
|
|
|
this._room = room; // FIXME do not use this
|
|
|
|
|
2021-09-14 21:57:05 +00:00
|
|
|
APP.store.dispatch(_conferenceWillJoin(room));
|
|
|
|
|
2017-10-13 19:31:05 +00:00
|
|
|
sendLocalParticipant(APP.store, room);
|
2016-04-07 17:08:00 +00:00
|
|
|
|
|
|
|
this._setupListeners();
|
2016-03-11 10:49:13 +00:00
|
|
|
},
|
|
|
|
|
2016-06-13 11:49:00 +00:00
|
|
|
/**
|
|
|
|
* Sets local video and audio streams.
|
|
|
|
* @param {JitsiLocalTrack[]} tracks=[]
|
|
|
|
* @returns {Promise[]}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_setLocalAudioVideoStreams(tracks = []) {
|
2021-09-13 17:33:04 +00:00
|
|
|
const promises = tracks.map(track => {
|
2016-06-13 11:49:00 +00:00
|
|
|
if (track.isAudioTrack()) {
|
|
|
|
return this.useAudioStream(track);
|
|
|
|
} else if (track.isVideoTrack()) {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug(`_setLocalAudioVideoStreams is calling useVideoStream with track: ${track}`);
|
|
|
|
|
2016-06-13 11:49:00 +00:00
|
|
|
return this.useVideoStream(track);
|
|
|
|
}
|
2021-03-05 18:17:39 +00:00
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
logger.error('Ignored not an audio nor a video track: ', track);
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
return Promise.resolve();
|
|
|
|
|
2016-06-13 11:49:00 +00:00
|
|
|
});
|
2021-09-13 17:33:04 +00:00
|
|
|
|
|
|
|
return Promise.allSettled(promises).then(() => {
|
|
|
|
this._localTracksInitialized = true;
|
|
|
|
logger.log(`Initialized with ${tracks.length} local tracks`);
|
|
|
|
});
|
2016-06-13 11:49:00 +00:00
|
|
|
},
|
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
_getConferenceOptions() {
|
2021-08-04 10:56:07 +00:00
|
|
|
const options = getConferenceOptions(APP.store.getState());
|
|
|
|
|
|
|
|
options.createVADProcessor = createRnnoiseProcessor;
|
|
|
|
|
|
|
|
return options;
|
2019-07-08 10:54:54 +00:00
|
|
|
},
|
|
|
|
|
2017-04-11 19:40:03 +00:00
|
|
|
/**
|
|
|
|
* Start using provided video stream.
|
|
|
|
* Stops previous video stream.
|
2020-06-26 08:54:12 +00:00
|
|
|
* @param {JitsiLocalTrack} newTrack - new track to use or null
|
2017-04-11 19:40:03 +00:00
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
2020-06-26 08:54:12 +00:00
|
|
|
useVideoStream(newTrack) {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug(`useVideoStream: ${newTrack}`);
|
|
|
|
|
2018-08-31 20:02:04 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
_replaceLocalVideoTrackQueue.enqueue(onFinish => {
|
2021-09-13 17:33:04 +00:00
|
|
|
const oldTrack = getLocalJitsiVideoTrack(APP.store.getState());
|
2020-06-26 08:54:12 +00:00
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
logger.debug(`useVideoStream: Replacing ${oldTrack} with ${newTrack}`);
|
2020-06-26 08:54:12 +00:00
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
if (oldTrack === newTrack) {
|
|
|
|
resolve();
|
|
|
|
onFinish();
|
2021-03-05 18:17:39 +00:00
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
return;
|
2020-06-26 08:54:12 +00:00
|
|
|
}
|
|
|
|
|
2018-08-31 20:02:04 +00:00
|
|
|
APP.store.dispatch(
|
2021-09-13 17:33:04 +00:00
|
|
|
replaceLocalTrack(oldTrack, newTrack, room))
|
2018-08-31 20:02:04 +00:00
|
|
|
.then(() => {
|
2020-06-26 08:54:12 +00:00
|
|
|
this._setSharingScreen(newTrack);
|
2021-05-04 12:57:34 +00:00
|
|
|
this.setVideoMuteStatus();
|
2018-08-31 20:02:04 +00:00
|
|
|
})
|
|
|
|
.then(resolve)
|
2021-03-05 18:17:39 +00:00
|
|
|
.catch(error => {
|
|
|
|
logger.error(`useVideoStream failed: ${error}`);
|
|
|
|
reject(error);
|
|
|
|
})
|
2018-08-31 20:02:04 +00:00
|
|
|
.then(onFinish);
|
2017-01-19 18:46:11 +00:00
|
|
|
});
|
2018-08-31 20:02:04 +00:00
|
|
|
});
|
2016-02-09 10:19:43 +00:00
|
|
|
},
|
|
|
|
|
2018-01-30 13:43:06 +00:00
|
|
|
/**
|
|
|
|
* Sets `this.isSharingScreen` depending on provided video stream.
|
|
|
|
* In case new screen sharing status is not equal previous one
|
|
|
|
* it updates desktop sharing buttons in UI
|
|
|
|
* and notifies external application.
|
|
|
|
*
|
|
|
|
* @param {JitsiLocalTrack} [newStream] new stream to use or null
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_setSharingScreen(newStream) {
|
|
|
|
const wasSharingScreen = this.isSharingScreen;
|
|
|
|
|
|
|
|
this.isSharingScreen = newStream && newStream.videoType === 'desktop';
|
|
|
|
|
|
|
|
if (wasSharingScreen !== this.isSharingScreen) {
|
2019-03-07 05:46:17 +00:00
|
|
|
const details = {};
|
|
|
|
|
|
|
|
if (this.isSharingScreen) {
|
|
|
|
details.sourceType = newStream.sourceType;
|
|
|
|
}
|
|
|
|
|
|
|
|
APP.API.notifyScreenSharingStatusChanged(
|
|
|
|
this.isSharingScreen, details);
|
2018-01-30 13:43:06 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-02-09 10:19:43 +00:00
|
|
|
/**
|
|
|
|
* Start using provided audio stream.
|
|
|
|
* Stops previous audio stream.
|
2020-06-26 08:54:12 +00:00
|
|
|
* @param {JitsiLocalTrack} newTrack - new track to use or null
|
2016-02-16 13:33:53 +00:00
|
|
|
* @returns {Promise}
|
2016-02-09 10:19:43 +00:00
|
|
|
*/
|
2020-06-26 08:54:12 +00:00
|
|
|
useAudioStream(newTrack) {
|
2018-08-31 20:02:04 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
_replaceLocalAudioTrackQueue.enqueue(onFinish => {
|
2021-09-13 17:33:04 +00:00
|
|
|
const oldTrack = getLocalJitsiAudioTrack(APP.store.getState());
|
2020-06-26 08:54:12 +00:00
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
if (oldTrack === newTrack) {
|
|
|
|
resolve();
|
|
|
|
onFinish();
|
2020-06-26 08:54:12 +00:00
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
return;
|
2020-06-26 08:54:12 +00:00
|
|
|
}
|
|
|
|
|
2018-08-31 20:02:04 +00:00
|
|
|
APP.store.dispatch(
|
2021-09-13 17:33:04 +00:00
|
|
|
replaceLocalTrack(oldTrack, newTrack, room))
|
2018-08-31 20:02:04 +00:00
|
|
|
.then(() => {
|
|
|
|
this.setAudioMuteStatus(this.isLocalAudioMuted());
|
|
|
|
})
|
|
|
|
.then(resolve)
|
|
|
|
.catch(reject)
|
|
|
|
.then(onFinish);
|
2017-01-19 18:46:11 +00:00
|
|
|
});
|
2018-08-31 20:02:04 +00:00
|
|
|
});
|
2016-02-09 10:19:43 +00:00
|
|
|
},
|
|
|
|
|
2017-04-05 15:14:26 +00:00
|
|
|
/**
|
|
|
|
* Returns whether or not the conference is currently in audio only mode.
|
|
|
|
*
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
isAudioOnly() {
|
2019-07-31 12:47:52 +00:00
|
|
|
return Boolean(APP.store.getState()['features/base/audio-only'].enabled);
|
2017-04-05 15:14:26 +00:00
|
|
|
},
|
2017-03-29 17:43:30 +00:00
|
|
|
|
2016-02-04 15:25:11 +00:00
|
|
|
videoSwitchInProgress: false,
|
2017-06-13 19:24:34 +00:00
|
|
|
|
2017-06-20 14:52:44 +00:00
|
|
|
/**
|
|
|
|
* This fields stores a handler which will create a Promise which turns off
|
|
|
|
* the screen sharing and restores the previous video state (was there
|
|
|
|
* any video, before switching to screen sharing ? was it muted ?).
|
|
|
|
*
|
|
|
|
* Once called this fields is cleared to <tt>null</tt>.
|
|
|
|
* @type {Function|null}
|
|
|
|
*/
|
|
|
|
_untoggleScreenSharing: null,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a Promise which turns off the screen sharing and restores
|
|
|
|
* the previous state described by the arguments.
|
|
|
|
*
|
|
|
|
* This method is bound to the appropriate values, after switching to screen
|
|
|
|
* sharing and stored in {@link _untoggleScreenSharing}.
|
|
|
|
*
|
|
|
|
* @param {boolean} didHaveVideo indicates if there was a camera video being
|
|
|
|
* used, before switching to screen sharing.
|
2021-09-22 12:11:43 +00:00
|
|
|
* @param {boolean} ignoreDidHaveVideo indicates if the camera video should be
|
|
|
|
* ignored when switching screen sharing off.
|
2017-06-20 14:52:44 +00:00
|
|
|
* @return {Promise} resolved after the screen sharing is turned off, or
|
|
|
|
* rejected with some error (no idea what kind of error, possible GUM error)
|
|
|
|
* in case it fails.
|
|
|
|
* @private
|
|
|
|
*/
|
2021-09-22 12:11:43 +00:00
|
|
|
async _turnScreenSharingOff(didHaveVideo, ignoreDidHaveVideo) {
|
2017-06-20 14:52:44 +00:00
|
|
|
this._untoggleScreenSharing = null;
|
|
|
|
this.videoSwitchInProgress = true;
|
2017-10-20 00:38:55 +00:00
|
|
|
|
2020-11-14 04:09:25 +00:00
|
|
|
APP.store.dispatch(stopReceiver());
|
2017-10-20 00:38:55 +00:00
|
|
|
|
2019-01-26 20:53:11 +00:00
|
|
|
this._stopProxyConnection();
|
2021-07-26 11:38:56 +00:00
|
|
|
|
2020-02-25 15:22:10 +00:00
|
|
|
if (config.enableScreenshotCapture) {
|
2021-07-26 11:38:56 +00:00
|
|
|
APP.store.dispatch(toggleScreenshotCaptureSummary(false));
|
2020-02-25 15:22:10 +00:00
|
|
|
}
|
2019-01-26 20:53:11 +00:00
|
|
|
|
2020-01-13 14:21:31 +00:00
|
|
|
// It can happen that presenter GUM is in progress while screensharing is being turned off. Here it needs to
|
|
|
|
// wait for that GUM to be resolved in order to prevent leaking the presenter track(this.localPresenterVideo
|
|
|
|
// will be null when SS is being turned off, but it will initialize once GUM resolves).
|
|
|
|
let promise = _prevMutePresenterVideo = _prevMutePresenterVideo.then(() => {
|
|
|
|
// mute the presenter track if it exists.
|
|
|
|
if (this.localPresenterVideo) {
|
|
|
|
APP.store.dispatch(setVideoMuted(true, MEDIA_TYPE.PRESENTER));
|
|
|
|
|
|
|
|
return this.localPresenterVideo.dispose().then(() => {
|
|
|
|
APP.store.dispatch(trackRemoved(this.localPresenterVideo));
|
|
|
|
this.localPresenterVideo = null;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
2017-06-20 14:52:44 +00:00
|
|
|
|
2020-03-26 12:17:44 +00:00
|
|
|
// If system audio was also shared stop the AudioMixerEffect and dispose of the desktop audio track.
|
|
|
|
if (this._mixerEffect) {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
|
|
|
|
await localAudio.setEffect(undefined);
|
2020-03-26 12:17:44 +00:00
|
|
|
await this._desktopAudioStream.dispose();
|
|
|
|
this._mixerEffect = undefined;
|
|
|
|
this._desktopAudioStream = undefined;
|
|
|
|
|
|
|
|
// In case there was no local audio when screen sharing was started the fact that we set the audio stream to
|
|
|
|
// null will take care of the desktop audio stream cleanup.
|
|
|
|
} else if (this._desktopAudioStream) {
|
|
|
|
await this.useAudioStream(null);
|
|
|
|
this._desktopAudioStream = undefined;
|
|
|
|
}
|
|
|
|
|
2021-04-12 07:37:39 +00:00
|
|
|
APP.store.dispatch(setScreenAudioShareState(false));
|
|
|
|
|
2021-09-22 12:11:43 +00:00
|
|
|
if (didHaveVideo && !ignoreDidHaveVideo) {
|
2020-01-13 14:21:31 +00:00
|
|
|
promise = promise.then(() => createLocalTracksF({ devices: [ 'video' ] }))
|
2021-03-05 18:17:39 +00:00
|
|
|
.then(([ stream ]) => {
|
|
|
|
logger.debug(`_turnScreenSharingOff using ${stream} for useVideoStream`);
|
|
|
|
|
|
|
|
return this.useVideoStream(stream);
|
|
|
|
})
|
2017-06-20 14:52:44 +00:00
|
|
|
.catch(error => {
|
|
|
|
logger.error('failed to switch back to local video', error);
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
return this.useVideoStream(null).then(() =>
|
|
|
|
|
2017-06-20 14:52:44 +00:00
|
|
|
// Still fail with the original err
|
2017-10-12 23:02:29 +00:00
|
|
|
Promise.reject(error)
|
|
|
|
);
|
2017-06-20 14:52:44 +00:00
|
|
|
});
|
|
|
|
} else {
|
2021-03-05 18:17:39 +00:00
|
|
|
promise = promise.then(() => {
|
|
|
|
logger.debug('_turnScreenSharingOff using null for useVideoStream');
|
|
|
|
|
|
|
|
return this.useVideoStream(null);
|
|
|
|
});
|
2019-11-26 10:57:03 +00:00
|
|
|
}
|
2019-09-19 13:28:57 +00:00
|
|
|
|
2017-06-20 14:52:44 +00:00
|
|
|
return promise.then(
|
|
|
|
() => {
|
|
|
|
this.videoSwitchInProgress = false;
|
2020-08-04 21:11:58 +00:00
|
|
|
sendAnalytics(createScreenSharingEvent('stopped'));
|
|
|
|
logger.info('Screen sharing stopped.');
|
2017-06-20 14:52:44 +00:00
|
|
|
},
|
|
|
|
error => {
|
|
|
|
this.videoSwitchInProgress = false;
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.error(`_turnScreenSharingOff failed: ${error}`);
|
|
|
|
|
2017-06-20 14:52:44 +00:00
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2017-06-13 19:24:34 +00:00
|
|
|
/**
|
2017-08-07 08:23:05 +00:00
|
|
|
* Toggles between screen sharing and camera video if the toggle parameter
|
|
|
|
* is not specified and starts the procedure for obtaining new screen
|
|
|
|
* sharing/video track otherwise.
|
|
|
|
*
|
|
|
|
* @param {boolean} [toggle] - If true - new screen sharing track will be
|
|
|
|
* obtained. If false - new video track will be obtain. If not specified -
|
|
|
|
* toggles between screen sharing and camera video.
|
2017-07-09 21:34:08 +00:00
|
|
|
* @param {Object} [options] - Screen sharing options that will be passed to
|
|
|
|
* createLocalTracks.
|
|
|
|
* @param {Array<string>} [options.desktopSharingSources] - Array with the
|
|
|
|
* sources that have to be displayed in the desktop picker window ('screen',
|
|
|
|
* 'window', etc.).
|
2021-09-22 12:11:43 +00:00
|
|
|
* @param {boolean} ignoreDidHaveVideo - if true ignore if video was on when sharing started.
|
2017-06-13 19:24:34 +00:00
|
|
|
* @return {Promise.<T>}
|
|
|
|
*/
|
2021-09-22 12:11:43 +00:00
|
|
|
async toggleScreenSharing(toggle = !this._untoggleScreenSharing, options = {}, ignoreDidHaveVideo) {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug(`toggleScreenSharing: ${toggle}`);
|
2016-02-04 15:25:11 +00:00
|
|
|
if (this.videoSwitchInProgress) {
|
2017-06-13 19:24:34 +00:00
|
|
|
return Promise.reject('Switch in progress.');
|
2016-02-04 15:25:11 +00:00
|
|
|
}
|
2020-11-03 09:44:41 +00:00
|
|
|
if (!JitsiMeetJS.isDesktopSharingEnabled()) {
|
|
|
|
return Promise.reject('Cannot toggle screen sharing: not supported.');
|
2016-02-04 15:25:11 +00:00
|
|
|
}
|
|
|
|
|
2017-04-05 15:14:26 +00:00
|
|
|
if (this.isAudioOnly()) {
|
2017-06-13 19:24:34 +00:00
|
|
|
return Promise.reject('No screensharing in audio only mode');
|
2017-04-05 15:14:26 +00:00
|
|
|
}
|
|
|
|
|
2017-08-07 08:23:05 +00:00
|
|
|
if (toggle) {
|
2019-11-26 10:57:03 +00:00
|
|
|
try {
|
|
|
|
await this._switchToScreenSharing(options);
|
|
|
|
|
|
|
|
return;
|
|
|
|
} catch (err) {
|
2019-11-27 17:36:44 +00:00
|
|
|
logger.error('Failed to switch to screensharing', err);
|
2019-11-26 10:57:03 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2017-06-29 17:43:35 +00:00
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2019-08-23 20:16:52 +00:00
|
|
|
return this._untoggleScreenSharing
|
2021-09-22 12:11:43 +00:00
|
|
|
? this._untoggleScreenSharing(ignoreDidHaveVideo)
|
2019-08-23 20:16:52 +00:00
|
|
|
: Promise.resolve();
|
2017-06-29 17:43:35 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates desktop (screensharing) {@link JitsiLocalTrack}
|
2019-01-26 20:53:11 +00:00
|
|
|
*
|
2017-06-29 17:43:35 +00:00
|
|
|
* @param {Object} [options] - Screen sharing options that will be passed to
|
|
|
|
* createLocalTracks.
|
2019-01-26 20:53:11 +00:00
|
|
|
* @param {Object} [options.desktopSharing]
|
|
|
|
* @param {Object} [options.desktopStream] - An existing desktop stream to
|
|
|
|
* use instead of creating a new desktop stream.
|
2017-06-29 17:43:35 +00:00
|
|
|
* @return {Promise.<JitsiLocalTrack>} - A Promise resolved with
|
|
|
|
* {@link JitsiLocalTrack} for the screensharing or rejected with
|
|
|
|
* {@link JitsiTrackError}.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_createDesktopTrack(options = {}) {
|
2019-11-27 17:36:44 +00:00
|
|
|
const didHaveVideo = !this.isLocalVideoMuted();
|
2017-06-29 17:43:35 +00:00
|
|
|
|
2019-01-26 20:53:11 +00:00
|
|
|
const getDesktopStreamPromise = options.desktopStream
|
|
|
|
? Promise.resolve([ options.desktopStream ])
|
|
|
|
: createLocalTracksF({
|
|
|
|
desktopSharingSourceDevice: options.desktopSharingSources
|
|
|
|
? null : config._desktopSharingSourceDevice,
|
|
|
|
desktopSharingSources: options.desktopSharingSources,
|
2020-06-18 23:15:49 +00:00
|
|
|
devices: [ 'desktop' ]
|
2019-01-26 20:53:11 +00:00
|
|
|
});
|
|
|
|
|
2020-03-26 12:17:44 +00:00
|
|
|
return getDesktopStreamPromise.then(desktopStreams => {
|
2017-06-29 17:43:35 +00:00
|
|
|
// Stores the "untoggle" handler which remembers whether was
|
|
|
|
// there any video before and whether was it muted.
|
|
|
|
this._untoggleScreenSharing
|
2019-11-26 10:57:03 +00:00
|
|
|
= this._turnScreenSharingOff.bind(this, didHaveVideo);
|
2020-03-26 12:17:44 +00:00
|
|
|
|
|
|
|
const desktopVideoStream = desktopStreams.find(stream => stream.getType() === MEDIA_TYPE.VIDEO);
|
2021-04-12 07:37:39 +00:00
|
|
|
const dekstopAudioStream = desktopStreams.find(stream => stream.getType() === MEDIA_TYPE.AUDIO);
|
|
|
|
|
|
|
|
if (dekstopAudioStream) {
|
|
|
|
dekstopAudioStream.on(
|
|
|
|
JitsiTrackEvents.LOCAL_TRACK_STOPPED,
|
|
|
|
() => {
|
|
|
|
logger.debug(`Local screensharing audio track stopped. ${this.isSharingScreen}`);
|
|
|
|
|
|
|
|
// Handle case where screen share was stopped from the browsers 'screen share in progress'
|
|
|
|
// window. If audio screen sharing is stopped via the normal UX flow this point shouldn't
|
|
|
|
// be reached.
|
|
|
|
isScreenAudioShared(APP.store.getState())
|
|
|
|
&& this._untoggleScreenSharing
|
|
|
|
&& this._untoggleScreenSharing();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2020-03-26 12:17:44 +00:00
|
|
|
|
|
|
|
if (desktopVideoStream) {
|
|
|
|
desktopVideoStream.on(
|
|
|
|
JitsiTrackEvents.LOCAL_TRACK_STOPPED,
|
|
|
|
() => {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug(`Local screensharing track stopped. ${this.isSharingScreen}`);
|
|
|
|
|
2020-03-26 12:17:44 +00:00
|
|
|
// If the stream was stopped during screen sharing
|
|
|
|
// session then we should switch back to video.
|
|
|
|
this.isSharingScreen
|
|
|
|
&& this._untoggleScreenSharing
|
|
|
|
&& this._untoggleScreenSharing();
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2020-03-26 12:17:44 +00:00
|
|
|
return desktopStreams;
|
2017-06-29 17:43:35 +00:00
|
|
|
}, error => {
|
|
|
|
throw error;
|
|
|
|
});
|
|
|
|
},
|
2016-03-04 21:21:07 +00:00
|
|
|
|
2019-11-26 10:57:03 +00:00
|
|
|
/**
|
|
|
|
* Creates a new instance of presenter effect. A new video track is created
|
|
|
|
* using the new set of constraints that are calculated based on
|
|
|
|
* the height of the desktop that is being currently shared.
|
|
|
|
*
|
|
|
|
* @param {number} height - The height of the desktop stream that is being
|
|
|
|
* currently shared.
|
|
|
|
* @param {string} cameraDeviceId - The device id of the camera to be used.
|
|
|
|
* @return {Promise<JitsiStreamPresenterEffect>} - A promise resolved with
|
|
|
|
* {@link JitsiStreamPresenterEffect} if it succeeds.
|
|
|
|
*/
|
2019-12-04 20:28:42 +00:00
|
|
|
async _createPresenterStreamEffect(height = null, cameraDeviceId = null) {
|
|
|
|
if (!this.localPresenterVideo) {
|
2020-11-06 20:52:41 +00:00
|
|
|
const camera = cameraDeviceId ?? getUserSelectedCameraDeviceId(APP.store.getState());
|
|
|
|
|
2019-12-04 20:28:42 +00:00
|
|
|
try {
|
2020-11-06 20:52:41 +00:00
|
|
|
this.localPresenterVideo = await createLocalPresenterTrack({ cameraDeviceId: camera }, height);
|
2019-12-04 20:28:42 +00:00
|
|
|
} catch (err) {
|
|
|
|
logger.error('Failed to create a camera track for presenter', err);
|
2019-11-26 10:57:03 +00:00
|
|
|
|
2019-12-04 20:28:42 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
APP.store.dispatch(trackAdded(this.localPresenterVideo));
|
2019-11-26 10:57:03 +00:00
|
|
|
}
|
|
|
|
try {
|
2019-12-04 20:28:42 +00:00
|
|
|
const effect = await createPresenterEffect(this.localPresenterVideo.stream);
|
2019-11-26 10:57:03 +00:00
|
|
|
|
|
|
|
return effect;
|
|
|
|
} catch (err) {
|
|
|
|
logger.error('Failed to create the presenter effect', err);
|
2019-12-04 20:28:42 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Tries to turn the presenter video track on or off. If a presenter track
|
|
|
|
* doesn't exist, a new video track is created.
|
|
|
|
*
|
|
|
|
* @param mute - true for mute and false for unmute.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
async _mutePresenterVideo(mute) {
|
|
|
|
const maybeShowErrorDialog = error => {
|
|
|
|
APP.store.dispatch(notifyCameraError(error));
|
|
|
|
};
|
|
|
|
|
2020-01-10 21:39:58 +00:00
|
|
|
// Check for NO-OP
|
|
|
|
if (mute && (!this.localPresenterVideo || this.localPresenterVideo.isMuted())) {
|
|
|
|
|
|
|
|
return;
|
|
|
|
} else if (!mute && this.localPresenterVideo && !this.localPresenterVideo.isMuted()) {
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-02-04 21:33:57 +00:00
|
|
|
// Create a new presenter track and apply the presenter effect.
|
2019-12-04 20:28:42 +00:00
|
|
|
if (!this.localPresenterVideo && !mute) {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
|
|
|
const { height, width } = localVideo.track.getSettings() ?? localVideo.track.getConstraints();
|
2020-11-09 15:03:27 +00:00
|
|
|
const isPortrait = height >= width;
|
2020-02-04 21:33:57 +00:00
|
|
|
const DESKTOP_STREAM_CAP = 720;
|
|
|
|
|
2020-11-09 15:03:27 +00:00
|
|
|
const highResolutionTrack
|
|
|
|
= (isPortrait && width > DESKTOP_STREAM_CAP) || (!isPortrait && height > DESKTOP_STREAM_CAP);
|
|
|
|
|
|
|
|
// Resizing the desktop track for presenter is causing blurriness of the desktop share on chrome.
|
|
|
|
// Disable resizing by default, enable it only when config.js setting is enabled.
|
2021-04-22 17:41:12 +00:00
|
|
|
const resizeDesktopStream = highResolutionTrack && config.videoQuality?.resizeDesktopForPresenter;
|
2020-11-09 15:03:27 +00:00
|
|
|
|
|
|
|
if (resizeDesktopStream) {
|
|
|
|
let desktopResizeConstraints = {};
|
|
|
|
|
2020-11-06 20:52:41 +00:00
|
|
|
if (height && width) {
|
|
|
|
const advancedConstraints = [ { aspectRatio: (width / height).toPrecision(4) } ];
|
2020-11-09 15:03:27 +00:00
|
|
|
const constraint = isPortrait ? { width: DESKTOP_STREAM_CAP } : { height: DESKTOP_STREAM_CAP };
|
|
|
|
|
|
|
|
advancedConstraints.push(constraint);
|
2020-11-06 20:52:41 +00:00
|
|
|
desktopResizeConstraints.advanced = advancedConstraints;
|
|
|
|
} else {
|
|
|
|
desktopResizeConstraints = {
|
|
|
|
width: 1280,
|
|
|
|
height: 720
|
|
|
|
};
|
2020-02-04 21:33:57 +00:00
|
|
|
}
|
2020-11-06 20:52:41 +00:00
|
|
|
|
2021-03-16 15:59:33 +00:00
|
|
|
// Apply the constraints on the desktop track.
|
2019-12-09 21:48:53 +00:00
|
|
|
try {
|
2021-09-13 17:33:04 +00:00
|
|
|
await localVideo.track.applyConstraints(desktopResizeConstraints);
|
2019-12-09 21:48:53 +00:00
|
|
|
} catch (err) {
|
|
|
|
logger.error('Failed to apply constraints on the desktop stream for presenter mode', err);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2020-11-06 20:52:41 +00:00
|
|
|
const trackHeight = resizeDesktopStream
|
2021-09-13 17:33:04 +00:00
|
|
|
? localVideo.track.getSettings().height ?? DESKTOP_STREAM_CAP
|
2020-11-06 20:52:41 +00:00
|
|
|
: height;
|
2019-12-04 20:28:42 +00:00
|
|
|
let effect;
|
|
|
|
|
|
|
|
try {
|
2020-11-09 15:03:27 +00:00
|
|
|
effect = await this._createPresenterStreamEffect(trackHeight);
|
2019-12-04 20:28:42 +00:00
|
|
|
} catch (err) {
|
2020-11-09 15:03:27 +00:00
|
|
|
logger.error('Failed to unmute Presenter Video', err);
|
2019-12-04 20:28:42 +00:00
|
|
|
maybeShowErrorDialog(err);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2020-11-09 15:03:27 +00:00
|
|
|
|
|
|
|
// Replace the desktop track on the peerconnection.
|
2019-12-04 20:28:42 +00:00
|
|
|
try {
|
2021-09-13 17:33:04 +00:00
|
|
|
await localVideo.setEffect(effect);
|
2019-12-04 20:28:42 +00:00
|
|
|
APP.store.dispatch(setVideoMuted(mute, MEDIA_TYPE.PRESENTER));
|
2021-05-04 12:57:34 +00:00
|
|
|
this.setVideoMuteStatus();
|
2019-12-04 20:28:42 +00:00
|
|
|
} catch (err) {
|
|
|
|
logger.error('Failed to apply the Presenter effect', err);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
APP.store.dispatch(setVideoMuted(mute, MEDIA_TYPE.PRESENTER));
|
2019-11-26 10:57:03 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-06-29 17:43:35 +00:00
|
|
|
/**
|
2018-01-03 21:24:07 +00:00
|
|
|
* Tries to switch to the screensharing mode by disposing camera stream and
|
2017-06-29 17:43:35 +00:00
|
|
|
* replacing it with a desktop one.
|
|
|
|
*
|
|
|
|
* @param {Object} [options] - Screen sharing options that will be passed to
|
|
|
|
* createLocalTracks.
|
|
|
|
*
|
|
|
|
* @return {Promise} - A Promise resolved if the operation succeeds or
|
|
|
|
* rejected with some unknown type of error in case it fails. Promise will
|
|
|
|
* be rejected immediately if {@link videoSwitchInProgress} is true.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_switchToScreenSharing(options = {}) {
|
|
|
|
if (this.videoSwitchInProgress) {
|
|
|
|
return Promise.reject('Switch in progress.');
|
|
|
|
}
|
2017-07-10 03:42:35 +00:00
|
|
|
|
2017-06-29 17:43:35 +00:00
|
|
|
this.videoSwitchInProgress = true;
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
return this._createDesktopTrack(options)
|
2020-03-26 12:17:44 +00:00
|
|
|
.then(async streams => {
|
2021-04-12 07:37:39 +00:00
|
|
|
let desktopVideoStream = streams.find(stream => stream.getType() === MEDIA_TYPE.VIDEO);
|
|
|
|
|
|
|
|
this._desktopAudioStream = streams.find(stream => stream.getType() === MEDIA_TYPE.AUDIO);
|
|
|
|
|
|
|
|
const { audioOnly = false } = options;
|
|
|
|
|
|
|
|
// If we're in audio only mode dispose of the video track otherwise the screensharing state will be
|
|
|
|
// inconsistent.
|
|
|
|
if (audioOnly) {
|
|
|
|
desktopVideoStream.dispose();
|
|
|
|
desktopVideoStream = undefined;
|
|
|
|
|
|
|
|
if (!this._desktopAudioStream) {
|
|
|
|
return Promise.reject(AUDIO_ONLY_SCREEN_SHARE_NO_TRACK);
|
|
|
|
}
|
|
|
|
}
|
2020-03-26 12:17:44 +00:00
|
|
|
|
|
|
|
if (desktopVideoStream) {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug(`_switchToScreenSharing is using ${desktopVideoStream} for useVideoStream`);
|
2020-04-21 07:33:25 +00:00
|
|
|
await this.useVideoStream(desktopVideoStream);
|
2020-03-26 12:17:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (this._desktopAudioStream) {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
|
2020-03-26 12:17:44 +00:00
|
|
|
// If there is a localAudio stream, mix in the desktop audio stream captured by the screen sharing
|
|
|
|
// api.
|
2021-09-13 17:33:04 +00:00
|
|
|
if (localAudio) {
|
2020-03-26 12:17:44 +00:00
|
|
|
this._mixerEffect = new AudioMixerEffect(this._desktopAudioStream);
|
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
await localAudio.setEffect(this._mixerEffect);
|
2020-03-26 12:17:44 +00:00
|
|
|
} else {
|
|
|
|
// If no local stream is present ( i.e. no input audio devices) we use the screen share audio
|
|
|
|
// stream as we would use a regular stream.
|
|
|
|
await this.useAudioStream(this._desktopAudioStream);
|
2021-04-12 07:37:39 +00:00
|
|
|
|
2020-03-26 12:17:44 +00:00
|
|
|
}
|
2021-04-12 07:37:39 +00:00
|
|
|
APP.store.dispatch(setScreenAudioShareState(true));
|
2020-03-26 12:17:44 +00:00
|
|
|
}
|
|
|
|
})
|
2017-10-16 20:37:13 +00:00
|
|
|
.then(() => {
|
|
|
|
this.videoSwitchInProgress = false;
|
2020-02-05 21:18:53 +00:00
|
|
|
if (config.enableScreenshotCapture) {
|
2021-07-26 11:38:56 +00:00
|
|
|
APP.store.dispatch(toggleScreenshotCaptureSummary(true));
|
2020-02-05 21:18:53 +00:00
|
|
|
}
|
2018-01-03 21:24:07 +00:00
|
|
|
sendAnalytics(createScreenSharingEvent('started'));
|
|
|
|
logger.log('Screen sharing started');
|
2017-10-16 20:37:13 +00:00
|
|
|
})
|
|
|
|
.catch(error => {
|
|
|
|
this.videoSwitchInProgress = false;
|
2016-03-04 21:20:05 +00:00
|
|
|
|
2017-10-16 20:37:13 +00:00
|
|
|
// Pawel: With this call I'm trying to preserve the original
|
|
|
|
// behaviour although it is not clear why would we "untoggle"
|
|
|
|
// on failure. I suppose it was to restore video in case there
|
|
|
|
// was some problem during "this.useVideoStream(desktopStream)".
|
|
|
|
// It's important to note that the handler will not be available
|
|
|
|
// if we fail early on trying to get desktop media (which makes
|
|
|
|
// sense, because the camera video is still being used, so
|
|
|
|
// nothing to "untoggle").
|
|
|
|
if (this._untoggleScreenSharing) {
|
|
|
|
this._untoggleScreenSharing();
|
|
|
|
}
|
2017-08-21 08:57:58 +00:00
|
|
|
|
2017-10-16 20:37:13 +00:00
|
|
|
// FIXME the code inside of _handleScreenSharingError is
|
|
|
|
// asynchronous, but does not return a Promise and is not part
|
|
|
|
// of the current Promise chain.
|
|
|
|
this._handleScreenSharingError(error);
|
|
|
|
|
|
|
|
return Promise.reject(error);
|
|
|
|
});
|
2017-06-29 17:43:35 +00:00
|
|
|
},
|
2016-03-04 21:55:44 +00:00
|
|
|
|
2017-06-29 17:43:35 +00:00
|
|
|
/**
|
|
|
|
* Handles {@link JitsiTrackError} returned by the lib-jitsi-meet when
|
|
|
|
* trying to create screensharing track. It will either do nothing if
|
2020-06-18 23:15:49 +00:00
|
|
|
* the dialog was canceled on user's request or display an error if
|
|
|
|
* screensharing couldn't be started.
|
2017-06-29 17:43:35 +00:00
|
|
|
* @param {JitsiTrackError} error - The error returned by
|
|
|
|
* {@link _createDesktopTrack} Promise.
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_handleScreenSharingError(error) {
|
2020-06-18 23:15:49 +00:00
|
|
|
if (error.name === JitsiTrackErrors.SCREENSHARING_USER_CANCELED) {
|
2017-06-29 17:43:35 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
logger.error('failed to share local desktop', error);
|
|
|
|
|
|
|
|
// Handling:
|
2019-01-10 23:39:17 +00:00
|
|
|
// JitsiTrackErrors.CONSTRAINT_FAILED
|
2020-06-18 23:15:49 +00:00
|
|
|
// JitsiTrackErrors.PERMISSION_DENIED
|
|
|
|
// JitsiTrackErrors.SCREENSHARING_GENERIC_ERROR
|
2017-06-29 17:43:35 +00:00
|
|
|
// and any other
|
2017-11-03 19:05:03 +00:00
|
|
|
let descriptionKey;
|
|
|
|
let titleKey;
|
2017-06-29 17:43:35 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
if (error.name === JitsiTrackErrors.PERMISSION_DENIED) {
|
2020-06-18 23:15:49 +00:00
|
|
|
descriptionKey = 'dialog.screenSharingPermissionDeniedError';
|
|
|
|
titleKey = 'dialog.screenSharingFailedTitle';
|
2019-01-10 23:39:17 +00:00
|
|
|
} else if (error.name === JitsiTrackErrors.CONSTRAINT_FAILED) {
|
|
|
|
descriptionKey = 'dialog.cameraConstraintFailedError';
|
|
|
|
titleKey = 'deviceError.cameraError';
|
2020-06-18 23:15:49 +00:00
|
|
|
} else if (error.name === JitsiTrackErrors.SCREENSHARING_GENERIC_ERROR) {
|
|
|
|
descriptionKey = 'dialog.screenSharingFailed';
|
|
|
|
titleKey = 'dialog.screenSharingFailedTitle';
|
2021-04-12 07:37:39 +00:00
|
|
|
} else if (error === AUDIO_ONLY_SCREEN_SHARE_NO_TRACK) {
|
|
|
|
descriptionKey = 'notify.screenShareNoAudio';
|
|
|
|
titleKey = 'notify.screenShareNoAudioTitle';
|
2016-02-04 15:25:11 +00:00
|
|
|
}
|
2017-06-29 17:43:35 +00:00
|
|
|
|
2017-11-03 19:05:03 +00:00
|
|
|
APP.UI.messageHandler.showError({
|
|
|
|
descriptionKey,
|
|
|
|
titleKey
|
|
|
|
});
|
2016-02-04 15:25:11 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2016-01-15 14:59:35 +00:00
|
|
|
/**
|
|
|
|
* Setup interaction between conference and UI.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
_setupListeners() {
|
2016-01-06 22:39:13 +00:00
|
|
|
// add local streams when joined to the conference
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.CONFERENCE_JOINED, () => {
|
2017-11-16 22:54:49 +00:00
|
|
|
this._onConferenceJoined();
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2017-03-07 16:50:17 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.CONFERENCE_LEFT,
|
2020-01-13 17:12:25 +00:00
|
|
|
(...args) => {
|
|
|
|
APP.store.dispatch(conferenceTimestampChanged(0));
|
|
|
|
APP.store.dispatch(conferenceLeft(room, ...args));
|
|
|
|
});
|
2017-02-27 21:42:28 +00:00
|
|
|
|
2021-02-03 10:28:39 +00:00
|
|
|
room.on(
|
|
|
|
JitsiConferenceEvents.CONFERENCE_UNIQUE_ID_SET,
|
2021-06-02 09:27:15 +00:00
|
|
|
(...args) => {
|
|
|
|
// Preserve the sessionId so that the value is accessible even after room
|
|
|
|
// is disconnected.
|
|
|
|
room.sessionId = room.getMeetingUniqueId();
|
|
|
|
APP.store.dispatch(conferenceUniqueIdSet(room, ...args));
|
|
|
|
});
|
2021-02-03 10:28:39 +00:00
|
|
|
|
2016-02-25 12:32:52 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.AUTH_STATUS_CHANGED,
|
2017-05-26 16:28:14 +00:00
|
|
|
(authEnabled, authLogin) =>
|
2018-06-20 20:19:53 +00:00
|
|
|
APP.store.dispatch(authStatusChanged(authEnabled, authLogin)));
|
2016-02-25 12:32:52 +00:00
|
|
|
|
2020-11-14 04:09:25 +00:00
|
|
|
room.on(JitsiConferenceEvents.PARTCIPANT_FEATURES_CHANGED, user => {
|
|
|
|
APP.store.dispatch(updateRemoteParticipantFeatures(user));
|
|
|
|
});
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.USER_JOINED, (id, user) => {
|
2018-08-16 15:11:06 +00:00
|
|
|
// The logic shared between RN and web.
|
|
|
|
commonUserJoinedHandling(APP.store, room, user);
|
2018-07-26 16:33:40 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
if (user.isHidden()) {
|
2016-04-26 21:38:07 +00:00
|
|
|
return;
|
2017-10-12 23:02:29 +00:00
|
|
|
}
|
2016-04-26 21:38:07 +00:00
|
|
|
|
2020-11-14 04:09:25 +00:00
|
|
|
APP.store.dispatch(updateRemoteParticipantFeatures(user));
|
2021-03-16 15:59:33 +00:00
|
|
|
logger.log(`USER ${id} connected:`, user);
|
2016-09-16 20:17:00 +00:00
|
|
|
APP.UI.addUser(user);
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
2017-08-29 15:08:16 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.USER_LEFT, (id, user) => {
|
2018-08-16 15:11:06 +00:00
|
|
|
// The logic shared between RN and web.
|
|
|
|
commonUserLeftHandling(APP.store, room, user);
|
2018-07-26 16:33:40 +00:00
|
|
|
|
2018-08-16 15:11:06 +00:00
|
|
|
if (user.isHidden()) {
|
2018-02-22 19:33:10 +00:00
|
|
|
return;
|
|
|
|
}
|
2018-07-26 16:33:40 +00:00
|
|
|
|
2018-07-23 18:11:47 +00:00
|
|
|
logger.log(`USER ${id} LEFT:`, user);
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.USER_STATUS_CHANGED, (id, status) => {
|
2017-07-31 23:33:22 +00:00
|
|
|
APP.store.dispatch(participantPresenceChanged(id, status));
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
const user = room.getParticipantById(id);
|
|
|
|
|
2017-05-19 15:12:24 +00:00
|
|
|
if (user) {
|
|
|
|
APP.UI.updateUserStatus(user, status);
|
|
|
|
}
|
|
|
|
});
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.USER_ROLE_CHANGED, (id, role) => {
|
2016-01-06 22:39:13 +00:00
|
|
|
if (this.isLocalId(id)) {
|
2016-11-11 15:00:54 +00:00
|
|
|
logger.info(`My role changed, new role: ${role}`);
|
2017-04-10 21:53:30 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(localParticipantRoleChanged(role));
|
2020-05-05 14:03:54 +00:00
|
|
|
APP.API.notifyUserRoleChanged(id, role);
|
2016-01-06 22:39:13 +00:00
|
|
|
} else {
|
2017-04-10 21:53:30 +00:00
|
|
|
APP.store.dispatch(participantRoleChanged(id, role));
|
2016-01-06 22:39:13 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
room.on(JitsiConferenceEvents.TRACK_ADDED, track => {
|
|
|
|
if (!track || track.isLocal()) {
|
2016-01-06 22:39:13 +00:00
|
|
|
return;
|
2017-10-12 23:02:29 +00:00
|
|
|
}
|
2016-01-29 22:06:54 +00:00
|
|
|
|
2017-06-20 16:30:54 +00:00
|
|
|
APP.store.dispatch(trackAdded(track));
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
room.on(JitsiConferenceEvents.TRACK_REMOVED, track => {
|
|
|
|
if (!track || track.isLocal()) {
|
2016-02-23 22:47:55 +00:00
|
|
|
return;
|
2017-10-12 23:02:29 +00:00
|
|
|
}
|
2016-02-23 22:47:55 +00:00
|
|
|
|
2017-06-20 16:30:54 +00:00
|
|
|
APP.store.dispatch(trackRemoved(track));
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.TRACK_AUDIO_LEVEL_CHANGED, (id, lvl) => {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
2017-10-12 23:02:29 +00:00
|
|
|
let newLvl = lvl;
|
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
if (this.isLocalId(id) && localAudio?.isMuted()) {
|
2017-10-12 23:02:29 +00:00
|
|
|
newLvl = 0;
|
2016-01-06 22:39:13 +00:00
|
|
|
}
|
2016-01-19 19:32:29 +00:00
|
|
|
|
2017-08-18 11:30:30 +00:00
|
|
|
if (config.debug) {
|
2017-10-12 23:02:29 +00:00
|
|
|
this.audioLevelsMap[id] = newLvl;
|
|
|
|
if (config.debugAudioLevels) {
|
|
|
|
logger.log(`AudioLevel:${id}/${newLvl}`);
|
|
|
|
}
|
2016-02-09 22:49:46 +00:00
|
|
|
}
|
2016-01-19 19:32:29 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
APP.UI.setAudioLevel(id, newLvl);
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2019-07-02 11:59:25 +00:00
|
|
|
room.on(JitsiConferenceEvents.TRACK_MUTE_CHANGED, (track, participantThatMutedUs) => {
|
2019-06-17 14:00:09 +00:00
|
|
|
if (participantThatMutedUs) {
|
2021-02-24 21:45:07 +00:00
|
|
|
APP.store.dispatch(participantMutedUs(participantThatMutedUs, track));
|
|
|
|
if (this.isSharingScreen && track.isVideoTrack()) {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug('TRACK_MUTE_CHANGED while screen sharing');
|
2021-02-24 21:45:07 +00:00
|
|
|
this._turnScreenSharingOff(false);
|
|
|
|
}
|
2019-06-17 14:00:09 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-02-17 14:40:24 +00:00
|
|
|
room.on(JitsiConferenceEvents.SUBJECT_CHANGED,
|
2019-03-12 17:45:53 +00:00
|
|
|
subject => APP.store.dispatch(conferenceSubjectChanged(subject)));
|
2016-09-09 01:19:18 +00:00
|
|
|
|
2016-01-26 21:26:28 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.LAST_N_ENDPOINTS_CHANGED,
|
2017-10-02 23:08:07 +00:00
|
|
|
(leavingIds, enteringIds) =>
|
|
|
|
APP.UI.handleLastNEndpoints(leavingIds, enteringIds));
|
2017-03-21 17:14:13 +00:00
|
|
|
|
2017-08-09 19:40:03 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.P2P_STATUS,
|
2017-10-12 23:02:29 +00:00
|
|
|
(jitsiConference, p2p) =>
|
|
|
|
APP.store.dispatch(p2pStatusChanged(p2p)));
|
2017-08-09 19:40:03 +00:00
|
|
|
|
2016-09-16 20:51:19 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.PARTICIPANT_CONN_STATUS_CHANGED,
|
2018-05-21 21:06:02 +00:00
|
|
|
(id, connectionStatus) => APP.store.dispatch(
|
|
|
|
participantConnectionStatusChanged(id, connectionStatus)));
|
2017-07-13 00:26:10 +00:00
|
|
|
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
room.on(
|
|
|
|
JitsiConferenceEvents.DOMINANT_SPEAKER_CHANGED,
|
2021-08-18 22:34:01 +00:00
|
|
|
(dominant, previous) => APP.store.dispatch(dominantSpeakerChanged(dominant, previous, room)));
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2020-01-13 17:12:25 +00:00
|
|
|
room.on(
|
|
|
|
JitsiConferenceEvents.CONFERENCE_CREATED_TIMESTAMP,
|
|
|
|
conferenceTimestamp => APP.store.dispatch(conferenceTimestampChanged(conferenceTimestamp)));
|
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.CONNECTION_INTERRUPTED, () => {
|
2017-07-13 00:26:10 +00:00
|
|
|
APP.store.dispatch(localParticipantConnectionStatusChanged(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiParticipantConnectionStatus.INTERRUPTED));
|
2016-07-22 18:42:41 +00:00
|
|
|
});
|
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.CONNECTION_RESTORED, () => {
|
2017-07-13 00:26:10 +00:00
|
|
|
APP.store.dispatch(localParticipantConnectionStatusChanged(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiParticipantConnectionStatus.ACTIVE));
|
2016-07-22 18:42:41 +00:00
|
|
|
});
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
room.on(
|
|
|
|
JitsiConferenceEvents.DISPLAY_NAME_CHANGED,
|
|
|
|
(id, displayName) => {
|
|
|
|
const formattedDisplayName
|
2019-01-15 11:28:07 +00:00
|
|
|
= getNormalizedDisplayName(displayName);
|
2017-10-12 23:02:29 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(participantUpdated({
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
conference: room,
|
2017-10-12 23:02:29 +00:00
|
|
|
id,
|
|
|
|
name: formattedDisplayName
|
|
|
|
}));
|
2017-12-05 03:27:17 +00:00
|
|
|
APP.API.notifyDisplayNameChanged(id, {
|
|
|
|
displayName: formattedDisplayName,
|
|
|
|
formattedDisplayName:
|
|
|
|
appendSuffix(
|
|
|
|
formattedDisplayName
|
|
|
|
|| interfaceConfig.DEFAULT_REMOTE_DISPLAY_NAME)
|
|
|
|
});
|
2017-10-12 23:02:29 +00:00
|
|
|
}
|
|
|
|
);
|
2018-06-26 22:56:22 +00:00
|
|
|
room.on(
|
|
|
|
JitsiConferenceEvents.BOT_TYPE_CHANGED,
|
|
|
|
(id, botType) => {
|
|
|
|
|
|
|
|
APP.store.dispatch(participantUpdated({
|
|
|
|
conference: room,
|
|
|
|
id,
|
|
|
|
botType
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
);
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2018-07-17 17:31:12 +00:00
|
|
|
room.on(
|
|
|
|
JitsiConferenceEvents.ENDPOINT_MESSAGE_RECEIVED,
|
2020-03-20 11:51:26 +00:00
|
|
|
(...args) => {
|
|
|
|
APP.store.dispatch(endpointMessageReceived(...args));
|
|
|
|
if (args && args.length >= 2) {
|
|
|
|
const [ sender, eventData ] = args;
|
|
|
|
|
|
|
|
if (eventData.name === ENDPOINT_TEXT_MESSAGE_NAME) {
|
|
|
|
APP.API.notifyEndpointTextMessageReceived({
|
|
|
|
senderInfo: {
|
|
|
|
jid: sender._jid,
|
|
|
|
id: sender._id
|
|
|
|
},
|
|
|
|
eventData
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2018-07-17 17:31:12 +00:00
|
|
|
|
2017-04-18 21:49:52 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.LOCK_STATE_CHANGED,
|
2017-04-18 21:49:52 +00:00
|
|
|
(...args) => APP.store.dispatch(lockStateChanged(room, ...args)));
|
2017-04-13 00:23:43 +00:00
|
|
|
|
2021-06-11 08:58:45 +00:00
|
|
|
room.on(JitsiConferenceEvents.KICKED, (participant, reason, isReplaced) => {
|
|
|
|
if (isReplaced) {
|
|
|
|
// this event triggers when the local participant is kicked, `participant`
|
|
|
|
// is the kicker. In replace participant case, kicker is undefined,
|
|
|
|
// as the server initiated it. We mark in store the local participant
|
|
|
|
// as being replaced based on jwt.
|
|
|
|
const localParticipant = getLocalParticipant(APP.store.getState());
|
|
|
|
|
|
|
|
APP.store.dispatch(participantUpdated({
|
|
|
|
conference: room,
|
|
|
|
id: localParticipant.id,
|
|
|
|
isReplaced
|
|
|
|
}));
|
2021-06-24 11:33:58 +00:00
|
|
|
|
|
|
|
// we send readyToClose when kicked participant is replace so that
|
|
|
|
// embedding app can choose to dispose the iframe API on the handler.
|
|
|
|
APP.API.notifyReadyToClose();
|
2021-06-11 08:58:45 +00:00
|
|
|
}
|
2019-06-26 21:53:48 +00:00
|
|
|
APP.store.dispatch(kickedOut(room, participant));
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2019-06-17 14:00:09 +00:00
|
|
|
room.on(JitsiConferenceEvents.PARTICIPANT_KICKED, (kicker, kicked) => {
|
|
|
|
APP.store.dispatch(participantKicked(kicker, kicked));
|
|
|
|
});
|
|
|
|
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.SUSPEND_DETECTED, () => {
|
2017-01-31 20:58:48 +00:00
|
|
|
APP.store.dispatch(suspendDetected());
|
2016-11-09 22:34:01 +00:00
|
|
|
});
|
|
|
|
|
2017-07-24 13:56:57 +00:00
|
|
|
APP.UI.addListener(UIEvents.AUDIO_MUTED, muted => {
|
2017-07-25 10:05:08 +00:00
|
|
|
this.muteAudio(muted);
|
2017-07-24 13:56:57 +00:00
|
|
|
});
|
2017-04-05 15:14:26 +00:00
|
|
|
APP.UI.addListener(UIEvents.VIDEO_MUTED, muted => {
|
2018-04-10 22:03:10 +00:00
|
|
|
this.muteVideo(muted);
|
2017-04-05 15:14:26 +00:00
|
|
|
});
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
room.addCommandListener(this.commands.defaults.ETHERPAD,
|
|
|
|
({ value }) => {
|
|
|
|
APP.UI.initEtherpad(value);
|
|
|
|
}
|
|
|
|
);
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2018-06-18 09:19:07 +00:00
|
|
|
APP.UI.addListener(UIEvents.EMAIL_CHANGED,
|
|
|
|
this.changeLocalEmail.bind(this));
|
2016-06-13 21:11:44 +00:00
|
|
|
room.addCommandListener(this.commands.defaults.EMAIL, (data, from) => {
|
2017-03-23 18:01:33 +00:00
|
|
|
APP.store.dispatch(participantUpdated({
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
conference: room,
|
2017-03-23 18:01:33 +00:00
|
|
|
id: from,
|
|
|
|
email: data.value
|
|
|
|
}));
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2016-10-27 20:49:29 +00:00
|
|
|
room.addCommandListener(
|
|
|
|
this.commands.defaults.AVATAR_URL,
|
|
|
|
(data, from) => {
|
2017-02-27 21:42:28 +00:00
|
|
|
APP.store.dispatch(
|
2017-03-23 18:01:33 +00:00
|
|
|
participantUpdated({
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
conference: room,
|
2017-03-23 18:01:33 +00:00
|
|
|
id: from,
|
|
|
|
avatarURL: data.value
|
|
|
|
}));
|
2017-10-02 23:08:07 +00:00
|
|
|
});
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-03-23 17:45:51 +00:00
|
|
|
APP.UI.addListener(UIEvents.NICKNAME_CHANGED,
|
|
|
|
this.changeLocalDisplayName.bind(this));
|
2016-06-13 21:11:44 +00:00
|
|
|
|
2015-12-31 15:23:23 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.START_MUTED_POLICY_CHANGED,
|
2016-03-02 15:39:39 +00:00
|
|
|
({ audio, video }) => {
|
2017-11-21 02:21:35 +00:00
|
|
|
APP.store.dispatch(
|
|
|
|
onStartMutedPolicyChanged(audio, video));
|
2016-01-06 22:39:13 +00:00
|
|
|
}
|
|
|
|
);
|
2017-10-10 23:31:40 +00:00
|
|
|
room.on(JitsiConferenceEvents.STARTED_MUTED, () => {
|
2021-03-05 15:18:34 +00:00
|
|
|
const audioMuted = room.isStartAudioMuted();
|
|
|
|
const videoMuted = room.isStartVideoMuted();
|
|
|
|
const localTracks = getLocalTracks(APP.store.getState()['features/base/tracks']);
|
|
|
|
const promises = [];
|
|
|
|
|
|
|
|
APP.store.dispatch(setAudioMuted(audioMuted));
|
|
|
|
APP.store.dispatch(setVideoMuted(videoMuted));
|
|
|
|
|
|
|
|
// Remove the tracks from the peerconnection.
|
|
|
|
for (const track of localTracks) {
|
2021-08-31 15:06:31 +00:00
|
|
|
// Always add the track on mobile Safari because of a known issue where audio playout doesn't happen
|
|
|
|
// if the user joins audio and video muted.
|
|
|
|
if (audioMuted && track.jitsiTrack?.getType() === MEDIA_TYPE.AUDIO && !isIosMobileBrowser()) {
|
2021-03-05 15:18:34 +00:00
|
|
|
promises.push(this.useAudioStream(null));
|
|
|
|
}
|
|
|
|
if (videoMuted && track.jitsiTrack?.getType() === MEDIA_TYPE.VIDEO) {
|
|
|
|
promises.push(this.useVideoStream(null));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Promise.allSettled(promises)
|
|
|
|
.then(() => APP.UI.notifyInitiallyMuted());
|
2016-01-13 03:38:58 +00:00
|
|
|
});
|
2016-01-06 22:39:13 +00:00
|
|
|
|
2017-08-09 19:40:03 +00:00
|
|
|
room.on(
|
2017-10-10 23:31:40 +00:00
|
|
|
JitsiConferenceEvents.DATA_CHANNEL_OPENED, () => {
|
2017-08-09 19:40:03 +00:00
|
|
|
APP.store.dispatch(dataChannelOpened());
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2016-01-06 22:39:13 +00:00
|
|
|
// call hangup
|
|
|
|
APP.UI.addListener(UIEvents.HANGUP, () => {
|
2016-10-05 21:33:09 +00:00
|
|
|
this.hangup(true);
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
// logout
|
|
|
|
APP.UI.addListener(UIEvents.LOGOUT, () => {
|
2016-10-05 21:33:09 +00:00
|
|
|
AuthHandler.logout(room).then(url => {
|
2016-02-25 13:52:15 +00:00
|
|
|
if (url) {
|
2017-02-05 03:51:41 +00:00
|
|
|
UIUtil.redirect(url);
|
2016-02-25 13:52:15 +00:00
|
|
|
} else {
|
2016-10-05 21:33:09 +00:00
|
|
|
this.hangup(true);
|
2016-02-25 13:52:15 +00:00
|
|
|
}
|
|
|
|
});
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
APP.UI.addListener(UIEvents.AUTH_CLICKED, () => {
|
2021-04-22 15:05:14 +00:00
|
|
|
AuthHandler.authenticate(room);
|
2016-01-06 22:39:13 +00:00
|
|
|
});
|
|
|
|
|
2016-02-09 10:19:43 +00:00
|
|
|
APP.UI.addListener(
|
|
|
|
UIEvents.VIDEO_DEVICE_CHANGED,
|
2017-10-12 23:02:29 +00:00
|
|
|
cameraDeviceId => {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
2017-10-17 18:20:33 +00:00
|
|
|
const videoWasMuted = this.isLocalVideoMuted();
|
|
|
|
|
2018-01-03 21:24:07 +00:00
|
|
|
sendAnalytics(createDeviceChangedEvent('video', 'input'));
|
2017-10-17 18:47:39 +00:00
|
|
|
|
2019-11-26 10:57:03 +00:00
|
|
|
// If both screenshare and video are in progress, restart the
|
|
|
|
// presenter mode with the new camera device.
|
|
|
|
if (this.isSharingScreen && !videoWasMuted) {
|
2021-09-13 17:33:04 +00:00
|
|
|
const { height } = localVideo.track.getSettings();
|
2019-11-26 10:57:03 +00:00
|
|
|
|
|
|
|
// dispose the existing presenter track and create a new
|
|
|
|
// camera track.
|
2020-01-13 14:21:31 +00:00
|
|
|
// FIXME JitsiLocalTrack.dispose is async and should be waited for
|
2020-01-10 21:54:45 +00:00
|
|
|
this.localPresenterVideo && this.localPresenterVideo.dispose();
|
2019-12-04 20:28:42 +00:00
|
|
|
this.localPresenterVideo = null;
|
2019-11-26 10:57:03 +00:00
|
|
|
|
|
|
|
return this._createPresenterStreamEffect(height, cameraDeviceId)
|
2021-09-13 17:33:04 +00:00
|
|
|
.then(effect => localVideo.setEffect(effect))
|
2019-11-26 10:57:03 +00:00
|
|
|
.then(() => {
|
2021-05-04 12:57:34 +00:00
|
|
|
this.setVideoMuteStatus();
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.log('Switched local video device while screen sharing and the video is unmuted');
|
2019-11-26 10:57:03 +00:00
|
|
|
this._updateVideoDeviceId();
|
|
|
|
})
|
|
|
|
.catch(err => APP.store.dispatch(notifyCameraError(err)));
|
|
|
|
|
2019-12-04 20:28:42 +00:00
|
|
|
// If screenshare is in progress but video is muted, update the default device
|
|
|
|
// id for video, dispose the existing presenter track and create a new effect
|
|
|
|
// that can be applied on un-mute.
|
2019-11-26 10:57:03 +00:00
|
|
|
} else if (this.isSharingScreen && videoWasMuted) {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.log('Switched local video device: while screen sharing and the video is muted');
|
2021-09-13 17:33:04 +00:00
|
|
|
const { height } = localVideo.track.getSettings();
|
2019-12-04 20:28:42 +00:00
|
|
|
|
2019-05-02 09:43:47 +00:00
|
|
|
this._updateVideoDeviceId();
|
2020-01-13 14:21:31 +00:00
|
|
|
|
|
|
|
// FIXME JitsiLocalTrack.dispose is async and should be waited for
|
2020-01-10 21:54:45 +00:00
|
|
|
this.localPresenterVideo && this.localPresenterVideo.dispose();
|
2019-12-04 20:28:42 +00:00
|
|
|
this.localPresenterVideo = null;
|
|
|
|
this._createPresenterStreamEffect(height, cameraDeviceId);
|
2019-11-26 10:57:03 +00:00
|
|
|
|
|
|
|
// if there is only video, switch to the new camera stream.
|
|
|
|
} else {
|
|
|
|
createLocalTracksF({
|
|
|
|
devices: [ 'video' ],
|
|
|
|
cameraDeviceId,
|
|
|
|
micDeviceId: null
|
|
|
|
})
|
|
|
|
.then(([ stream ]) => {
|
|
|
|
// if we are in audio only mode or video was muted before
|
|
|
|
// changing device, then mute
|
|
|
|
if (this.isAudioOnly() || videoWasMuted) {
|
|
|
|
return stream.mute()
|
|
|
|
.then(() => stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
return stream;
|
|
|
|
})
|
2021-03-05 18:17:39 +00:00
|
|
|
.then(stream => {
|
|
|
|
logger.log('Switching the local video device.');
|
|
|
|
|
|
|
|
return this.useVideoStream(stream);
|
|
|
|
})
|
2019-11-26 10:57:03 +00:00
|
|
|
.then(() => {
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.log('Switched local video device.');
|
2019-11-26 10:57:03 +00:00
|
|
|
this._updateVideoDeviceId();
|
|
|
|
})
|
2021-03-05 18:17:39 +00:00
|
|
|
.catch(error => {
|
|
|
|
logger.error(`Switching the local video device failed: ${error}`);
|
|
|
|
|
|
|
|
return APP.store.dispatch(notifyCameraError(error));
|
|
|
|
});
|
2019-11-26 10:57:03 +00:00
|
|
|
}
|
2016-02-09 10:19:43 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
APP.UI.addListener(
|
|
|
|
UIEvents.AUDIO_DEVICE_CHANGED,
|
2017-10-12 23:02:29 +00:00
|
|
|
micDeviceId => {
|
2017-10-17 18:20:33 +00:00
|
|
|
const audioWasMuted = this.isLocalAudioMuted();
|
|
|
|
|
2020-06-03 21:49:08 +00:00
|
|
|
// When the 'default' mic needs to be selected, we need to
|
|
|
|
// pass the real device id to gUM instead of 'default' in order
|
|
|
|
// to get the correct MediaStreamTrack from chrome because of the
|
|
|
|
// following bug.
|
|
|
|
// https://bugs.chromium.org/p/chromium/issues/detail?id=997689
|
|
|
|
const hasDefaultMicChanged = micDeviceId === 'default';
|
|
|
|
|
2018-01-03 21:24:07 +00:00
|
|
|
sendAnalytics(createDeviceChangedEvent('audio', 'input'));
|
2017-08-14 13:25:37 +00:00
|
|
|
createLocalTracksF({
|
2017-10-12 23:02:29 +00:00
|
|
|
devices: [ 'audio' ],
|
2016-06-24 09:47:13 +00:00
|
|
|
cameraDeviceId: null,
|
2020-06-03 21:49:08 +00:00
|
|
|
micDeviceId: hasDefaultMicChanged
|
|
|
|
? getDefaultDeviceId(APP.store.getState(), 'audioInput')
|
|
|
|
: micDeviceId
|
2016-06-24 09:47:13 +00:00
|
|
|
})
|
2017-10-12 23:02:29 +00:00
|
|
|
.then(([ stream ]) => {
|
2017-10-17 18:20:33 +00:00
|
|
|
// if audio was muted before changing the device, mute
|
|
|
|
// with the new device
|
|
|
|
if (audioWasMuted) {
|
|
|
|
return stream.mute()
|
|
|
|
.then(() => stream);
|
|
|
|
}
|
|
|
|
|
|
|
|
return stream;
|
|
|
|
})
|
2020-03-26 12:17:44 +00:00
|
|
|
.then(async stream => {
|
|
|
|
// In case screen sharing audio is also shared we mix it with new input stream. The old _mixerEffect
|
|
|
|
// will be cleaned up when the existing track is replaced.
|
|
|
|
if (this._mixerEffect) {
|
|
|
|
this._mixerEffect = new AudioMixerEffect(this._desktopAudioStream);
|
|
|
|
|
|
|
|
await stream.setEffect(this._mixerEffect);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.useAudioStream(stream);
|
|
|
|
})
|
2019-05-03 17:25:33 +00:00
|
|
|
.then(() => {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
|
|
|
|
if (localAudio && hasDefaultMicChanged) {
|
2020-06-03 21:49:08 +00:00
|
|
|
// workaround for the default device to be shown as selected in the
|
|
|
|
// settings even when the real device id was passed to gUM because of the
|
|
|
|
// above mentioned chrome bug.
|
2021-09-13 17:33:04 +00:00
|
|
|
localAudio._realDeviceId = localAudio.deviceId = 'default';
|
2020-06-03 21:49:08 +00:00
|
|
|
}
|
2021-09-13 17:33:04 +00:00
|
|
|
logger.log(`switched local audio device: ${localAudio?.getDeviceId()}`);
|
2019-05-02 09:43:47 +00:00
|
|
|
|
|
|
|
this._updateAudioDeviceId();
|
2016-06-24 09:47:13 +00:00
|
|
|
})
|
2017-10-12 23:02:29 +00:00
|
|
|
.catch(err => {
|
2019-05-29 21:17:07 +00:00
|
|
|
APP.store.dispatch(notifyMicError(err));
|
2016-06-24 09:47:13 +00:00
|
|
|
});
|
2016-02-09 10:19:43 +00:00
|
|
|
}
|
|
|
|
);
|
|
|
|
|
2017-04-05 15:14:26 +00:00
|
|
|
APP.UI.addListener(UIEvents.TOGGLE_AUDIO_ONLY, audioOnly => {
|
2017-07-24 15:36:19 +00:00
|
|
|
|
|
|
|
// FIXME On web video track is stored both in redux and in
|
|
|
|
// 'localVideo' field, video is attempted to be unmuted twice when
|
|
|
|
// turning off the audio only mode. This will crash the app with
|
|
|
|
// 'unmute operation is already in progress'.
|
|
|
|
// Because there's no logic in redux about creating new track in
|
|
|
|
// case unmute when not track exists the things have to go through
|
|
|
|
// muteVideo logic in such case.
|
|
|
|
const tracks = APP.store.getState()['features/base/tracks'];
|
|
|
|
const isTrackInRedux
|
|
|
|
= Boolean(
|
|
|
|
tracks.find(
|
|
|
|
track => track.jitsiTrack
|
|
|
|
&& track.jitsiTrack.getType() === 'video'));
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-07-24 15:36:19 +00:00
|
|
|
if (!isTrackInRedux) {
|
|
|
|
this.muteVideo(audioOnly);
|
|
|
|
}
|
2017-04-05 15:14:26 +00:00
|
|
|
|
|
|
|
// Immediately update the UI by having remote videos and the large
|
|
|
|
// video update themselves instead of waiting for some other event
|
|
|
|
// to cause the update, usually PARTICIPANT_CONN_STATUS_CHANGED.
|
|
|
|
// There is no guarantee another event will trigger the update
|
|
|
|
// immediately and in all situations, for example because a remote
|
|
|
|
// participant is having connection trouble so no status changes.
|
2021-01-21 20:46:47 +00:00
|
|
|
const displayedUserId = APP.UI.getLargeVideoID();
|
|
|
|
|
|
|
|
if (displayedUserId) {
|
|
|
|
APP.UI.updateLargeVideo(displayedUserId, true);
|
|
|
|
}
|
2017-04-05 15:14:26 +00:00
|
|
|
});
|
|
|
|
|
2016-02-04 15:25:11 +00:00
|
|
|
APP.UI.addListener(
|
2021-09-22 12:11:43 +00:00
|
|
|
UIEvents.TOGGLE_SCREENSHARING, ({ enabled, audioOnly, ignoreDidHaveVideo }) => {
|
|
|
|
this.toggleScreenSharing(enabled, { audioOnly }, ignoreDidHaveVideo);
|
2021-04-12 07:37:39 +00:00
|
|
|
}
|
2016-01-06 22:39:13 +00:00
|
|
|
);
|
2016-04-25 20:39:31 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2019-07-12 13:08:34 +00:00
|
|
|
/**
|
|
|
|
* Cleanups local conference on suspend.
|
|
|
|
*/
|
|
|
|
onSuspendDetected() {
|
|
|
|
// After wake up, we will be in a state where conference is left
|
|
|
|
// there will be dialog shown to user.
|
|
|
|
// We do not want video/audio as we show an overlay and after it
|
|
|
|
// user need to rejoin or close, while waking up we can detect
|
|
|
|
// camera wakeup as a problem with device.
|
|
|
|
// We also do not care about device change, which happens
|
|
|
|
// on resume after suspending PC.
|
|
|
|
if (this.deviceChangeListener) {
|
|
|
|
JitsiMeetJS.mediaDevices.removeEventListener(
|
|
|
|
JitsiMediaDevicesEvents.DEVICE_LIST_CHANGED,
|
|
|
|
this.deviceChangeListener);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-16 22:54:49 +00:00
|
|
|
/**
|
|
|
|
* Callback invoked when the conference has been successfully joined.
|
|
|
|
* Initializes the UI and various other features.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onConferenceJoined() {
|
|
|
|
APP.UI.initConference();
|
|
|
|
|
2021-01-13 10:10:27 +00:00
|
|
|
if (!config.disableShortcuts) {
|
|
|
|
APP.keyboardshortcut.init();
|
|
|
|
}
|
2017-11-16 22:54:49 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(conferenceJoined(room));
|
|
|
|
},
|
|
|
|
|
2016-06-13 11:49:00 +00:00
|
|
|
/**
|
2019-05-07 08:53:01 +00:00
|
|
|
* Updates the list of current devices.
|
|
|
|
* @param {boolean} setDeviceListChangeHandler - Whether to add the deviceList change handlers.
|
2016-06-13 11:49:00 +00:00
|
|
|
* @private
|
2018-08-06 15:24:59 +00:00
|
|
|
* @returns {Promise}
|
2016-06-13 11:49:00 +00:00
|
|
|
*/
|
2019-05-07 08:53:01 +00:00
|
|
|
_initDeviceList(setDeviceListChangeHandler = false) {
|
2018-07-12 03:57:44 +00:00
|
|
|
const { mediaDevices } = JitsiMeetJS;
|
|
|
|
|
|
|
|
if (mediaDevices.isDeviceListAvailable()
|
|
|
|
&& mediaDevices.isDeviceChangeAvailable()) {
|
2019-05-07 08:53:01 +00:00
|
|
|
if (setDeviceListChangeHandler) {
|
|
|
|
this.deviceChangeListener = devices =>
|
|
|
|
window.setTimeout(() => this._onDeviceListChanged(devices), 0);
|
|
|
|
mediaDevices.addEventListener(
|
|
|
|
JitsiMediaDevicesEvents.DEVICE_LIST_CHANGED,
|
|
|
|
this.deviceChangeListener);
|
|
|
|
}
|
2018-08-06 15:24:59 +00:00
|
|
|
|
|
|
|
const { dispatch } = APP.store;
|
|
|
|
|
|
|
|
return dispatch(getAvailableDevices())
|
|
|
|
.then(devices => {
|
|
|
|
// Ugly way to synchronize real device IDs with local
|
|
|
|
// storage and settings menu. This is a workaround until
|
|
|
|
// getConstraints() method will be implemented in browsers.
|
2019-05-02 09:43:47 +00:00
|
|
|
this._updateAudioDeviceId();
|
2018-08-06 15:24:59 +00:00
|
|
|
|
2019-05-02 09:43:47 +00:00
|
|
|
this._updateVideoDeviceId();
|
2018-08-06 15:24:59 +00:00
|
|
|
|
|
|
|
APP.UI.onAvailableDevicesChanged(devices);
|
|
|
|
});
|
core: refactor routing
Unfortunately, as the Jitsi Meet development evolved the routing mechanism
became more complex and thre logic ended up spread across multiple parts of the
codebase, which made it hard to follow and extend.
This change aims to fix that by rewriting the routing logic and centralizing it
in (pretty much) a single place, with no implicit inter-dependencies.
In order to arrive there, however, some extra changes were needed, which were
not caught early enough and are thus part of this change:
- JitsiMeetJS initialization is now synchronous: there is nothing async about
it, and the only async requirement (Temasys support) was lifted. See [0].
- WebRTC support can be detected early: building on top of the above, WebRTC
support can now be detected immediately, so take advantage of this to simplify
how we handle unsupported browsers. See [0].
The new router takes decissions based on the Redux state at the time of
invocation. A route can be represented by either a component or a URl reference,
with the latter taking precedence. On mobile, obviously, there is no concept of
URL reference so routing is based solely on components.
[0]: https://github.com/jitsi/lib-jitsi-meet/pull/779
2018-06-29 07:58:31 +00:00
|
|
|
}
|
2018-08-06 15:24:59 +00:00
|
|
|
|
|
|
|
return Promise.resolve();
|
2016-06-13 11:49:00 +00:00
|
|
|
},
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2019-05-02 09:43:47 +00:00
|
|
|
/**
|
|
|
|
* Updates the settings for the currently used video device, extracting
|
|
|
|
* the device id from the used track.
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_updateVideoDeviceId() {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
|
|
|
|
|
|
|
if (localVideo && localVideo.videoType === 'camera') {
|
2019-05-02 09:43:47 +00:00
|
|
|
APP.store.dispatch(updateSettings({
|
2021-09-13 17:33:04 +00:00
|
|
|
cameraDeviceId: localVideo.getDeviceId()
|
2019-05-02 09:43:47 +00:00
|
|
|
}));
|
|
|
|
}
|
2019-12-04 20:28:42 +00:00
|
|
|
|
|
|
|
// If screenshare is in progress, get the device id from the presenter track.
|
|
|
|
if (this.localPresenterVideo) {
|
|
|
|
APP.store.dispatch(updateSettings({
|
|
|
|
cameraDeviceId: this.localPresenterVideo.getDeviceId()
|
|
|
|
}));
|
|
|
|
}
|
2019-05-02 09:43:47 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates the settings for the currently used audio device, extracting
|
|
|
|
* the device id from the used track.
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
_updateAudioDeviceId() {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
|
|
|
|
if (localAudio) {
|
2019-05-02 09:43:47 +00:00
|
|
|
APP.store.dispatch(updateSettings({
|
2021-09-13 17:33:04 +00:00
|
|
|
micDeviceId: localAudio.getDeviceId()
|
2019-05-02 09:43:47 +00:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2016-06-13 11:49:00 +00:00
|
|
|
/**
|
|
|
|
* Event listener for JitsiMediaDevicesEvents.DEVICE_LIST_CHANGED to
|
|
|
|
* handle change of available media devices.
|
|
|
|
* @private
|
|
|
|
* @param {MediaDeviceInfo[]} devices
|
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
|
|
|
_onDeviceListChanged(devices) {
|
2019-05-03 17:25:33 +00:00
|
|
|
const oldDevices = APP.store.getState()['features/base/devices'].availableDevices;
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
2019-05-03 17:25:33 +00:00
|
|
|
|
2018-07-13 17:31:28 +00:00
|
|
|
APP.store.dispatch(updateDeviceList(devices));
|
2016-06-13 11:49:00 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
const newDevices
|
|
|
|
= mediaDeviceHelper.getNewMediaDevicesAfterDeviceListChanged(
|
2017-08-18 11:30:30 +00:00
|
|
|
devices,
|
|
|
|
this.isSharingScreen,
|
2021-09-13 17:33:04 +00:00
|
|
|
localVideo,
|
|
|
|
localAudio);
|
2017-10-12 23:02:29 +00:00
|
|
|
const promises = [];
|
|
|
|
const audioWasMuted = this.isLocalAudioMuted();
|
|
|
|
const videoWasMuted = this.isLocalVideoMuted();
|
2019-04-12 16:10:38 +00:00
|
|
|
const requestedInput = {
|
|
|
|
audio: Boolean(newDevices.audioinput),
|
|
|
|
video: Boolean(newDevices.videoinput)
|
|
|
|
};
|
2016-06-13 11:49:00 +00:00
|
|
|
|
|
|
|
if (typeof newDevices.audiooutput !== 'undefined') {
|
2018-07-13 16:21:26 +00:00
|
|
|
const { dispatch } = APP.store;
|
|
|
|
const setAudioOutputPromise
|
|
|
|
= setAudioOutputDeviceId(newDevices.audiooutput, dispatch)
|
|
|
|
.catch(); // Just ignore any errors in catch block.
|
|
|
|
|
|
|
|
|
|
|
|
promises.push(setAudioOutputPromise);
|
2016-06-13 11:49:00 +00:00
|
|
|
}
|
|
|
|
|
2019-04-12 16:10:38 +00:00
|
|
|
// Handles the use case when the default device is changed (we are always stopping the streams because it's
|
|
|
|
// simpler):
|
|
|
|
// If the default device is changed we need to first stop the local streams and then call GUM. Otherwise GUM
|
|
|
|
// will return a stream using the old default device.
|
2021-09-13 17:33:04 +00:00
|
|
|
if (requestedInput.audio && localAudio) {
|
|
|
|
localAudio.stopStream();
|
2019-04-12 16:10:38 +00:00
|
|
|
}
|
|
|
|
|
2021-09-13 17:33:04 +00:00
|
|
|
if (requestedInput.video && localVideo) {
|
|
|
|
localVideo.stopStream();
|
2019-04-12 16:10:38 +00:00
|
|
|
}
|
|
|
|
|
2019-05-03 17:25:33 +00:00
|
|
|
// Let's handle unknown/non-preferred devices
|
|
|
|
const newAvailDevices
|
|
|
|
= APP.store.getState()['features/base/devices'].availableDevices;
|
2019-05-20 20:35:42 +00:00
|
|
|
let newAudioDevices = [];
|
|
|
|
let oldAudioDevices = [];
|
2019-05-03 17:25:33 +00:00
|
|
|
|
|
|
|
if (typeof newDevices.audiooutput === 'undefined') {
|
2019-05-20 20:35:42 +00:00
|
|
|
newAudioDevices = newAvailDevices.audioOutput;
|
|
|
|
oldAudioDevices = oldDevices.audioOutput;
|
2019-05-03 17:25:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!requestedInput.audio) {
|
2019-05-20 20:35:42 +00:00
|
|
|
newAudioDevices = newAudioDevices.concat(newAvailDevices.audioInput);
|
|
|
|
oldAudioDevices = oldAudioDevices.concat(oldDevices.audioInput);
|
|
|
|
}
|
|
|
|
|
|
|
|
// check for audio
|
|
|
|
if (newAudioDevices.length > 0) {
|
2019-05-03 17:25:33 +00:00
|
|
|
APP.store.dispatch(
|
2019-05-20 20:35:42 +00:00
|
|
|
checkAndNotifyForNewDevice(newAudioDevices, oldAudioDevices));
|
2019-05-03 17:25:33 +00:00
|
|
|
}
|
|
|
|
|
2019-05-20 20:35:42 +00:00
|
|
|
// check for video
|
2019-05-03 17:25:33 +00:00
|
|
|
if (!requestedInput.video) {
|
|
|
|
APP.store.dispatch(
|
|
|
|
checkAndNotifyForNewDevice(newAvailDevices.videoInput, oldDevices.videoInput));
|
|
|
|
}
|
|
|
|
|
2020-06-03 21:49:08 +00:00
|
|
|
// When the 'default' mic needs to be selected, we need to
|
|
|
|
// pass the real device id to gUM instead of 'default' in order
|
|
|
|
// to get the correct MediaStreamTrack from chrome because of the
|
|
|
|
// following bug.
|
|
|
|
// https://bugs.chromium.org/p/chromium/issues/detail?id=997689
|
|
|
|
const hasDefaultMicChanged = newDevices.audioinput === 'default';
|
|
|
|
|
2020-10-22 19:48:17 +00:00
|
|
|
// This is the case when the local video is muted and a preferred device is connected.
|
|
|
|
if (requestedInput.video && this.isLocalVideoMuted()) {
|
|
|
|
// We want to avoid creating a new video track in order to prevent turning on the camera.
|
|
|
|
requestedInput.video = false;
|
|
|
|
APP.store.dispatch(updateSettings({ // Update the current selected camera for the device selection dialog.
|
|
|
|
cameraDeviceId: newDevices.videoinput
|
|
|
|
}));
|
|
|
|
delete newDevices.videoinput;
|
|
|
|
|
|
|
|
// Removing the current video track in order to force the unmute to select the preferred device.
|
2021-03-05 18:17:39 +00:00
|
|
|
logger.debug('_onDeviceListChanged: Removing the current video track.');
|
2020-10-22 19:48:17 +00:00
|
|
|
this.useVideoStream(null);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2016-06-13 11:49:00 +00:00
|
|
|
promises.push(
|
|
|
|
mediaDeviceHelper.createLocalTracksAfterDeviceListChanged(
|
2017-08-14 13:25:37 +00:00
|
|
|
createLocalTracksF,
|
2016-06-13 11:49:00 +00:00
|
|
|
newDevices.videoinput,
|
2020-06-03 21:49:08 +00:00
|
|
|
hasDefaultMicChanged
|
|
|
|
? getDefaultDeviceId(APP.store.getState(), 'audioInput')
|
|
|
|
: newDevices.audioinput)
|
2018-09-04 15:29:50 +00:00
|
|
|
.then(tracks => {
|
|
|
|
// If audio or video muted before, or we unplugged current
|
|
|
|
// device and selected new one, then mute new track.
|
|
|
|
const muteSyncPromises = tracks.map(track => {
|
|
|
|
if ((track.isVideoTrack() && videoWasMuted)
|
|
|
|
|| (track.isAudioTrack() && audioWasMuted)) {
|
|
|
|
return track.mute();
|
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.resolve();
|
|
|
|
});
|
|
|
|
|
|
|
|
return Promise.all(muteSyncPromises)
|
2019-04-12 16:10:38 +00:00
|
|
|
.then(() =>
|
|
|
|
Promise.all(Object.keys(requestedInput).map(mediaType => {
|
|
|
|
if (requestedInput[mediaType]) {
|
|
|
|
const useStream
|
|
|
|
= mediaType === 'audio'
|
|
|
|
? this.useAudioStream.bind(this)
|
|
|
|
: this.useVideoStream.bind(this);
|
2021-09-13 17:33:04 +00:00
|
|
|
const track = tracks.find(t => t.getType() === mediaType) || null;
|
2019-04-12 16:10:38 +00:00
|
|
|
|
|
|
|
// Use the new stream or null if we failed to obtain it.
|
2021-09-13 17:33:04 +00:00
|
|
|
return useStream(track)
|
2019-05-01 14:13:25 +00:00
|
|
|
.then(() => {
|
2021-09-13 17:33:04 +00:00
|
|
|
if (track?.isAudioTrack() && hasDefaultMicChanged) {
|
2020-06-03 21:49:08 +00:00
|
|
|
// workaround for the default device to be shown as selected in the
|
|
|
|
// settings even when the real device id was passed to gUM because of
|
|
|
|
// the above mentioned chrome bug.
|
2021-09-13 17:33:04 +00:00
|
|
|
track._realDeviceId = track.deviceId = 'default';
|
2020-06-03 21:49:08 +00:00
|
|
|
}
|
2019-05-02 09:43:47 +00:00
|
|
|
mediaType === 'audio'
|
|
|
|
? this._updateAudioDeviceId()
|
|
|
|
: this._updateVideoDeviceId();
|
2019-05-01 14:13:25 +00:00
|
|
|
});
|
2019-04-12 16:10:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return Promise.resolve();
|
|
|
|
})));
|
2018-09-04 15:29:50 +00:00
|
|
|
})
|
2016-06-13 11:49:00 +00:00
|
|
|
.then(() => {
|
2018-09-04 15:29:50 +00:00
|
|
|
// Log and sync known mute state.
|
2017-10-17 18:02:41 +00:00
|
|
|
if (audioWasMuted) {
|
2018-01-03 21:24:07 +00:00
|
|
|
sendAnalytics(createTrackMutedEvent(
|
|
|
|
'audio',
|
|
|
|
'device list changed'));
|
2017-10-09 21:40:38 +00:00
|
|
|
logger.log('Audio mute: device list changed');
|
2016-06-13 11:49:00 +00:00
|
|
|
muteLocalAudio(true);
|
|
|
|
}
|
|
|
|
|
2017-10-17 18:02:41 +00:00
|
|
|
if (!this.isSharingScreen && videoWasMuted) {
|
2018-01-03 21:24:07 +00:00
|
|
|
sendAnalytics(createTrackMutedEvent(
|
|
|
|
'video',
|
|
|
|
'device list changed'));
|
2017-10-09 21:40:38 +00:00
|
|
|
logger.log('Video mute: device list changed');
|
2016-06-13 11:49:00 +00:00
|
|
|
muteLocalVideo(true);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
|
|
|
|
return Promise.all(promises)
|
|
|
|
.then(() => {
|
|
|
|
APP.UI.onAvailableDevicesChanged(devices);
|
|
|
|
});
|
2016-06-20 21:13:17 +00:00
|
|
|
},
|
2017-07-24 13:56:57 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Determines whether or not the audio button should be enabled.
|
|
|
|
*/
|
|
|
|
updateAudioIconEnabled() {
|
2021-09-13 17:33:04 +00:00
|
|
|
const localAudio = getLocalJitsiAudioTrack(APP.store.getState());
|
|
|
|
const audioMediaDevices = APP.store.getState()['features/base/devices'].availableDevices.audioInput;
|
|
|
|
const audioDeviceCount = audioMediaDevices ? audioMediaDevices.length : 0;
|
2017-07-24 13:56:57 +00:00
|
|
|
|
|
|
|
// The audio functionality is considered available if there are any
|
|
|
|
// audio devices detected or if the local audio stream already exists.
|
2021-09-13 17:33:04 +00:00
|
|
|
const available = audioDeviceCount > 0 || Boolean(localAudio);
|
2017-07-24 13:56:57 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(setAudioAvailable(available));
|
2017-08-04 08:15:11 +00:00
|
|
|
APP.API.notifyAudioAvailabilityChanged(available);
|
2017-07-24 13:56:57 +00:00
|
|
|
},
|
|
|
|
|
2017-06-28 14:15:46 +00:00
|
|
|
/**
|
|
|
|
* Determines whether or not the video button should be enabled.
|
|
|
|
*/
|
|
|
|
updateVideoIconEnabled() {
|
|
|
|
const videoMediaDevices
|
2019-03-28 16:29:30 +00:00
|
|
|
= APP.store.getState()['features/base/devices'].availableDevices.videoInput;
|
2017-06-28 14:15:46 +00:00
|
|
|
const videoDeviceCount
|
|
|
|
= videoMediaDevices ? videoMediaDevices.length : 0;
|
2021-09-13 17:33:04 +00:00
|
|
|
const localVideo = getLocalJitsiVideoTrack(APP.store.getState());
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-06-28 14:15:46 +00:00
|
|
|
// The video functionality is considered available if there are any
|
|
|
|
// video devices detected or if there is local video stream already
|
|
|
|
// active which could be either screensharing stream or a video track
|
|
|
|
// created before the permissions were rejected (through browser
|
|
|
|
// config).
|
2021-09-13 17:33:04 +00:00
|
|
|
const available = videoDeviceCount > 0 || Boolean(localVideo);
|
2017-06-28 14:15:46 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(setVideoAvailable(available));
|
2017-08-04 08:15:11 +00:00
|
|
|
APP.API.notifyVideoAvailabilityChanged(available);
|
2017-06-28 14:15:46 +00:00
|
|
|
},
|
2016-06-20 21:13:17 +00:00
|
|
|
|
2016-10-05 21:33:09 +00:00
|
|
|
/**
|
|
|
|
* Disconnect from the conference and optionally request user feedback.
|
|
|
|
* @param {boolean} [requestFeedback=false] if user feedback should be
|
|
|
|
* requested
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
hangup(requestFeedback = false) {
|
2020-11-14 04:09:25 +00:00
|
|
|
APP.store.dispatch(disableReceiver());
|
2019-01-01 21:19:34 +00:00
|
|
|
|
2019-01-26 20:53:11 +00:00
|
|
|
this._stopProxyConnection();
|
|
|
|
|
2019-01-01 21:19:34 +00:00
|
|
|
APP.store.dispatch(destroyLocalTracks());
|
|
|
|
this._localTracksInitialized = false;
|
2017-06-05 18:19:25 +00:00
|
|
|
|
2018-05-01 19:42:08 +00:00
|
|
|
// Remove unnecessary event listeners from firing callbacks.
|
2018-07-09 18:46:26 +00:00
|
|
|
if (this.deviceChangeListener) {
|
|
|
|
JitsiMeetJS.mediaDevices.removeEventListener(
|
|
|
|
JitsiMediaDevicesEvents.DEVICE_LIST_CHANGED,
|
|
|
|
this.deviceChangeListener);
|
|
|
|
}
|
2018-05-01 19:42:08 +00:00
|
|
|
|
2019-01-01 21:19:34 +00:00
|
|
|
APP.UI.removeAllListeners();
|
|
|
|
|
2017-08-07 16:20:44 +00:00
|
|
|
let requestFeedbackPromise;
|
|
|
|
|
|
|
|
if (requestFeedback) {
|
|
|
|
requestFeedbackPromise
|
|
|
|
= APP.store.dispatch(maybeOpenFeedbackDialog(room))
|
2017-10-12 23:02:29 +00:00
|
|
|
|
2017-08-07 16:20:44 +00:00
|
|
|
// false because the thank you dialog shouldn't be displayed
|
|
|
|
.catch(() => Promise.resolve(false));
|
|
|
|
} else {
|
|
|
|
requestFeedbackPromise = Promise.resolve(true);
|
|
|
|
}
|
|
|
|
|
2021-06-02 09:27:15 +00:00
|
|
|
Promise.all([
|
|
|
|
requestFeedbackPromise,
|
|
|
|
this.leaveRoomAndDisconnect()
|
|
|
|
])
|
|
|
|
.then(values => {
|
2019-01-18 00:04:35 +00:00
|
|
|
this._room = undefined;
|
|
|
|
room = undefined;
|
|
|
|
|
2020-08-04 10:46:13 +00:00
|
|
|
/**
|
|
|
|
* Don't call {@code notifyReadyToClose} if the promotional page flag is set
|
|
|
|
* and let the page take care of sending the message, since there will be
|
|
|
|
* a redirect to the page regardlessly.
|
|
|
|
*/
|
|
|
|
if (!interfaceConfig.SHOW_PROMOTIONAL_CLOSE_PAGE) {
|
|
|
|
APP.API.notifyReadyToClose();
|
|
|
|
}
|
2021-06-02 09:27:15 +00:00
|
|
|
APP.store.dispatch(maybeRedirectToWelcomePage(values[0]));
|
2016-10-12 21:30:44 +00:00
|
|
|
});
|
2017-01-12 21:53:17 +00:00
|
|
|
},
|
|
|
|
|
2018-08-01 20:37:15 +00:00
|
|
|
/**
|
|
|
|
* Leaves the room and calls JitsiConnection.disconnect.
|
|
|
|
*
|
|
|
|
* @returns {Promise}
|
|
|
|
*/
|
|
|
|
leaveRoomAndDisconnect() {
|
|
|
|
APP.store.dispatch(conferenceWillLeave(room));
|
|
|
|
|
2020-04-29 12:52:44 +00:00
|
|
|
if (room && room.isJoined()) {
|
2019-08-13 11:59:41 +00:00
|
|
|
return room.leave().then(disconnect, disconnect);
|
|
|
|
}
|
|
|
|
|
|
|
|
return disconnect();
|
2018-08-01 20:37:15 +00:00
|
|
|
},
|
|
|
|
|
2017-01-12 21:53:17 +00:00
|
|
|
/**
|
|
|
|
* Changes the email for the local user
|
|
|
|
* @param email {string} the new email
|
|
|
|
*/
|
|
|
|
changeLocalEmail(email = '') {
|
2017-10-06 17:52:23 +00:00
|
|
|
const localParticipant = getLocalParticipant(APP.store.getState());
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
const formattedEmail = String(email).trim();
|
2017-01-12 21:53:17 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
if (formattedEmail === localParticipant.email) {
|
2017-01-12 21:53:17 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-03-23 18:01:33 +00:00
|
|
|
|
2017-10-06 17:52:23 +00:00
|
|
|
const localId = localParticipant.id;
|
2017-03-23 18:01:33 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(participantUpdated({
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
// XXX Only the local participant is allowed to update without
|
|
|
|
// stating the JitsiConference instance (i.e. participant property
|
|
|
|
// `conference` for a remote participant) because the local
|
|
|
|
// participant is uniquely identified by the very fact that there is
|
|
|
|
// only one local participant.
|
|
|
|
|
2017-03-23 18:01:33 +00:00
|
|
|
id: localId,
|
|
|
|
local: true,
|
2017-12-19 23:11:54 +00:00
|
|
|
email: formattedEmail
|
2017-03-23 18:01:33 +00:00
|
|
|
}));
|
2017-01-12 21:53:17 +00:00
|
|
|
|
2018-04-12 19:58:20 +00:00
|
|
|
APP.store.dispatch(updateSettings({
|
|
|
|
email: formattedEmail
|
|
|
|
}));
|
2018-06-18 09:19:07 +00:00
|
|
|
APP.API.notifyEmailChanged(localId, {
|
|
|
|
email: formattedEmail
|
|
|
|
});
|
2017-10-12 23:02:29 +00:00
|
|
|
sendData(commands.EMAIL, formattedEmail);
|
2017-01-12 21:53:17 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Changes the avatar url for the local user
|
|
|
|
* @param url {string} the new url
|
|
|
|
*/
|
|
|
|
changeLocalAvatarUrl(url = '') {
|
2017-10-06 17:52:23 +00:00
|
|
|
const { avatarURL, id } = getLocalParticipant(APP.store.getState());
|
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
const formattedUrl = String(url).trim();
|
2017-01-12 21:53:17 +00:00
|
|
|
|
2017-10-12 23:02:29 +00:00
|
|
|
if (formattedUrl === avatarURL) {
|
2017-01-12 21:53:17 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-03-23 18:01:33 +00:00
|
|
|
|
|
|
|
APP.store.dispatch(participantUpdated({
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
// XXX Only the local participant is allowed to update without
|
|
|
|
// stating the JitsiConference instance (i.e. participant property
|
|
|
|
// `conference` for a remote participant) because the local
|
|
|
|
// participant is uniquely identified by the very fact that there is
|
|
|
|
// only one local participant.
|
|
|
|
|
2017-10-06 17:52:23 +00:00
|
|
|
id,
|
2017-03-23 18:01:33 +00:00
|
|
|
local: true,
|
2017-10-12 23:02:29 +00:00
|
|
|
avatarURL: formattedUrl
|
2017-03-23 18:01:33 +00:00
|
|
|
}));
|
2017-01-12 21:53:17 +00:00
|
|
|
|
2018-04-12 19:58:20 +00:00
|
|
|
APP.store.dispatch(updateSettings({
|
|
|
|
avatarURL: formattedUrl
|
|
|
|
}));
|
2017-01-12 21:53:17 +00:00
|
|
|
sendData(commands.AVATAR_URL, url);
|
2016-12-09 23:15:04 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends a message via the data channel.
|
2017-01-18 00:16:18 +00:00
|
|
|
* @param {string} to the id of the endpoint that should receive the
|
|
|
|
* message. If "" - the message will be sent to all participants.
|
|
|
|
* @param {object} payload the payload of the message.
|
2016-12-09 23:15:04 +00:00
|
|
|
* @throws NetworkError or InvalidStateError or Error if the operation
|
|
|
|
* fails.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
sendEndpointMessage(to, payload) {
|
2016-12-09 23:15:04 +00:00
|
|
|
room.sendEndpointMessage(to, payload);
|
2017-01-23 18:07:08 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds new listener.
|
|
|
|
* @param {String} eventName the name of the event
|
|
|
|
* @param {Function} listener the listener.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
addListener(eventName, listener) {
|
2017-01-23 18:07:08 +00:00
|
|
|
eventEmitter.addListener(eventName, listener);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Removes listener.
|
|
|
|
* @param {String} eventName the name of the event that triggers the
|
|
|
|
* listener
|
|
|
|
* @param {Function} listener the listener.
|
|
|
|
*/
|
2017-04-11 19:40:03 +00:00
|
|
|
removeListener(eventName, listener) {
|
2017-01-23 18:07:08 +00:00
|
|
|
eventEmitter.removeListener(eventName, listener);
|
2017-03-21 17:14:13 +00:00
|
|
|
},
|
|
|
|
|
2017-03-23 17:45:51 +00:00
|
|
|
/**
|
|
|
|
* Changes the display name for the local user
|
|
|
|
* @param nickname {string} the new display name
|
|
|
|
*/
|
|
|
|
changeLocalDisplayName(nickname = '') {
|
2019-01-15 11:28:07 +00:00
|
|
|
const formattedNickname = getNormalizedDisplayName(nickname);
|
2017-10-06 17:52:23 +00:00
|
|
|
const { id, name } = getLocalParticipant(APP.store.getState());
|
2017-03-23 17:45:51 +00:00
|
|
|
|
2017-10-06 17:52:23 +00:00
|
|
|
if (formattedNickname === name) {
|
2017-03-23 17:45:51 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2017-05-03 23:57:52 +00:00
|
|
|
APP.store.dispatch(participantUpdated({
|
Associate remote participant w/ JitsiConference (_UPDATED)
The commit message of "Associate remote participant w/ JitsiConference
(_JOINED)" explains the motivation for this commit.
Practically, _JOINED and _LEFT combined with "Remove remote participants
who are no longer of interest" should alleviate the problem with
multiplying remote participants to an acceptable level of annoyance.
Technically though, a remote participant cannot be identified by an ID
only. The ID is (somewhat) "unique" in the context of a single
JitsiConference instance. So in order to not have to scratch our heads
over an obscure corner, racing case, it's better to always identify
remote participants by the pair id-conference. Unfortunately, that's a
bit of a high order given the existing source code. So I've implemented
the cases which are the easiest so that new source code written with
participantUpdated is more likely to identify a remote participant with
the pair id-conference.
Additionally, the commit "Reduce direct read access to the
features/base/participants redux state" brings more control back to the
functions of the feature base/participants so that one day we can (if we
choose to) do something like, for example:
If getParticipants is called with a conference, it returns the
participants from features/base/participants who are associated with the
specified conference. If no conference is specified in the function
call, then default to the conference which is the primary focus of the
app at the time of the function call. Added to the above, this should
allow us to further reduce the cases in which we're identifying remote
participants by id only and get us even closer to a more "predictable"
behavior in corner, racing cases.
2018-05-22 21:47:43 +00:00
|
|
|
// XXX Only the local participant is allowed to update without
|
|
|
|
// stating the JitsiConference instance (i.e. participant property
|
|
|
|
// `conference` for a remote participant) because the local
|
|
|
|
// participant is uniquely identified by the very fact that there is
|
|
|
|
// only one local participant.
|
|
|
|
|
2017-10-06 17:52:23 +00:00
|
|
|
id,
|
2017-05-03 23:57:52 +00:00
|
|
|
local: true,
|
|
|
|
name: formattedNickname
|
|
|
|
}));
|
|
|
|
|
2018-04-12 19:58:20 +00:00
|
|
|
APP.store.dispatch(updateSettings({
|
|
|
|
displayName: formattedNickname
|
|
|
|
}));
|
2017-06-22 21:28:57 +00:00
|
|
|
},
|
|
|
|
|
2019-01-26 20:53:11 +00:00
|
|
|
/**
|
|
|
|
* Callback invoked by the external api create or update a direct connection
|
|
|
|
* from the local client to an external client.
|
|
|
|
*
|
|
|
|
* @param {Object} event - The object containing information that should be
|
|
|
|
* passed to the {@code ProxyConnectionService}.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
onProxyConnectionEvent(event) {
|
|
|
|
if (!this._proxyConnection) {
|
|
|
|
this._proxyConnection = new JitsiMeetJS.ProxyConnectionService({
|
2019-03-13 18:15:56 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Pass the {@code JitsiConnection} instance which will be used
|
|
|
|
* to fetch TURN credentials.
|
|
|
|
*/
|
|
|
|
jitsiConnection: APP.connection,
|
|
|
|
|
2019-01-26 20:53:11 +00:00
|
|
|
/**
|
|
|
|
* The proxy connection feature is currently tailored towards
|
|
|
|
* taking a proxied video stream and showing it as a local
|
|
|
|
* desktop screen.
|
|
|
|
*/
|
|
|
|
convertVideoToDesktop: true,
|
|
|
|
|
2019-07-29 22:21:53 +00:00
|
|
|
/**
|
|
|
|
* Callback invoked when the connection has been closed
|
|
|
|
* automatically. Triggers cleanup of screensharing if active.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
onConnectionClosed: () => {
|
|
|
|
if (this._untoggleScreenSharing) {
|
|
|
|
this._untoggleScreenSharing();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2019-01-26 20:53:11 +00:00
|
|
|
/**
|
|
|
|
* Callback invoked to pass messages from the local client back
|
|
|
|
* out to the external client.
|
|
|
|
*
|
|
|
|
* @param {string} peerJid - The jid of the intended recipient
|
|
|
|
* of the message.
|
|
|
|
* @param {Object} data - The message that should be sent. For
|
|
|
|
* screensharing this is an iq.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
onSendMessage: (peerJid, data) =>
|
|
|
|
APP.API.sendProxyConnectionEvent({
|
|
|
|
data,
|
|
|
|
to: peerJid
|
|
|
|
}),
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Callback invoked when the remote peer of the proxy connection
|
|
|
|
* has provided a video stream, intended to be used as a local
|
|
|
|
* desktop stream.
|
|
|
|
*
|
|
|
|
* @param {JitsiLocalTrack} remoteProxyStream - The media
|
|
|
|
* stream to use as a local desktop stream.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
onRemoteStream: desktopStream => {
|
|
|
|
if (desktopStream.videoType !== 'desktop') {
|
|
|
|
logger.warn('Received a non-desktop stream to proxy.');
|
|
|
|
desktopStream.dispose();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.toggleScreenSharing(undefined, { desktopStream });
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
this._proxyConnection.processMessage(event);
|
|
|
|
},
|
|
|
|
|
2017-08-04 08:15:11 +00:00
|
|
|
/**
|
|
|
|
* Sets the video muted status.
|
|
|
|
*/
|
2021-05-04 12:57:34 +00:00
|
|
|
setVideoMuteStatus() {
|
2020-10-23 22:48:56 +00:00
|
|
|
APP.UI.setVideoMuted(this.getMyUserId());
|
2017-08-04 08:15:11 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sets the audio muted status.
|
|
|
|
*
|
|
|
|
* @param {boolean} muted - New muted status.
|
|
|
|
*/
|
|
|
|
setAudioMuteStatus(muted) {
|
2017-08-18 11:30:30 +00:00
|
|
|
APP.UI.setAudioMuted(this.getMyUserId(), muted);
|
|
|
|
APP.API.notifyAudioMutedStatusChanged(muted);
|
2018-01-19 22:19:55 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dispatches the passed in feedback for submission. The submitted score
|
|
|
|
* should be a number inclusively between 1 through 5, or -1 for no score.
|
|
|
|
*
|
|
|
|
* @param {number} score - a number between 1 and 5 (inclusive) or -1 for no
|
|
|
|
* score.
|
|
|
|
* @param {string} message - An optional message to attach to the feedback
|
|
|
|
* in addition to the score.
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
submitFeedback(score = -1, message = '') {
|
|
|
|
if (score === -1 || (score >= 1 && score <= 5)) {
|
|
|
|
APP.store.dispatch(submitFeedback(score, message, room));
|
|
|
|
}
|
2019-01-26 20:53:11 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Terminates any proxy screensharing connection that is active.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_stopProxyConnection() {
|
|
|
|
if (this._proxyConnection) {
|
|
|
|
this._proxyConnection.stop();
|
|
|
|
}
|
|
|
|
|
|
|
|
this._proxyConnection = null;
|
2017-08-18 11:30:30 +00:00
|
|
|
}
|
2016-07-08 01:44:04 +00:00
|
|
|
};
|