feat(presenter): add Presenter Mode
- Adds the ability to share video as a "PiP" when screenshare is in progress. - Add a method for creating a local presenter track. - Make sure isLocalVideoTrackMuted returns the correct mute state when only screenshare is present. - Make sure we get the updated window size of the window being shared before painting it on the canvas. - Make sure we check if the shared window has been resized
This commit is contained in:
parent
db6a2673de
commit
0a64bf2068
244
conference.js
244
conference.js
|
@ -93,10 +93,15 @@ import {
|
||||||
participantRoleChanged,
|
participantRoleChanged,
|
||||||
participantUpdated
|
participantUpdated
|
||||||
} from './react/features/base/participants';
|
} from './react/features/base/participants';
|
||||||
import { updateSettings } from './react/features/base/settings';
|
|
||||||
import {
|
import {
|
||||||
|
getUserSelectedCameraDeviceId,
|
||||||
|
updateSettings
|
||||||
|
} from './react/features/base/settings';
|
||||||
|
import {
|
||||||
|
createLocalPresenterTrack,
|
||||||
createLocalTracksF,
|
createLocalTracksF,
|
||||||
destroyLocalTracks,
|
destroyLocalTracks,
|
||||||
|
isLocalVideoTrackMuted,
|
||||||
isLocalTrackMuted,
|
isLocalTrackMuted,
|
||||||
isUserInteractionRequiredForUnmute,
|
isUserInteractionRequiredForUnmute,
|
||||||
replaceLocalTrack,
|
replaceLocalTrack,
|
||||||
|
@ -113,6 +118,7 @@ import {
|
||||||
import { mediaPermissionPromptVisibilityChanged } from './react/features/overlay';
|
import { mediaPermissionPromptVisibilityChanged } from './react/features/overlay';
|
||||||
import { suspendDetected } from './react/features/power-monitor';
|
import { suspendDetected } from './react/features/power-monitor';
|
||||||
import { setSharedVideoStatus } from './react/features/shared-video';
|
import { setSharedVideoStatus } from './react/features/shared-video';
|
||||||
|
import { createPresenterEffect } from './react/features/stream-effects/presenter';
|
||||||
import { endpointMessageReceived } from './react/features/subtitles';
|
import { endpointMessageReceived } from './react/features/subtitles';
|
||||||
|
|
||||||
const logger = require('jitsi-meet-logger').getLogger(__filename);
|
const logger = require('jitsi-meet-logger').getLogger(__filename);
|
||||||
|
@ -437,6 +443,11 @@ export default {
|
||||||
*/
|
*/
|
||||||
localAudio: null,
|
localAudio: null,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The local presenter video track (if any).
|
||||||
|
*/
|
||||||
|
localPresenterVideo: null,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The local video track (if any).
|
* The local video track (if any).
|
||||||
* FIXME tracks from redux store should be the single source of truth, but
|
* FIXME tracks from redux store should be the single source of truth, but
|
||||||
|
@ -722,9 +733,8 @@ export default {
|
||||||
isLocalVideoMuted() {
|
isLocalVideoMuted() {
|
||||||
// If the tracks are not ready, read from base/media state
|
// If the tracks are not ready, read from base/media state
|
||||||
return this._localTracksInitialized
|
return this._localTracksInitialized
|
||||||
? isLocalTrackMuted(
|
? isLocalVideoTrackMuted(
|
||||||
APP.store.getState()['features/base/tracks'],
|
APP.store.getState()['features/base/tracks'])
|
||||||
MEDIA_TYPE.VIDEO)
|
|
||||||
: isVideoMutedByUser(APP.store);
|
: isVideoMutedByUser(APP.store);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -798,6 +808,55 @@ export default {
|
||||||
this.muteAudio(!this.isLocalAudioMuted(), showUI);
|
this.muteAudio(!this.isLocalAudioMuted(), showUI);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
async mutePresenterVideo(mute, showUI = true) {
|
||||||
|
const maybeShowErrorDialog = error => {
|
||||||
|
showUI && APP.store.dispatch(notifyCameraError(error));
|
||||||
|
};
|
||||||
|
|
||||||
|
if (mute) {
|
||||||
|
try {
|
||||||
|
await this.localVideo.setEffect(undefined);
|
||||||
|
APP.store.dispatch(
|
||||||
|
setVideoMuted(mute, MEDIA_TYPE.PRESENTER));
|
||||||
|
this._untoggleScreenSharing
|
||||||
|
= this._turnScreenSharingOff.bind(this, false);
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to mute the Presenter video');
|
||||||
|
}
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const { height } = this.localVideo.track.getSettings();
|
||||||
|
const defaultCamera
|
||||||
|
= getUserSelectedCameraDeviceId(APP.store.getState());
|
||||||
|
let effect;
|
||||||
|
|
||||||
|
try {
|
||||||
|
effect = await this._createPresenterStreamEffect(height,
|
||||||
|
defaultCamera);
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to unmute Presenter Video');
|
||||||
|
maybeShowErrorDialog(err);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
await this.localVideo.setEffect(effect);
|
||||||
|
APP.store.dispatch(setVideoMuted(mute, MEDIA_TYPE.PRESENTER));
|
||||||
|
this._untoggleScreenSharing
|
||||||
|
= this._turnScreenSharingOff.bind(this, true);
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to apply the Presenter effect', err);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Simulates toolbar button click for video mute. Used by shortcuts and API.
|
* Simulates toolbar button click for video mute. Used by shortcuts and API.
|
||||||
* @param mute true for mute and false for unmute.
|
* @param mute true for mute and false for unmute.
|
||||||
|
@ -812,6 +871,10 @@ export default {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this.isSharingScreen) {
|
||||||
|
return this.mutePresenterVideo(mute);
|
||||||
|
}
|
||||||
|
|
||||||
// If not ready to modify track's state yet adjust the base/media
|
// If not ready to modify track's state yet adjust the base/media
|
||||||
if (!this._localTracksInitialized) {
|
if (!this._localTracksInitialized) {
|
||||||
// This will only modify base/media.video.muted which is then synced
|
// This will only modify base/media.video.muted which is then synced
|
||||||
|
@ -1351,7 +1414,7 @@ export default {
|
||||||
* in case it fails.
|
* in case it fails.
|
||||||
* @private
|
* @private
|
||||||
*/
|
*/
|
||||||
_turnScreenSharingOff(didHaveVideo, wasVideoMuted) {
|
_turnScreenSharingOff(didHaveVideo) {
|
||||||
this._untoggleScreenSharing = null;
|
this._untoggleScreenSharing = null;
|
||||||
this.videoSwitchInProgress = true;
|
this.videoSwitchInProgress = true;
|
||||||
const { receiver } = APP.remoteControl;
|
const { receiver } = APP.remoteControl;
|
||||||
|
@ -1369,13 +1432,7 @@ export default {
|
||||||
.then(([ stream ]) => this.useVideoStream(stream))
|
.then(([ stream ]) => this.useVideoStream(stream))
|
||||||
.then(() => {
|
.then(() => {
|
||||||
sendAnalytics(createScreenSharingEvent('stopped'));
|
sendAnalytics(createScreenSharingEvent('stopped'));
|
||||||
logger.log('Screen sharing stopped, switching to video.');
|
logger.log('Screen sharing stopped.');
|
||||||
|
|
||||||
if (!this.localVideo && wasVideoMuted) {
|
|
||||||
return Promise.reject('No local video to be muted!');
|
|
||||||
} else if (wasVideoMuted && this.localVideo) {
|
|
||||||
return this.localVideo.mute();
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.catch(error => {
|
.catch(error => {
|
||||||
logger.error('failed to switch back to local video', error);
|
logger.error('failed to switch back to local video', error);
|
||||||
|
@ -1390,6 +1447,16 @@ export default {
|
||||||
promise = this.useVideoStream(null);
|
promise = this.useVideoStream(null);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// mute the presenter track if it exists.
|
||||||
|
if (this.localPresenterVideo) {
|
||||||
|
APP.store.dispatch(
|
||||||
|
setVideoMuted(true, MEDIA_TYPE.PRESENTER));
|
||||||
|
this.localPresenterVideo.dispose();
|
||||||
|
APP.store.dispatch(
|
||||||
|
trackRemoved(this.localPresenterVideo));
|
||||||
|
this.localPresenterVideo = null;
|
||||||
|
}
|
||||||
|
|
||||||
return promise.then(
|
return promise.then(
|
||||||
() => {
|
() => {
|
||||||
this.videoSwitchInProgress = false;
|
this.videoSwitchInProgress = false;
|
||||||
|
@ -1415,7 +1482,7 @@ export default {
|
||||||
* 'window', etc.).
|
* 'window', etc.).
|
||||||
* @return {Promise.<T>}
|
* @return {Promise.<T>}
|
||||||
*/
|
*/
|
||||||
toggleScreenSharing(toggle = !this._untoggleScreenSharing, options = {}) {
|
async toggleScreenSharing(toggle = !this._untoggleScreenSharing, options = {}) {
|
||||||
if (this.videoSwitchInProgress) {
|
if (this.videoSwitchInProgress) {
|
||||||
return Promise.reject('Switch in progress.');
|
return Promise.reject('Switch in progress.');
|
||||||
}
|
}
|
||||||
|
@ -1429,7 +1496,41 @@ export default {
|
||||||
}
|
}
|
||||||
|
|
||||||
if (toggle) {
|
if (toggle) {
|
||||||
return this._switchToScreenSharing(options);
|
const wasVideoMuted = this.isLocalVideoMuted();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this._switchToScreenSharing(options);
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to switch to screensharing', err);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (wasVideoMuted) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const { height } = this.localVideo.track.getSettings();
|
||||||
|
const defaultCamera
|
||||||
|
= getUserSelectedCameraDeviceId(APP.store.getState());
|
||||||
|
let effect;
|
||||||
|
|
||||||
|
try {
|
||||||
|
effect = await this._createPresenterStreamEffect(
|
||||||
|
height, defaultCamera);
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to create the presenter effect');
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
await this.localVideo.setEffect(effect);
|
||||||
|
muteLocalVideo(false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to create the presenter effect', err);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return this._untoggleScreenSharing
|
return this._untoggleScreenSharing
|
||||||
|
@ -1455,7 +1556,6 @@ export default {
|
||||||
let externalInstallation = false;
|
let externalInstallation = false;
|
||||||
let DSExternalInstallationInProgress = false;
|
let DSExternalInstallationInProgress = false;
|
||||||
const didHaveVideo = Boolean(this.localVideo);
|
const didHaveVideo = Boolean(this.localVideo);
|
||||||
const wasVideoMuted = this.isLocalVideoMuted();
|
|
||||||
|
|
||||||
const getDesktopStreamPromise = options.desktopStream
|
const getDesktopStreamPromise = options.desktopStream
|
||||||
? Promise.resolve([ options.desktopStream ])
|
? Promise.resolve([ options.desktopStream ])
|
||||||
|
@ -1506,8 +1606,7 @@ export default {
|
||||||
// Stores the "untoggle" handler which remembers whether was
|
// Stores the "untoggle" handler which remembers whether was
|
||||||
// there any video before and whether was it muted.
|
// there any video before and whether was it muted.
|
||||||
this._untoggleScreenSharing
|
this._untoggleScreenSharing
|
||||||
= this._turnScreenSharingOff
|
= this._turnScreenSharingOff.bind(this, didHaveVideo);
|
||||||
.bind(this, didHaveVideo, wasVideoMuted);
|
|
||||||
desktopStream.on(
|
desktopStream.on(
|
||||||
JitsiTrackEvents.LOCAL_TRACK_STOPPED,
|
JitsiTrackEvents.LOCAL_TRACK_STOPPED,
|
||||||
() => {
|
() => {
|
||||||
|
@ -1532,6 +1631,45 @@ export default {
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 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.
|
||||||
|
*/
|
||||||
|
async _createPresenterStreamEffect(height, cameraDeviceId = null) {
|
||||||
|
let presenterTrack;
|
||||||
|
|
||||||
|
try {
|
||||||
|
presenterTrack = await createLocalPresenterTrack({
|
||||||
|
cameraDeviceId
|
||||||
|
},
|
||||||
|
height);
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to create a camera track for presenter', err);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this.localPresenterVideo = presenterTrack;
|
||||||
|
try {
|
||||||
|
const effect = await createPresenterEffect(presenterTrack.stream);
|
||||||
|
|
||||||
|
APP.store.dispatch(trackAdded(this.localPresenterVideo));
|
||||||
|
|
||||||
|
return effect;
|
||||||
|
} catch (err) {
|
||||||
|
logger.error('Failed to create the presenter effect', err);
|
||||||
|
APP.store.dispatch(
|
||||||
|
setVideoMuted(true, MEDIA_TYPE.PRESENTER));
|
||||||
|
APP.store.dispatch(notifyCameraError(err));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Tries to switch to the screensharing mode by disposing camera stream and
|
* Tries to switch to the screensharing mode by disposing camera stream and
|
||||||
* replacing it with a desktop one.
|
* replacing it with a desktop one.
|
||||||
|
@ -1992,36 +2130,56 @@ export default {
|
||||||
const videoWasMuted = this.isLocalVideoMuted();
|
const videoWasMuted = this.isLocalVideoMuted();
|
||||||
|
|
||||||
sendAnalytics(createDeviceChangedEvent('video', 'input'));
|
sendAnalytics(createDeviceChangedEvent('video', 'input'));
|
||||||
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;
|
// If both screenshare and video are in progress, restart the
|
||||||
})
|
// presenter mode with the new camera device.
|
||||||
.then(stream => {
|
if (this.isSharingScreen && !videoWasMuted) {
|
||||||
// if we are screen sharing we do not want to stop it
|
const { height } = this.localVideo.track.getSettings();
|
||||||
if (this.isSharingScreen) {
|
|
||||||
return Promise.resolve();
|
|
||||||
}
|
|
||||||
|
|
||||||
return this.useVideoStream(stream);
|
// dispose the existing presenter track and create a new
|
||||||
})
|
// camera track.
|
||||||
.then(() => {
|
APP.store.dispatch(setVideoMuted(true, MEDIA_TYPE.PRESENTER));
|
||||||
|
|
||||||
|
return this._createPresenterStreamEffect(height, cameraDeviceId)
|
||||||
|
.then(effect => this.localVideo.setEffect(effect))
|
||||||
|
.then(() => {
|
||||||
|
muteLocalVideo(false);
|
||||||
|
this.setVideoMuteStatus(false);
|
||||||
|
logger.log('switched local video device');
|
||||||
|
this._updateVideoDeviceId();
|
||||||
|
})
|
||||||
|
.catch(err => APP.store.dispatch(notifyCameraError(err)));
|
||||||
|
|
||||||
|
// If screenshare is in progress but video is muted,
|
||||||
|
// update the default device id for video.
|
||||||
|
} else if (this.isSharingScreen && videoWasMuted) {
|
||||||
logger.log('switched local video device');
|
logger.log('switched local video device');
|
||||||
this._updateVideoDeviceId();
|
this._updateVideoDeviceId();
|
||||||
})
|
|
||||||
.catch(err => {
|
// if there is only video, switch to the new camera stream.
|
||||||
APP.store.dispatch(notifyCameraError(err));
|
} 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;
|
||||||
|
})
|
||||||
|
.then(stream => this.useVideoStream(stream))
|
||||||
|
.then(() => {
|
||||||
|
logger.log('switched local video device');
|
||||||
|
this._updateVideoDeviceId();
|
||||||
|
})
|
||||||
|
.catch(err => APP.store.dispatch(notifyCameraError(err)));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -147,9 +147,10 @@ MiddlewareRegistry.register(store => next => action => {
|
||||||
const state = getState();
|
const state = getState();
|
||||||
const { localTracksDuration } = state['features/analytics'];
|
const { localTracksDuration } = state['features/analytics'];
|
||||||
|
|
||||||
if (localTracksDuration.conference.startedTime === -1) {
|
if (localTracksDuration.conference.startedTime === -1 || action.mediaType === 'presenter') {
|
||||||
// We don't want to track the media duration if the conference is not joined yet because otherwise we won't
|
// We don't want to track the media duration if the conference is not joined yet because otherwise we won't
|
||||||
// be able to compare them with the conference duration (from conference join to conference will leave).
|
// be able to compare them with the conference duration (from conference join to conference will leave).
|
||||||
|
// Also, do not track media duration for presenter tracks.
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
dispatch({
|
dispatch({
|
||||||
|
|
|
@ -46,6 +46,7 @@ import {
|
||||||
getCurrentConference
|
getCurrentConference
|
||||||
} from './functions';
|
} from './functions';
|
||||||
import logger from './logger';
|
import logger from './logger';
|
||||||
|
import { MEDIA_TYPE } from '../media';
|
||||||
|
|
||||||
declare var APP: Object;
|
declare var APP: Object;
|
||||||
|
|
||||||
|
@ -589,7 +590,10 @@ function _syncReceiveVideoQuality({ getState }, next, action) {
|
||||||
function _trackAddedOrRemoved(store, next, action) {
|
function _trackAddedOrRemoved(store, next, action) {
|
||||||
const track = action.track;
|
const track = action.track;
|
||||||
|
|
||||||
if (track && track.local) {
|
// TODO All track swapping should happen here instead of conference.js.
|
||||||
|
// Since we swap the tracks for the web client in conference.js, ignore
|
||||||
|
// presenter tracks here and do not add/remove them to/from the conference.
|
||||||
|
if (track && track.local && track.mediaType !== MEDIA_TYPE.PRESENTER) {
|
||||||
return (
|
return (
|
||||||
_syncConferenceLocalTracksWithState(store, action)
|
_syncConferenceLocalTracksWithState(store, action)
|
||||||
.then(() => next(action)));
|
.then(() => next(action)));
|
||||||
|
|
|
@ -11,7 +11,11 @@ import {
|
||||||
STORE_VIDEO_TRANSFORM,
|
STORE_VIDEO_TRANSFORM,
|
||||||
TOGGLE_CAMERA_FACING_MODE
|
TOGGLE_CAMERA_FACING_MODE
|
||||||
} from './actionTypes';
|
} from './actionTypes';
|
||||||
import { CAMERA_FACING_MODE, VIDEO_MUTISM_AUTHORITY } from './constants';
|
import {
|
||||||
|
CAMERA_FACING_MODE,
|
||||||
|
MEDIA_TYPE,
|
||||||
|
VIDEO_MUTISM_AUTHORITY
|
||||||
|
} from './constants';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Action to adjust the availability of the local audio.
|
* Action to adjust the availability of the local audio.
|
||||||
|
@ -89,6 +93,7 @@ export function setVideoAvailable(available: boolean) {
|
||||||
*
|
*
|
||||||
* @param {boolean} muted - True if the local video is to be muted or false if
|
* @param {boolean} muted - True if the local video is to be muted or false if
|
||||||
* the local video is to be unmuted.
|
* the local video is to be unmuted.
|
||||||
|
* @param {MEDIA_TYPE} mediaType - The type of media.
|
||||||
* @param {number} authority - The {@link VIDEO_MUTISM_AUTHORITY} which is
|
* @param {number} authority - The {@link VIDEO_MUTISM_AUTHORITY} which is
|
||||||
* muting/unmuting the local video.
|
* muting/unmuting the local video.
|
||||||
* @param {boolean} ensureTrack - True if we want to ensure that a new track is
|
* @param {boolean} ensureTrack - True if we want to ensure that a new track is
|
||||||
|
@ -97,6 +102,7 @@ export function setVideoAvailable(available: boolean) {
|
||||||
*/
|
*/
|
||||||
export function setVideoMuted(
|
export function setVideoMuted(
|
||||||
muted: boolean,
|
muted: boolean,
|
||||||
|
mediaType: MEDIA_TYPE = MEDIA_TYPE.VIDEO,
|
||||||
authority: number = VIDEO_MUTISM_AUTHORITY.USER,
|
authority: number = VIDEO_MUTISM_AUTHORITY.USER,
|
||||||
ensureTrack: boolean = false) {
|
ensureTrack: boolean = false) {
|
||||||
return (dispatch: Dispatch<any>, getState: Function) => {
|
return (dispatch: Dispatch<any>, getState: Function) => {
|
||||||
|
@ -107,6 +113,8 @@ export function setVideoMuted(
|
||||||
|
|
||||||
return dispatch({
|
return dispatch({
|
||||||
type: SET_VIDEO_MUTED,
|
type: SET_VIDEO_MUTED,
|
||||||
|
authority,
|
||||||
|
mediaType,
|
||||||
ensureTrack,
|
ensureTrack,
|
||||||
muted: newValue
|
muted: newValue
|
||||||
});
|
});
|
||||||
|
|
|
@ -15,6 +15,7 @@ export const CAMERA_FACING_MODE = {
|
||||||
*/
|
*/
|
||||||
export const MEDIA_TYPE = {
|
export const MEDIA_TYPE = {
|
||||||
AUDIO: 'audio',
|
AUDIO: 'audio',
|
||||||
|
PRESENTER: 'presenter',
|
||||||
VIDEO: 'video'
|
VIDEO: 'video'
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -17,7 +17,11 @@ import { getPropertyValue } from '../settings';
|
||||||
import { setTrackMuted, TRACK_ADDED } from '../tracks';
|
import { setTrackMuted, TRACK_ADDED } from '../tracks';
|
||||||
|
|
||||||
import { setAudioMuted, setCameraFacingMode, setVideoMuted } from './actions';
|
import { setAudioMuted, setCameraFacingMode, setVideoMuted } from './actions';
|
||||||
import { CAMERA_FACING_MODE, VIDEO_MUTISM_AUTHORITY } from './constants';
|
import {
|
||||||
|
CAMERA_FACING_MODE,
|
||||||
|
MEDIA_TYPE,
|
||||||
|
VIDEO_MUTISM_AUTHORITY
|
||||||
|
} from './constants';
|
||||||
import logger from './logger';
|
import logger from './logger';
|
||||||
import {
|
import {
|
||||||
_AUDIO_INITIAL_MEDIA_STATE,
|
_AUDIO_INITIAL_MEDIA_STATE,
|
||||||
|
@ -45,7 +49,10 @@ MiddlewareRegistry.register(store => next => action => {
|
||||||
const result = next(action);
|
const result = next(action);
|
||||||
const { track } = action;
|
const { track } = action;
|
||||||
|
|
||||||
track.local && _syncTrackMutedState(store, track);
|
// Don't sync track mute state with the redux store for screenshare
|
||||||
|
// since video mute state represents local camera mute state only.
|
||||||
|
track.local && track.videoType !== 'desktop'
|
||||||
|
&& _syncTrackMutedState(store, track);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -72,7 +79,7 @@ function _appStateChanged({ dispatch }, next, action) {
|
||||||
|
|
||||||
sendAnalytics(createTrackMutedEvent('video', 'background mode', mute));
|
sendAnalytics(createTrackMutedEvent('video', 'background mode', mute));
|
||||||
|
|
||||||
dispatch(setVideoMuted(mute, VIDEO_MUTISM_AUTHORITY.BACKGROUND));
|
dispatch(setVideoMuted(mute, MEDIA_TYPE.VIDEO, VIDEO_MUTISM_AUTHORITY.BACKGROUND));
|
||||||
|
|
||||||
return next(action);
|
return next(action);
|
||||||
}
|
}
|
||||||
|
@ -94,7 +101,11 @@ function _setAudioOnly({ dispatch }, next, action) {
|
||||||
|
|
||||||
sendAnalytics(createTrackMutedEvent('video', 'audio-only mode', audioOnly));
|
sendAnalytics(createTrackMutedEvent('video', 'audio-only mode', audioOnly));
|
||||||
|
|
||||||
dispatch(setVideoMuted(audioOnly, VIDEO_MUTISM_AUTHORITY.AUDIO_ONLY, ensureVideoTrack));
|
// Make sure we mute both the desktop and video tracks.
|
||||||
|
dispatch(setVideoMuted(
|
||||||
|
audioOnly, MEDIA_TYPE.VIDEO, VIDEO_MUTISM_AUTHORITY.AUDIO_ONLY, ensureVideoTrack));
|
||||||
|
dispatch(setVideoMuted(
|
||||||
|
audioOnly, MEDIA_TYPE.PRESENTER, VIDEO_MUTISM_AUTHORITY.AUDIO_ONLY, ensureVideoTrack));
|
||||||
|
|
||||||
return next(action);
|
return next(action);
|
||||||
}
|
}
|
||||||
|
@ -231,7 +242,9 @@ function _setRoom({ dispatch, getState }, next, action) {
|
||||||
*/
|
*/
|
||||||
function _syncTrackMutedState({ getState }, track) {
|
function _syncTrackMutedState({ getState }, track) {
|
||||||
const state = getState()['features/base/media'];
|
const state = getState()['features/base/media'];
|
||||||
const muted = Boolean(state[track.mediaType].muted);
|
const mediaType = track.mediaType === MEDIA_TYPE.PRESENTER
|
||||||
|
? MEDIA_TYPE.VIDEO : track.mediaType;
|
||||||
|
const muted = Boolean(state[mediaType].muted);
|
||||||
|
|
||||||
// XXX If muted state of track when it was added is different from our media
|
// XXX If muted state of track when it was added is different from our media
|
||||||
// muted state, we need to mute track and explicitly modify 'muted' property
|
// muted state, we need to mute track and explicitly modify 'muted' property
|
||||||
|
|
|
@ -10,6 +10,47 @@ import {
|
||||||
|
|
||||||
import logger from './logger';
|
import logger from './logger';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a local video track for presenter. The constraints are computed based
|
||||||
|
* on the height of the desktop that is being shared.
|
||||||
|
*
|
||||||
|
* @param {Object} options - The options with which the local presenter track
|
||||||
|
* is to be created.
|
||||||
|
* @param {string|null} [options.cameraDeviceId] - Camera device id or
|
||||||
|
* {@code undefined} to use app's settings.
|
||||||
|
* @param {number} desktopHeight - The height of the desktop that is being
|
||||||
|
* shared.
|
||||||
|
* @returns {Promise<JitsiLocalTrack>}
|
||||||
|
*/
|
||||||
|
export async function createLocalPresenterTrack(options, desktopHeight) {
|
||||||
|
const { cameraDeviceId } = options;
|
||||||
|
|
||||||
|
// compute the constraints of the camera track based on the resolution
|
||||||
|
// of the desktop screen that is being shared.
|
||||||
|
const cameraHeights = [ 180, 270, 360, 540, 720 ];
|
||||||
|
const proportion = 4;
|
||||||
|
const result = cameraHeights.find(
|
||||||
|
height => (desktopHeight / proportion) < height);
|
||||||
|
const constraints = {
|
||||||
|
video: {
|
||||||
|
aspectRatio: 4 / 3,
|
||||||
|
height: {
|
||||||
|
exact: result
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const [ videoTrack ] = await JitsiMeetJS.createLocalTracks(
|
||||||
|
{
|
||||||
|
cameraDeviceId,
|
||||||
|
constraints,
|
||||||
|
devices: [ 'video' ]
|
||||||
|
});
|
||||||
|
|
||||||
|
videoTrack.type = MEDIA_TYPE.PRESENTER;
|
||||||
|
|
||||||
|
return videoTrack;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create local tracks of specific types.
|
* Create local tracks of specific types.
|
||||||
*
|
*
|
||||||
|
@ -53,11 +94,15 @@ export function createLocalTracksF(
|
||||||
|
|
||||||
const state = store.getState();
|
const state = store.getState();
|
||||||
const {
|
const {
|
||||||
constraints,
|
|
||||||
desktopSharingFrameRate,
|
desktopSharingFrameRate,
|
||||||
firefox_fake_device, // eslint-disable-line camelcase
|
firefox_fake_device, // eslint-disable-line camelcase
|
||||||
resolution
|
resolution
|
||||||
} = state['features/base/config'];
|
} = state['features/base/config'];
|
||||||
|
const constraints = options.constraints
|
||||||
|
?? state['features/base/config'].constraints;
|
||||||
|
|
||||||
|
// Do not load blur effect if option for ignoring effects is present.
|
||||||
|
// This is needed when we are creating a video track for presenter mode.
|
||||||
const loadEffectsPromise = state['features/blur'].blurEnabled
|
const loadEffectsPromise = state['features/blur'].blurEnabled
|
||||||
? getBlurEffect()
|
? getBlurEffect()
|
||||||
.then(blurEffect => [ blurEffect ])
|
.then(blurEffect => [ blurEffect ])
|
||||||
|
@ -157,6 +202,18 @@ export function getLocalVideoTrack(tracks) {
|
||||||
return getLocalTrack(tracks, MEDIA_TYPE.VIDEO);
|
return getLocalTrack(tracks, MEDIA_TYPE.VIDEO);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the media type of the local video, presenter or video.
|
||||||
|
*
|
||||||
|
* @param {Track[]} tracks - List of all tracks.
|
||||||
|
* @returns {MEDIA_TYPE}
|
||||||
|
*/
|
||||||
|
export function getLocalVideoType(tracks) {
|
||||||
|
const presenterTrack = getLocalTrack(tracks, MEDIA_TYPE.PRESENTER);
|
||||||
|
|
||||||
|
return presenterTrack ? MEDIA_TYPE.PRESENTER : MEDIA_TYPE.VIDEO;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns track of specified media type for specified participant id.
|
* Returns track of specified media type for specified participant id.
|
||||||
*
|
*
|
||||||
|
@ -197,6 +254,29 @@ export function getTracksByMediaType(tracks, mediaType) {
|
||||||
return tracks.filter(t => t.mediaType === mediaType);
|
return tracks.filter(t => t.mediaType === mediaType);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the local video track in the given set of tracks is muted.
|
||||||
|
*
|
||||||
|
* @param {Track[]} tracks - List of all tracks.
|
||||||
|
* @returns {Track[]}
|
||||||
|
*/
|
||||||
|
export function isLocalVideoTrackMuted(tracks) {
|
||||||
|
const presenterTrack = getLocalTrack(tracks, MEDIA_TYPE.PRESENTER);
|
||||||
|
const videoTrack = getLocalTrack(tracks, MEDIA_TYPE.VIDEO);
|
||||||
|
|
||||||
|
// Make sure we check the mute status of only camera tracks, i.e.,
|
||||||
|
// presenter track when it exists, camera track when the presenter
|
||||||
|
// track doesn't exist.
|
||||||
|
if (presenterTrack) {
|
||||||
|
return isLocalTrackMuted(tracks, MEDIA_TYPE.PRESENTER);
|
||||||
|
} else if (videoTrack) {
|
||||||
|
return videoTrack.videoType === 'camera'
|
||||||
|
? isLocalTrackMuted(tracks, MEDIA_TYPE.VIDEO) : true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Checks if the first local track in the given tracks set is muted.
|
* Checks if the first local track in the given tracks set is muted.
|
||||||
*
|
*
|
||||||
|
|
|
@ -6,6 +6,7 @@ import {
|
||||||
SET_AUDIO_MUTED,
|
SET_AUDIO_MUTED,
|
||||||
SET_CAMERA_FACING_MODE,
|
SET_CAMERA_FACING_MODE,
|
||||||
SET_VIDEO_MUTED,
|
SET_VIDEO_MUTED,
|
||||||
|
VIDEO_MUTISM_AUTHORITY,
|
||||||
TOGGLE_CAMERA_FACING_MODE,
|
TOGGLE_CAMERA_FACING_MODE,
|
||||||
toggleCameraFacingMode
|
toggleCameraFacingMode
|
||||||
} from '../media';
|
} from '../media';
|
||||||
|
@ -89,7 +90,7 @@ MiddlewareRegistry.register(store => next => action => {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
_setMuted(store, action, MEDIA_TYPE.VIDEO);
|
_setMuted(store, action, action.mediaType);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case TOGGLE_CAMERA_FACING_MODE: {
|
case TOGGLE_CAMERA_FACING_MODE: {
|
||||||
|
@ -131,7 +132,7 @@ MiddlewareRegistry.register(store => next => action => {
|
||||||
const { jitsiTrack } = action.track;
|
const { jitsiTrack } = action.track;
|
||||||
const muted = jitsiTrack.isMuted();
|
const muted = jitsiTrack.isMuted();
|
||||||
const participantID = jitsiTrack.getParticipantId();
|
const participantID = jitsiTrack.getParticipantId();
|
||||||
const isVideoTrack = jitsiTrack.isVideoTrack();
|
const isVideoTrack = jitsiTrack.type !== MEDIA_TYPE.AUDIO;
|
||||||
|
|
||||||
if (isVideoTrack) {
|
if (isVideoTrack) {
|
||||||
if (jitsiTrack.isLocal()) {
|
if (jitsiTrack.isLocal()) {
|
||||||
|
@ -255,7 +256,7 @@ function _removeNoDataFromSourceNotification({ getState, dispatch }, track) {
|
||||||
* @private
|
* @private
|
||||||
* @returns {void}
|
* @returns {void}
|
||||||
*/
|
*/
|
||||||
function _setMuted(store, { ensureTrack, muted }, mediaType: MEDIA_TYPE) {
|
function _setMuted(store, { ensureTrack, authority, muted }, mediaType: MEDIA_TYPE) {
|
||||||
const localTrack
|
const localTrack
|
||||||
= _getLocalTrack(store, mediaType, /* includePending */ true);
|
= _getLocalTrack(store, mediaType, /* includePending */ true);
|
||||||
|
|
||||||
|
@ -265,8 +266,12 @@ function _setMuted(store, { ensureTrack, muted }, mediaType: MEDIA_TYPE) {
|
||||||
// `jitsiTrack`, then the `muted` state will be applied once the
|
// `jitsiTrack`, then the `muted` state will be applied once the
|
||||||
// `jitsiTrack` is created.
|
// `jitsiTrack` is created.
|
||||||
const { jitsiTrack } = localTrack;
|
const { jitsiTrack } = localTrack;
|
||||||
|
const isAudioOnly = authority === VIDEO_MUTISM_AUTHORITY.AUDIO_ONLY;
|
||||||
|
|
||||||
jitsiTrack && setTrackMuted(jitsiTrack, muted);
|
// screenshare cannot be muted or unmuted using the video mute button
|
||||||
|
// anymore, unless it is muted by audioOnly.
|
||||||
|
jitsiTrack && (jitsiTrack.videoType !== 'desktop' || isAudioOnly)
|
||||||
|
&& setTrackMuted(jitsiTrack, muted);
|
||||||
} else if (!muted && ensureTrack && typeof APP === 'undefined') {
|
} else if (!muted && ensureTrack && typeof APP === 'undefined') {
|
||||||
// FIXME: This only runs on mobile now because web has its own way of
|
// FIXME: This only runs on mobile now because web has its own way of
|
||||||
// creating local tracks. Adjust the check once they are unified.
|
// creating local tracks. Adjust the check once they are unified.
|
||||||
|
|
|
@ -0,0 +1,161 @@
|
||||||
|
// @flow
|
||||||
|
|
||||||
|
import {
|
||||||
|
CLEAR_INTERVAL,
|
||||||
|
INTERVAL_TIMEOUT,
|
||||||
|
SET_INTERVAL,
|
||||||
|
timerWorkerScript
|
||||||
|
} from './TimeWorker';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a modified MediaStream that adds video as pip on a desktop stream.
|
||||||
|
* <tt>JitsiStreamPresenterEffect</tt> does the processing of the original
|
||||||
|
* desktop stream.
|
||||||
|
*/
|
||||||
|
export default class JitsiStreamPresenterEffect {
|
||||||
|
_canvas: HTMLCanvasElement;
|
||||||
|
_ctx: CanvasRenderingContext2D;
|
||||||
|
_desktopElement: HTMLVideoElement;
|
||||||
|
_desktopStream: MediaStream;
|
||||||
|
_frameRate: number;
|
||||||
|
_onVideoFrameTimer: Function;
|
||||||
|
_onVideoFrameTimerWorker: Function;
|
||||||
|
_renderVideo: Function;
|
||||||
|
_videoFrameTimerWorker: Worker;
|
||||||
|
_videoElement: HTMLVideoElement;
|
||||||
|
isEnabled: Function;
|
||||||
|
startEffect: Function;
|
||||||
|
stopEffect: Function;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Represents a modified MediaStream that adds a camera track at the
|
||||||
|
* bottom right corner of the desktop track using a HTML canvas.
|
||||||
|
* <tt>JitsiStreamPresenterEffect</tt> does the processing of the original
|
||||||
|
* video stream.
|
||||||
|
*
|
||||||
|
* @param {MediaStream} videoStream - The video stream which is user for
|
||||||
|
* creating the canvas.
|
||||||
|
*/
|
||||||
|
constructor(videoStream: MediaStream) {
|
||||||
|
const videoDiv = document.createElement('div');
|
||||||
|
const firstVideoTrack = videoStream.getVideoTracks()[0];
|
||||||
|
const { height, width, frameRate } = firstVideoTrack.getSettings() ?? firstVideoTrack.getConstraints();
|
||||||
|
|
||||||
|
this._canvas = document.createElement('canvas');
|
||||||
|
this._ctx = this._canvas.getContext('2d');
|
||||||
|
|
||||||
|
if (document.body !== null) {
|
||||||
|
document.body.appendChild(this._canvas);
|
||||||
|
}
|
||||||
|
this._desktopElement = document.createElement('video');
|
||||||
|
this._videoElement = document.createElement('video');
|
||||||
|
videoDiv.appendChild(this._videoElement);
|
||||||
|
videoDiv.appendChild(this._desktopElement);
|
||||||
|
if (document.body !== null) {
|
||||||
|
document.body.appendChild(videoDiv);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Set the video element properties
|
||||||
|
this._frameRate = parseInt(frameRate, 10);
|
||||||
|
this._videoElement.width = parseInt(width, 10);
|
||||||
|
this._videoElement.height = parseInt(height, 10);
|
||||||
|
this._videoElement.autoplay = true;
|
||||||
|
this._videoElement.srcObject = videoStream;
|
||||||
|
|
||||||
|
// set the style attribute of the div to make it invisible
|
||||||
|
videoDiv.style.display = 'none';
|
||||||
|
|
||||||
|
// Bind event handler so it is only bound once for every instance.
|
||||||
|
this._onVideoFrameTimer = this._onVideoFrameTimer.bind(this);
|
||||||
|
this._videoFrameTimerWorker = new Worker(timerWorkerScript);
|
||||||
|
this._videoFrameTimerWorker.onmessage = this._onVideoFrameTimer;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* EventHandler onmessage for the videoFrameTimerWorker WebWorker.
|
||||||
|
*
|
||||||
|
* @private
|
||||||
|
* @param {EventHandler} response - The onmessage EventHandler parameter.
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
|
_onVideoFrameTimer(response) {
|
||||||
|
if (response.data.id === INTERVAL_TIMEOUT) {
|
||||||
|
this._renderVideo();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loop function to render the video frame input and draw presenter effect.
|
||||||
|
*
|
||||||
|
* @private
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
|
_renderVideo() {
|
||||||
|
// adjust the canvas width/height on every frame incase the window has been resized.
|
||||||
|
const [ track ] = this._desktopStream.getVideoTracks();
|
||||||
|
const { height, width } = track.getSettings() ?? track.getConstraints();
|
||||||
|
|
||||||
|
this._canvas.width = parseInt(width, 10);
|
||||||
|
this._canvas.height = parseInt(height, 10);
|
||||||
|
this._ctx.drawImage(this._desktopElement, 0, 0, this._canvas.width, this._canvas.height);
|
||||||
|
this._ctx.drawImage(this._videoElement, this._canvas.width - this._videoElement.width, this._canvas.height
|
||||||
|
- this._videoElement.height, this._videoElement.width, this._videoElement.height);
|
||||||
|
|
||||||
|
// draw a border around the video element.
|
||||||
|
this._ctx.beginPath();
|
||||||
|
this._ctx.lineWidth = 2;
|
||||||
|
this._ctx.strokeStyle = '#A9A9A9'; // dark grey
|
||||||
|
this._ctx.rect(this._canvas.width - this._videoElement.width, this._canvas.height - this._videoElement.height,
|
||||||
|
this._videoElement.width, this._videoElement.height);
|
||||||
|
this._ctx.stroke();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the local track supports this effect.
|
||||||
|
*
|
||||||
|
* @param {JitsiLocalTrack} jitsiLocalTrack - Track to apply effect.
|
||||||
|
* @returns {boolean} - Returns true if this effect can run on the
|
||||||
|
* specified track, false otherwise.
|
||||||
|
*/
|
||||||
|
isEnabled(jitsiLocalTrack: Object) {
|
||||||
|
return jitsiLocalTrack.isVideoTrack() && jitsiLocalTrack.videoType === 'desktop';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts loop to capture video frame and render presenter effect.
|
||||||
|
*
|
||||||
|
* @param {MediaStream} desktopStream - Stream to be used for processing.
|
||||||
|
* @returns {MediaStream} - The stream with the applied effect.
|
||||||
|
*/
|
||||||
|
startEffect(desktopStream: MediaStream) {
|
||||||
|
const firstVideoTrack = desktopStream.getVideoTracks()[0];
|
||||||
|
const { height, width } = firstVideoTrack.getSettings() ?? firstVideoTrack.getConstraints();
|
||||||
|
|
||||||
|
// set the desktop element properties.
|
||||||
|
this._desktopStream = desktopStream;
|
||||||
|
this._desktopElement.width = parseInt(width, 10);
|
||||||
|
this._desktopElement.height = parseInt(height, 10);
|
||||||
|
this._desktopElement.autoplay = true;
|
||||||
|
this._desktopElement.srcObject = desktopStream;
|
||||||
|
this._canvas.width = parseInt(width, 10);
|
||||||
|
this._canvas.height = parseInt(height, 10);
|
||||||
|
this._videoFrameTimerWorker.postMessage({
|
||||||
|
id: SET_INTERVAL,
|
||||||
|
timeMs: 1000 / this._frameRate
|
||||||
|
});
|
||||||
|
|
||||||
|
return this._canvas.captureStream(this._frameRate);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Stops the capture and render loop.
|
||||||
|
*
|
||||||
|
* @returns {void}
|
||||||
|
*/
|
||||||
|
stopEffect() {
|
||||||
|
this._videoFrameTimerWorker.postMessage({
|
||||||
|
id: CLEAR_INTERVAL
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
|
@ -0,0 +1,62 @@
|
||||||
|
// @flow
|
||||||
|
|
||||||
|
/**
|
||||||
|
* SET_INTERVAL constant is used to set interval and it is set in
|
||||||
|
* the id property of the request.data property. timeMs property must
|
||||||
|
* also be set. request.data example:
|
||||||
|
*
|
||||||
|
* {
|
||||||
|
* id: SET_INTERVAL,
|
||||||
|
* timeMs: 33
|
||||||
|
* }
|
||||||
|
*/
|
||||||
|
export const SET_INTERVAL = 1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* CLEAR_INTERVAL constant is used to clear the interval and it is set in
|
||||||
|
* the id property of the request.data property.
|
||||||
|
*
|
||||||
|
* {
|
||||||
|
* id: CLEAR_INTERVAL
|
||||||
|
* }
|
||||||
|
*/
|
||||||
|
export const CLEAR_INTERVAL = 2;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* INTERVAL_TIMEOUT constant is used as response and it is set in the id
|
||||||
|
* property.
|
||||||
|
*
|
||||||
|
* {
|
||||||
|
* id: INTERVAL_TIMEOUT
|
||||||
|
* }
|
||||||
|
*/
|
||||||
|
export const INTERVAL_TIMEOUT = 3;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The following code is needed as string to create a URL from a Blob.
|
||||||
|
* The URL is then passed to a WebWorker. Reason for this is to enable
|
||||||
|
* use of setInterval that is not throttled when tab is inactive.
|
||||||
|
*/
|
||||||
|
const code = `
|
||||||
|
var timer;
|
||||||
|
|
||||||
|
onmessage = function(request) {
|
||||||
|
switch (request.data.id) {
|
||||||
|
case ${SET_INTERVAL}: {
|
||||||
|
timer = setInterval(() => {
|
||||||
|
postMessage({ id: ${INTERVAL_TIMEOUT} });
|
||||||
|
}, request.data.timeMs);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case ${CLEAR_INTERVAL}: {
|
||||||
|
if (timer) {
|
||||||
|
clearInterval(timer);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
`;
|
||||||
|
|
||||||
|
export const timerWorkerScript
|
||||||
|
= URL.createObjectURL(new Blob([ code ], { type: 'application/javascript' }));
|
|
@ -0,0 +1,19 @@
|
||||||
|
// @flow
|
||||||
|
|
||||||
|
import JitsiStreamPresenterEffect from './JitsiStreamPresenterEffect';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Creates a new instance of JitsiStreamPresenterEffect.
|
||||||
|
*
|
||||||
|
* @param {MediaStream} stream - The video stream which will be used for
|
||||||
|
* creating the presenter effect.
|
||||||
|
* @returns {Promise<JitsiStreamPresenterEffect>}
|
||||||
|
*/
|
||||||
|
export function createPresenterEffect(stream: MediaStream) {
|
||||||
|
if (!MediaStreamTrack.prototype.getSettings
|
||||||
|
&& !MediaStreamTrack.prototype.getConstraints) {
|
||||||
|
return Promise.reject(new Error('JitsiStreamPresenterEffect not supported!'));
|
||||||
|
}
|
||||||
|
|
||||||
|
return Promise.resolve(new JitsiStreamPresenterEffect(stream));
|
||||||
|
}
|
|
@ -10,14 +10,13 @@ import {
|
||||||
import { setAudioOnly } from '../../base/audio-only';
|
import { setAudioOnly } from '../../base/audio-only';
|
||||||
import { translate } from '../../base/i18n';
|
import { translate } from '../../base/i18n';
|
||||||
import {
|
import {
|
||||||
MEDIA_TYPE,
|
|
||||||
VIDEO_MUTISM_AUTHORITY,
|
VIDEO_MUTISM_AUTHORITY,
|
||||||
setVideoMuted
|
setVideoMuted
|
||||||
} from '../../base/media';
|
} from '../../base/media';
|
||||||
import { connect } from '../../base/redux';
|
import { connect } from '../../base/redux';
|
||||||
import { AbstractVideoMuteButton } from '../../base/toolbox';
|
import { AbstractVideoMuteButton } from '../../base/toolbox';
|
||||||
import type { AbstractButtonProps } from '../../base/toolbox';
|
import type { AbstractButtonProps } from '../../base/toolbox';
|
||||||
import { isLocalTrackMuted } from '../../base/tracks';
|
import { getLocalVideoType, isLocalVideoTrackMuted } from '../../base/tracks';
|
||||||
import UIEvents from '../../../../service/UI/UIEvents';
|
import UIEvents from '../../../../service/UI/UIEvents';
|
||||||
|
|
||||||
declare var APP: Object;
|
declare var APP: Object;
|
||||||
|
@ -32,6 +31,11 @@ type Props = AbstractButtonProps & {
|
||||||
*/
|
*/
|
||||||
_audioOnly: boolean,
|
_audioOnly: boolean,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* MEDIA_TYPE of the local video.
|
||||||
|
*/
|
||||||
|
_videoMediaType: string,
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Whether video is currently muted or not.
|
* Whether video is currently muted or not.
|
||||||
*/
|
*/
|
||||||
|
@ -136,10 +140,12 @@ class VideoMuteButton extends AbstractVideoMuteButton<Props, *> {
|
||||||
this.props.dispatch(
|
this.props.dispatch(
|
||||||
setAudioOnly(false, /* ensureTrack */ true));
|
setAudioOnly(false, /* ensureTrack */ true));
|
||||||
}
|
}
|
||||||
|
const mediaType = this.props._videoMediaType;
|
||||||
|
|
||||||
this.props.dispatch(
|
this.props.dispatch(
|
||||||
setVideoMuted(
|
setVideoMuted(
|
||||||
videoMuted,
|
videoMuted,
|
||||||
|
mediaType,
|
||||||
VIDEO_MUTISM_AUTHORITY.USER,
|
VIDEO_MUTISM_AUTHORITY.USER,
|
||||||
/* ensureTrack */ true));
|
/* ensureTrack */ true));
|
||||||
|
|
||||||
|
@ -167,7 +173,8 @@ function _mapStateToProps(state): Object {
|
||||||
|
|
||||||
return {
|
return {
|
||||||
_audioOnly: Boolean(audioOnly),
|
_audioOnly: Boolean(audioOnly),
|
||||||
_videoMuted: isLocalTrackMuted(tracks, MEDIA_TYPE.VIDEO)
|
_videoMediaType: getLocalVideoType(tracks),
|
||||||
|
_videoMuted: isLocalVideoTrackMuted(tracks)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue