// @flow import { SET_FILMSTRIP_ENABLED } from '../../filmstrip/actionTypes'; import { SELECT_LARGE_VIDEO_PARTICIPANT } from '../../large-video/actionTypes'; import { APP_STATE_CHANGED } from '../../mobile/background/actionTypes'; import { SCREEN_SHARE_REMOTE_PARTICIPANTS_UPDATED, SET_TILE_VIEW } from '../../video-layout/actionTypes'; import { SET_AUDIO_ONLY } from '../audio-only/actionTypes'; import { CONFERENCE_JOINED } from '../conference/actionTypes'; import { PARTICIPANT_JOINED, PARTICIPANT_KICKED, PARTICIPANT_LEFT } from '../participants/actionTypes'; import { getParticipantById, getParticipantCount } from '../participants/functions'; import { MiddlewareRegistry } from '../redux'; import { isLocalVideoTrackDesktop } from '../tracks/functions'; import { limitLastN } from './functions'; import logger from './logger'; declare var APP: Object; MiddlewareRegistry.register(store => next => action => { const result = next(action); switch (action.type) { case APP_STATE_CHANGED: case CONFERENCE_JOINED: case PARTICIPANT_JOINED: case PARTICIPANT_KICKED: case PARTICIPANT_LEFT: case SCREEN_SHARE_REMOTE_PARTICIPANTS_UPDATED: case SELECT_LARGE_VIDEO_PARTICIPANT: case SET_AUDIO_ONLY: case SET_FILMSTRIP_ENABLED: case SET_TILE_VIEW: _updateLastN(store); break; } return result; }); /** * Updates the last N value in the conference based on the current state of the redux store. * * @param {Store} store - The redux store. * @private * @returns {void} */ function _updateLastN({ getState }) { const state = getState(); const { conference } = state['features/base/conference']; const { enabled: audioOnly } = state['features/base/audio-only']; const { appState } = state['features/background'] || {}; const { enabled: filmStripEnabled } = state['features/filmstrip']; const config = state['features/base/config']; const { lastNLimits } = state['features/base/lastn']; const participantCount = getParticipantCount(state); if (!conference) { logger.debug('There is no active conference, not updating last N'); return; } let lastN = typeof config.channelLastN === 'undefined' ? -1 : config.channelLastN; // Apply last N limit based on the # of participants and channelLastN settings. const limitedLastN = limitLastN(participantCount, lastNLimits); if (limitedLastN !== undefined) { lastN = lastN === -1 ? limitedLastN : Math.min(limitedLastN, lastN); } if (typeof appState !== 'undefined' && appState !== 'active') { lastN = isLocalVideoTrackDesktop(state) ? 1 : 0; } else if (audioOnly) { const { remoteScreenShares, tileViewEnabled } = state['features/video-layout']; const largeVideoParticipantId = state['features/large-video'].participantId; const largeVideoParticipant = largeVideoParticipantId ? getParticipantById(state, largeVideoParticipantId) : undefined; // Use tileViewEnabled state from redux here instead of determining if client should be in tile // view since we make an exception only for screenshare when in audio-only mode. If the user unpins // the screenshare, lastN will be set to 0 here. It will be set to 1 if screenshare has been auto pinned. if (!tileViewEnabled && largeVideoParticipant && !largeVideoParticipant.local) { lastN = (remoteScreenShares || []).includes(largeVideoParticipantId) ? 1 : 0; } else { lastN = 0; } } else if (!filmStripEnabled) { lastN = 1; } if (conference.getLastN() === lastN) { return; } logger.info(`Setting last N to: ${lastN}`); try { conference.setLastN(lastN); } catch (err) { logger.error(`Failed to set lastN: ${err}`); } }