2021-06-23 11:23:44 +00:00
|
|
|
// @flow
|
|
|
|
|
|
|
|
import { MEDIA_TYPE, type MediaType } from '../base/media/constants';
|
2021-07-09 12:36:19 +00:00
|
|
|
import { isLocalParticipantModerator } from '../base/participants/functions';
|
2022-05-19 08:41:29 +00:00
|
|
|
import { isInBreakoutRoom } from '../breakout-rooms/functions';
|
2021-06-23 11:23:44 +00:00
|
|
|
|
|
|
|
import { MEDIA_TYPE_TO_WHITELIST_STORE_KEY, MEDIA_TYPE_TO_PENDING_STORE_KEY } from './constants';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns this feature's root state.
|
|
|
|
*
|
|
|
|
* @param {Object} state - Global state.
|
|
|
|
* @returns {Object} Feature state.
|
|
|
|
*/
|
|
|
|
const getState = state => state['features/av-moderation'];
|
|
|
|
|
2021-07-09 12:36:19 +00:00
|
|
|
/**
|
|
|
|
* We use to construct once the empty array so we can keep the same instance between calls
|
|
|
|
* of getParticipantsAskingToAudioUnmute.
|
|
|
|
*
|
|
|
|
* @type {*[]}
|
|
|
|
*/
|
|
|
|
const EMPTY_ARRAY = [];
|
|
|
|
|
2021-06-23 11:23:44 +00:00
|
|
|
/**
|
|
|
|
* Returns whether moderation is enabled per media type.
|
|
|
|
*
|
|
|
|
* @param {MEDIA_TYPE} mediaType - The media type to check.
|
|
|
|
* @param {Object} state - Global state.
|
|
|
|
* @returns {null|boolean|*}
|
|
|
|
*/
|
|
|
|
export const isEnabledFromState = (mediaType: MediaType, state: Object) =>
|
|
|
|
(mediaType === MEDIA_TYPE.AUDIO
|
2021-06-30 07:34:17 +00:00
|
|
|
? getState(state)?.audioModerationEnabled
|
|
|
|
: getState(state)?.videoModerationEnabled) === true;
|
2021-06-23 11:23:44 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether moderation is enabled per media type.
|
|
|
|
*
|
|
|
|
* @param {MEDIA_TYPE} mediaType - The media type to check.
|
|
|
|
* @returns {null|boolean|*}
|
|
|
|
*/
|
|
|
|
export const isEnabled = (mediaType: MediaType) => (state: Object) => isEnabledFromState(mediaType, state);
|
|
|
|
|
2021-07-09 12:36:19 +00:00
|
|
|
/**
|
|
|
|
* Returns whether moderation is supported by the backend.
|
|
|
|
*
|
|
|
|
* @returns {null|boolean}
|
|
|
|
*/
|
|
|
|
export const isSupported = () => (state: Object) => {
|
|
|
|
const { conference } = state['features/base/conference'];
|
|
|
|
|
2022-05-19 08:41:29 +00:00
|
|
|
return Boolean(!isInBreakoutRoom(state) && conference?.isAVModerationSupported());
|
2021-07-09 12:36:19 +00:00
|
|
|
};
|
|
|
|
|
2021-06-23 11:23:44 +00:00
|
|
|
/**
|
|
|
|
* Returns whether local participant is approved to unmute a media type.
|
|
|
|
*
|
|
|
|
* @param {MEDIA_TYPE} mediaType - The media type to check.
|
|
|
|
* @param {Object} state - Global state.
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
export const isLocalParticipantApprovedFromState = (mediaType: MediaType, state: Object) => {
|
|
|
|
const approved = (mediaType === MEDIA_TYPE.AUDIO
|
|
|
|
? getState(state).audioUnmuteApproved
|
|
|
|
: getState(state).videoUnmuteApproved) === true;
|
|
|
|
|
|
|
|
return approved || isLocalParticipantModerator(state);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns whether local participant is approved to unmute a media type.
|
|
|
|
*
|
|
|
|
* @param {MEDIA_TYPE} mediaType - The media type to check.
|
|
|
|
* @returns {null|boolean|*}
|
|
|
|
*/
|
|
|
|
export const isLocalParticipantApproved = (mediaType: MediaType) =>
|
|
|
|
(state: Object) =>
|
|
|
|
isLocalParticipantApprovedFromState(mediaType, state);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a selector creator which determines if the participant is approved or not for a media type.
|
|
|
|
*
|
|
|
|
* @param {string} id - The participant id.
|
|
|
|
* @param {MEDIA_TYPE} mediaType - The media type to check.
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
export const isParticipantApproved = (id: string, mediaType: MediaType) => (state: Object) => {
|
|
|
|
const storeKey = MEDIA_TYPE_TO_WHITELIST_STORE_KEY[mediaType];
|
|
|
|
|
|
|
|
return Boolean(getState(state)[storeKey][id]);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a selector creator which determines if the participant is pending or not for a media type.
|
|
|
|
*
|
2021-07-09 12:36:19 +00:00
|
|
|
* @param {Participant} participant - The participant.
|
2021-06-23 11:23:44 +00:00
|
|
|
* @param {MEDIA_TYPE} mediaType - The media type to check.
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
2021-07-09 12:36:19 +00:00
|
|
|
export const isParticipantPending = (participant: Object, mediaType: MediaType) => (state: Object) => {
|
2021-06-23 11:23:44 +00:00
|
|
|
const storeKey = MEDIA_TYPE_TO_PENDING_STORE_KEY[mediaType];
|
|
|
|
const arr = getState(state)[storeKey];
|
|
|
|
|
2021-07-09 12:36:19 +00:00
|
|
|
return Boolean(arr.find(pending => pending.id === participant.id));
|
2021-06-23 11:23:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Selector which returns a list with all the participants asking to audio unmute.
|
2022-07-14 07:10:08 +00:00
|
|
|
* This is visible only for the moderator.
|
2021-06-23 11:23:44 +00:00
|
|
|
*
|
|
|
|
* @param {Object} state - The global state.
|
|
|
|
* @returns {Array<Object>}
|
|
|
|
*/
|
|
|
|
export const getParticipantsAskingToAudioUnmute = (state: Object) => {
|
|
|
|
if (isLocalParticipantModerator(state)) {
|
2021-07-09 12:36:19 +00:00
|
|
|
return getState(state).pendingAudio;
|
2021-06-23 11:23:44 +00:00
|
|
|
}
|
|
|
|
|
2021-07-09 12:36:19 +00:00
|
|
|
return EMPTY_ARRAY;
|
2021-06-23 11:23:44 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns true if a special notification can be displayed when a participant
|
|
|
|
* tries to unmute.
|
|
|
|
*
|
|
|
|
* @param {MediaType} mediaType - 'audio' or 'video' media type.
|
|
|
|
* @param {Object} state - The global state.
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
export const shouldShowModeratedNotification = (mediaType: MediaType, state: Object) =>
|
|
|
|
isEnabledFromState(mediaType, state)
|
|
|
|
&& !isLocalParticipantApprovedFromState(mediaType, state);
|