This commit is contained in:
278
react/features/external-api/middleware.ts
Normal file
278
react/features/external-api/middleware.ts
Normal file
@@ -0,0 +1,278 @@
|
||||
// @ts-expect-error
|
||||
import { getJitsiMeetTransport } from '../../../modules/transport';
|
||||
import {
|
||||
CONFERENCE_FAILED,
|
||||
CONFERENCE_JOINED,
|
||||
DATA_CHANNEL_CLOSED,
|
||||
DATA_CHANNEL_OPENED,
|
||||
KICKED_OUT
|
||||
} from '../base/conference/actionTypes';
|
||||
import { SET_CONFIG } from '../base/config/actionTypes';
|
||||
import { NOTIFY_CAMERA_ERROR, NOTIFY_MIC_ERROR } from '../base/devices/actionTypes';
|
||||
import { JitsiConferenceErrors } from '../base/lib-jitsi-meet';
|
||||
import {
|
||||
DOMINANT_SPEAKER_CHANGED,
|
||||
PARTICIPANT_JOINED,
|
||||
PARTICIPANT_KICKED,
|
||||
PARTICIPANT_LEFT,
|
||||
PARTICIPANT_ROLE_CHANGED,
|
||||
SET_LOADABLE_AVATAR_URL
|
||||
} from '../base/participants/actionTypes';
|
||||
import {
|
||||
getDominantSpeakerParticipant,
|
||||
getLocalParticipant,
|
||||
getParticipantById,
|
||||
getParticipantDisplayName
|
||||
} from '../base/participants/functions';
|
||||
import MiddlewareRegistry from '../base/redux/MiddlewareRegistry';
|
||||
import { getBaseUrl } from '../base/util/helpers';
|
||||
import { appendSuffix } from '../display-name/functions';
|
||||
import { SUBMIT_FEEDBACK_ERROR, SUBMIT_FEEDBACK_SUCCESS } from '../feedback/actionTypes';
|
||||
import { SET_FILMSTRIP_VISIBLE } from '../filmstrip/actionTypes';
|
||||
import { iAmVisitor } from '../visitors/functions';
|
||||
|
||||
import './subscriber';
|
||||
|
||||
/**
|
||||
* The middleware of the feature {@code external-api}.
|
||||
*
|
||||
* @returns {Function}
|
||||
*/
|
||||
MiddlewareRegistry.register(store => next => action => {
|
||||
// We need to do these before executing the rest of the middelware chain
|
||||
switch (action.type) {
|
||||
case DOMINANT_SPEAKER_CHANGED: {
|
||||
const dominantSpeaker = getDominantSpeakerParticipant(store.getState());
|
||||
|
||||
if (dominantSpeaker?.id !== action.participant.id) {
|
||||
const result = next(action);
|
||||
|
||||
APP.API.notifyDominantSpeakerChanged(action.participant.id);
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
case SET_LOADABLE_AVATAR_URL: {
|
||||
const { id, loadableAvatarUrl } = action.participant;
|
||||
const participant = getParticipantById(
|
||||
store.getState(),
|
||||
id
|
||||
);
|
||||
|
||||
const result = next(action);
|
||||
|
||||
if (participant) {
|
||||
if (loadableAvatarUrl) {
|
||||
participant.loadableAvatarUrl !== loadableAvatarUrl && APP.API.notifyAvatarChanged(
|
||||
id,
|
||||
loadableAvatarUrl
|
||||
);
|
||||
} else {
|
||||
// There is no loadable explicit URL. In this case the Avatar component would
|
||||
// decide to render initials or the default avatar, but the external API needs
|
||||
// a URL when it needs to be rendered, so if there is no initials, we return the default
|
||||
// Avatar URL as if it was a usual avatar URL. If there are (or may be) initials
|
||||
// we send undefined to signal the api user that it's not an URL that needs to be rendered.
|
||||
//
|
||||
// NOTE: we may implement a special URL format later to signal that the avatar is based
|
||||
// on initials, that API consumers can handle as they want, e.g. initials://jm
|
||||
APP.API.notifyAvatarChanged(
|
||||
id,
|
||||
participant.name ? undefined : _getDefaultAvatarUrl()
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return result;
|
||||
}
|
||||
}
|
||||
|
||||
const result = next(action);
|
||||
|
||||
// These should happen after the rest of the middleware chain ran
|
||||
switch (action.type) {
|
||||
case CONFERENCE_FAILED: {
|
||||
if (action.conference
|
||||
&& action.error.name === JitsiConferenceErrors.PASSWORD_REQUIRED) {
|
||||
APP.API.notifyOnPasswordRequired();
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
case CONFERENCE_JOINED: {
|
||||
const state = store.getState();
|
||||
const { defaultLocalDisplayName } = state['features/base/config'];
|
||||
const { room } = state['features/base/conference'];
|
||||
const { loadableAvatarUrl, name, id, email } = getLocalParticipant(state) ?? {};
|
||||
const breakoutRoom = APP.conference.roomName.toString() !== room?.toLowerCase();
|
||||
|
||||
// we use APP.conference.roomName as we do not update state['features/base/conference'].room when
|
||||
// moving between rooms in case of breakout rooms and it stays always with the name of the main room
|
||||
APP.API.notifyConferenceJoined(
|
||||
APP.conference.roomName,
|
||||
id,
|
||||
{
|
||||
displayName: name,
|
||||
formattedDisplayName: appendSuffix(
|
||||
name ?? '',
|
||||
defaultLocalDisplayName
|
||||
),
|
||||
avatarURL: loadableAvatarUrl,
|
||||
breakoutRoom,
|
||||
email,
|
||||
visitor: iAmVisitor(state)
|
||||
}
|
||||
);
|
||||
break;
|
||||
}
|
||||
|
||||
case DATA_CHANNEL_CLOSED:
|
||||
APP.API.notifyDataChannelClosed(action.code, action.reason);
|
||||
break;
|
||||
|
||||
case DATA_CHANNEL_OPENED:
|
||||
APP.API.notifyDataChannelOpened();
|
||||
break;
|
||||
|
||||
case KICKED_OUT: {
|
||||
const state = store.getState();
|
||||
const localParticipant = getLocalParticipant(state);
|
||||
|
||||
if (!localParticipant) {
|
||||
break;
|
||||
}
|
||||
|
||||
const actor = action.participant;
|
||||
|
||||
APP.API.notifyKickedOut(
|
||||
{
|
||||
id: localParticipant.id,
|
||||
name: getParticipantDisplayName(state, localParticipant.id),
|
||||
local: true
|
||||
},
|
||||
{
|
||||
id: actor?.getId(),
|
||||
name: actor?.getDisplayName()
|
||||
}
|
||||
);
|
||||
break;
|
||||
}
|
||||
|
||||
case NOTIFY_CAMERA_ERROR:
|
||||
if (action.error) {
|
||||
APP.API.notifyOnCameraError(
|
||||
action.error.name, action.error.message);
|
||||
}
|
||||
break;
|
||||
|
||||
case NOTIFY_MIC_ERROR:
|
||||
if (action.error) {
|
||||
APP.API.notifyOnMicError(action.error.name, action.error.message);
|
||||
}
|
||||
break;
|
||||
|
||||
case PARTICIPANT_KICKED: {
|
||||
const state = store.getState();
|
||||
const kickedParticipant = getParticipantById(state, action.kicked);
|
||||
const kickerParticipant = getParticipantById(state, action.kicker);
|
||||
|
||||
if (!kickerParticipant || !kickedParticipant) {
|
||||
break;
|
||||
}
|
||||
|
||||
APP.API.notifyKickedOut(
|
||||
{
|
||||
id: kickedParticipant.id,
|
||||
local: kickedParticipant.local,
|
||||
name: getParticipantDisplayName(state, kickedParticipant.id)
|
||||
},
|
||||
{
|
||||
id: kickerParticipant.id,
|
||||
local: kickerParticipant.local,
|
||||
name: getParticipantDisplayName(state, kickerParticipant.id)
|
||||
});
|
||||
break;
|
||||
}
|
||||
|
||||
case PARTICIPANT_LEFT: {
|
||||
const { participant } = action;
|
||||
const { fakeParticipant } = participant;
|
||||
|
||||
// Skip sending participant left event for fake participants.
|
||||
if (fakeParticipant) {
|
||||
break;
|
||||
}
|
||||
|
||||
APP.API.notifyUserLeft(action.participant.id);
|
||||
break;
|
||||
}
|
||||
case PARTICIPANT_JOINED: {
|
||||
const state = store.getState();
|
||||
const { defaultRemoteDisplayName } = state['features/base/config'];
|
||||
const { participant } = action;
|
||||
const { fakeParticipant, id, local, name } = participant;
|
||||
|
||||
// The version of external api outside of middleware did not emit
|
||||
// the local participant being created.
|
||||
if (!local) {
|
||||
// Skip sending participant joined event for fake participants.
|
||||
if (fakeParticipant) {
|
||||
break;
|
||||
}
|
||||
|
||||
APP.API.notifyUserJoined(id, {
|
||||
displayName: name,
|
||||
formattedDisplayName: appendSuffix(
|
||||
name || defaultRemoteDisplayName)
|
||||
});
|
||||
}
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case PARTICIPANT_ROLE_CHANGED:
|
||||
APP.API.notifyUserRoleChanged(action.participant.id, action.participant.role);
|
||||
break;
|
||||
|
||||
case SET_CONFIG: {
|
||||
const state = store.getState();
|
||||
const { disableBeforeUnloadHandlers = false } = state['features/base/config'];
|
||||
|
||||
/**
|
||||
* Disposing the API when the user closes the page.
|
||||
*/
|
||||
window.addEventListener(disableBeforeUnloadHandlers ? 'unload' : 'beforeunload', () => {
|
||||
APP.API.notifyConferenceLeft(APP.conference.roomName);
|
||||
APP.API.dispose();
|
||||
getJitsiMeetTransport().dispose();
|
||||
});
|
||||
|
||||
break;
|
||||
}
|
||||
|
||||
case SET_FILMSTRIP_VISIBLE:
|
||||
APP.API.notifyFilmstripDisplayChanged(action.visible);
|
||||
break;
|
||||
|
||||
case SUBMIT_FEEDBACK_ERROR:
|
||||
APP.API.notifyFeedbackSubmitted(action.error || 'Unknown error');
|
||||
break;
|
||||
|
||||
case SUBMIT_FEEDBACK_SUCCESS:
|
||||
APP.API.notifyFeedbackSubmitted();
|
||||
break;
|
||||
}
|
||||
|
||||
return result;
|
||||
});
|
||||
|
||||
/**
|
||||
* Returns the absolute URL of the default avatar.
|
||||
*
|
||||
* @returns {string}
|
||||
*/
|
||||
function _getDefaultAvatarUrl() {
|
||||
return new URL('images/avatar.png', getBaseUrl()).href;
|
||||
}
|
||||
73
react/features/external-api/subscriber.ts
Normal file
73
react/features/external-api/subscriber.ts
Normal file
@@ -0,0 +1,73 @@
|
||||
import { getLocalParticipant } from '../base/participants/functions';
|
||||
import StateListenerRegistry from '../base/redux/StateListenerRegistry';
|
||||
import { appendSuffix } from '../display-name/functions';
|
||||
import { shouldDisplayTileView } from '../video-layout/functions';
|
||||
|
||||
/**
|
||||
* StateListenerRegistry provides a reliable way of detecting changes to
|
||||
* preferred layout state and dispatching additional actions.
|
||||
*/
|
||||
StateListenerRegistry.register(
|
||||
/* selector */ state => shouldDisplayTileView(state),
|
||||
/* listener */ displayTileView => {
|
||||
APP.API.notifyTileViewChanged(displayTileView);
|
||||
});
|
||||
|
||||
StateListenerRegistry.register(
|
||||
/* selector */ state => state['features/base/settings'].displayName,
|
||||
/* listener */ (displayName, store) => {
|
||||
const localParticipant = getLocalParticipant(store.getState());
|
||||
const { defaultLocalDisplayName } = store.getState()['features/base/config'];
|
||||
|
||||
// Initial setting of the display name happens on app
|
||||
// initialization, before the local participant is ready. The initial
|
||||
// settings is not desired to be fired anyways, only changes.
|
||||
if (localParticipant) {
|
||||
const { id } = localParticipant;
|
||||
|
||||
APP.API.notifyDisplayNameChanged(id, {
|
||||
displayName,
|
||||
formattedDisplayName: appendSuffix(
|
||||
displayName,
|
||||
defaultLocalDisplayName
|
||||
)
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
StateListenerRegistry.register(
|
||||
/* selector */ state => state['features/base/settings'].email,
|
||||
/* listener */ (email, store) => {
|
||||
const localParticipant = getLocalParticipant(store.getState());
|
||||
|
||||
// Initial setting of the email happens on app
|
||||
// initialization, before the local participant is ready. The initial
|
||||
// settings is not desired to be fired anyways, only changes.
|
||||
if (localParticipant) {
|
||||
const { id } = localParticipant;
|
||||
|
||||
APP.API.notifyEmailChanged(id, {
|
||||
email
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Updates the on stage participant value.
|
||||
*/
|
||||
StateListenerRegistry.register(
|
||||
/* selector */ state => state['features/large-video'].participantId,
|
||||
/* listener */ participantId => {
|
||||
APP.API.notifyOnStageParticipantChanged(participantId);
|
||||
}
|
||||
);
|
||||
|
||||
/**
|
||||
* Updates the on audio only value.
|
||||
*/
|
||||
StateListenerRegistry.register(
|
||||
/* selector */ state => state['features/base/audio-only'].enabled,
|
||||
/* listener */ enabled => {
|
||||
APP.API.notifyAudioOnlyChanged(enabled);
|
||||
}
|
||||
);
|
||||
Reference in New Issue
Block a user