// @flow import { batch } from 'react-redux'; import { getConferenceState } from '../base/conference'; import { JitsiConferenceEvents } from '../base/lib-jitsi-meet'; import { MEDIA_TYPE } from '../base/media'; import { getLocalParticipant, getRemoteParticipants, isLocalParticipantModerator, isParticipantModerator, PARTICIPANT_UPDATED, raiseHand } from '../base/participants'; import { MiddlewareRegistry, StateListenerRegistry } from '../base/redux'; import { hideNotification, showNotification } from '../notifications'; import { muteLocal } from '../video-menu/actions.any'; import { LOCAL_PARTICIPANT_MODERATION_NOTIFICATION, REQUEST_DISABLE_AUDIO_MODERATION, REQUEST_DISABLE_VIDEO_MODERATION, REQUEST_ENABLE_AUDIO_MODERATION, REQUEST_ENABLE_VIDEO_MODERATION } from './actionTypes'; import { disableModeration, dismissPendingParticipant, dismissPendingAudioParticipant, enableModeration, localParticipantApproved, participantApproved, participantPendingAudio } from './actions'; import { isEnabledFromState, isParticipantApproved, isParticipantPending } from './functions'; const VIDEO_MODERATION_NOTIFICATION_ID = 'video-moderation'; const AUDIO_MODERATION_NOTIFICATION_ID = 'audio-moderation'; const CS_MODERATION_NOTIFICATION_ID = 'video-moderation'; MiddlewareRegistry.register(({ dispatch, getState }) => next => action => { const { type } = action; const { conference } = getConferenceState(getState()); switch (type) { case LOCAL_PARTICIPANT_MODERATION_NOTIFICATION: { let descriptionKey; let titleKey; let uid; switch (action.mediaType) { case MEDIA_TYPE.AUDIO: { titleKey = 'notify.moderationInEffectTitle'; uid = AUDIO_MODERATION_NOTIFICATION_ID; break; } case MEDIA_TYPE.VIDEO: { titleKey = 'notify.moderationInEffectVideoTitle'; uid = VIDEO_MODERATION_NOTIFICATION_ID; break; } case MEDIA_TYPE.PRESENTER: { titleKey = 'notify.moderationInEffectCSTitle'; uid = CS_MODERATION_NOTIFICATION_ID; break; } } dispatch(showNotification({ customActionNameKey: 'notify.raiseHandAction', customActionHandler: () => batch(() => { dispatch(raiseHand(true)); dispatch(hideNotification(uid)); }), descriptionKey, sticky: true, titleKey, uid })); break; } case REQUEST_DISABLE_AUDIO_MODERATION: { conference.disableAVModeration(MEDIA_TYPE.AUDIO); break; } case REQUEST_DISABLE_VIDEO_MODERATION: { conference.disableAVModeration(MEDIA_TYPE.VIDEO); break; } case REQUEST_ENABLE_AUDIO_MODERATION: { conference.enableAVModeration(MEDIA_TYPE.AUDIO); break; } case REQUEST_ENABLE_VIDEO_MODERATION: { conference.enableAVModeration(MEDIA_TYPE.VIDEO); break; } case PARTICIPANT_UPDATED: { const state = getState(); const audioModerationEnabled = isEnabledFromState(MEDIA_TYPE.AUDIO, state); const participant = action.participant; if (participant && audioModerationEnabled) { if (isLocalParticipantModerator(state)) { // this is handled only by moderators if (participant.raisedHand) { // if participant raises hand show notification !isParticipantApproved(participant.id, MEDIA_TYPE.AUDIO)(state) && dispatch(participantPendingAudio(participant)); } else { // if participant lowers hand hide notification isParticipantPending(participant, MEDIA_TYPE.AUDIO)(state) && dispatch(dismissPendingAudioParticipant(participant)); } } else if (participant.id === getLocalParticipant(state).id && /* the new role */ isParticipantModerator(participant)) { // this is the granted moderator case getRemoteParticipants(state).forEach(p => { p.raisedHand && !isParticipantApproved(p.id, MEDIA_TYPE.AUDIO)(state) && dispatch(participantPendingAudio(p)); }); } } break; } } return next(action); }); /** * Registers a change handler for state['features/base/conference'].conference to * set the event listeners needed for the A/V moderation feature to operate. */ StateListenerRegistry.register( state => state['features/base/conference'].conference, (conference, { dispatch }, previousConference) => { if (conference && !previousConference) { // local participant is allowed to unmute conference.on(JitsiConferenceEvents.AV_MODERATION_APPROVED, ({ mediaType }) => { dispatch(localParticipantApproved(mediaType)); // Audio & video moderation are both enabled at the same time. // Avoid displaying 2 different notifications. if (mediaType === MEDIA_TYPE.AUDIO) { dispatch(showNotification({ titleKey: 'notify.hostAskedUnmute', sticky: true, customActionNameKey: 'notify.unmute', customActionHandler: () => dispatch(muteLocal(false, MEDIA_TYPE.AUDIO)) })); } }); conference.on(JitsiConferenceEvents.AV_MODERATION_CHANGED, ({ enabled, mediaType, actor }) => { enabled ? dispatch(enableModeration(mediaType, actor)) : dispatch(disableModeration(mediaType, actor)); }); // this is received by moderators conference.on( JitsiConferenceEvents.AV_MODERATION_PARTICIPANT_APPROVED, ({ participant, mediaType }) => { const { _id: id } = participant; batch(() => { // store in the whitelist dispatch(participantApproved(id, mediaType)); // remove from pending list dispatch(dismissPendingParticipant(id, mediaType)); }); }); } });