307 lines
10 KiB
TypeScript
307 lines
10 KiB
TypeScript
import i18next from 'i18next';
|
|
import _ from 'lodash';
|
|
|
|
import { createBreakoutRoomsEvent } from '../analytics/AnalyticsEvents';
|
|
import { sendAnalytics } from '../analytics/functions';
|
|
import { IStore } from '../app/types';
|
|
import {
|
|
conferenceLeft,
|
|
conferenceWillLeave,
|
|
createConference
|
|
} from '../base/conference/actions';
|
|
import { CONFERENCE_LEAVE_REASONS } from '../base/conference/constants';
|
|
import { getCurrentConference } from '../base/conference/functions';
|
|
import { setAudioMuted, setVideoMuted } from '../base/media/actions';
|
|
import { MEDIA_TYPE } from '../base/media/constants';
|
|
import { getRemoteParticipants } from '../base/participants/functions';
|
|
import { createDesiredLocalTracks } from '../base/tracks/actions';
|
|
import {
|
|
getLocalTracks,
|
|
isLocalCameraTrackMuted,
|
|
isLocalTrackMuted
|
|
} from '../base/tracks/functions';
|
|
import { clearNotifications, showNotification } from '../notifications/actions';
|
|
import { NOTIFICATION_TIMEOUT_TYPE } from '../notifications/constants';
|
|
|
|
import { _RESET_BREAKOUT_ROOMS, _UPDATE_ROOM_COUNTER } from './actionTypes';
|
|
import { FEATURE_KEY } from './constants';
|
|
import {
|
|
getBreakoutRooms,
|
|
getMainRoom,
|
|
getRoomByJid
|
|
} from './functions';
|
|
import logger from './logger';
|
|
|
|
/**
|
|
* Action to create a breakout room.
|
|
*
|
|
* @param {string} name - Name / subject for the breakout room.
|
|
* @returns {Function}
|
|
*/
|
|
export function createBreakoutRoom(name?: string) {
|
|
return (dispatch: IStore['dispatch'], getState: IStore['getState']) => {
|
|
const state = getState();
|
|
let { roomCounter } = state[FEATURE_KEY];
|
|
const subject = name || i18next.t('breakoutRooms.defaultName', { index: ++roomCounter });
|
|
|
|
sendAnalytics(createBreakoutRoomsEvent('create'));
|
|
|
|
dispatch({
|
|
type: _UPDATE_ROOM_COUNTER,
|
|
roomCounter
|
|
});
|
|
|
|
getCurrentConference(state)?.getBreakoutRooms()
|
|
?.createBreakoutRoom(subject);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Action to close a room and send participants to the main room.
|
|
*
|
|
* @param {string} roomId - The id of the room to close.
|
|
* @returns {Function}
|
|
*/
|
|
export function closeBreakoutRoom(roomId: string) {
|
|
return (dispatch: IStore['dispatch'], getState: IStore['getState']) => {
|
|
const rooms = getBreakoutRooms(getState);
|
|
const room = rooms[roomId];
|
|
const mainRoom = getMainRoom(getState);
|
|
|
|
sendAnalytics(createBreakoutRoomsEvent('close'));
|
|
|
|
if (room && mainRoom) {
|
|
Object.values(room.participants).forEach(p => {
|
|
dispatch(sendParticipantToRoom(p.jid, mainRoom.id));
|
|
});
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Action to remove a breakout room.
|
|
*
|
|
* @param {string} breakoutRoomJid - The jid of the breakout room to remove.
|
|
* @returns {Function}
|
|
*/
|
|
export function removeBreakoutRoom(breakoutRoomJid: string) {
|
|
return (dispatch: IStore['dispatch'], getState: IStore['getState']) => {
|
|
sendAnalytics(createBreakoutRoomsEvent('remove'));
|
|
const room = getRoomByJid(getState, breakoutRoomJid);
|
|
|
|
if (!room) {
|
|
logger.error('The room to remove was not found.');
|
|
|
|
return;
|
|
}
|
|
|
|
if (Object.keys(room.participants).length > 0) {
|
|
dispatch(closeBreakoutRoom(room.id));
|
|
}
|
|
getCurrentConference(getState)?.getBreakoutRooms()
|
|
?.removeBreakoutRoom(breakoutRoomJid);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Action to auto-assign the participants to breakout rooms.
|
|
*
|
|
* @returns {Function}
|
|
*/
|
|
export function autoAssignToBreakoutRooms() {
|
|
return (dispatch: IStore['dispatch'], getState: IStore['getState']) => {
|
|
const rooms = getBreakoutRooms(getState);
|
|
const breakoutRooms = _.filter(rooms, room => !room.isMainRoom);
|
|
|
|
if (breakoutRooms) {
|
|
sendAnalytics(createBreakoutRoomsEvent('auto.assign'));
|
|
const participantIds = Array.from(getRemoteParticipants(getState).keys());
|
|
const length = Math.ceil(participantIds.length / breakoutRooms.length);
|
|
|
|
_.chunk(_.shuffle(participantIds), length).forEach((group, index) =>
|
|
group.forEach(participantId => {
|
|
dispatch(sendParticipantToRoom(participantId, breakoutRooms[index].id));
|
|
})
|
|
);
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Action to send a participant to a room.
|
|
*
|
|
* @param {string} participantId - The participant id.
|
|
* @param {string} roomId - The room id.
|
|
* @returns {Function}
|
|
*/
|
|
export function sendParticipantToRoom(participantId: string, roomId: string) {
|
|
return (dispatch: IStore['dispatch'], getState: IStore['getState']) => {
|
|
const rooms = getBreakoutRooms(getState);
|
|
const room = rooms[roomId];
|
|
|
|
if (!room) {
|
|
logger.warn(`Invalid room: ${roomId}`);
|
|
|
|
return;
|
|
}
|
|
|
|
// Get the full JID of the participant. We could be getting the endpoint ID or
|
|
// a participant JID. We want to find the connection JID.
|
|
const participantJid = _findParticipantJid(getState, participantId);
|
|
|
|
if (!participantJid) {
|
|
logger.warn(`Could not find participant ${participantId}`);
|
|
|
|
return;
|
|
}
|
|
|
|
getCurrentConference(getState)?.getBreakoutRooms()
|
|
?.sendParticipantToRoom(participantJid, room.jid);
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Action to move to a room.
|
|
*
|
|
* @param {string} roomId - The room id to move to. If omitted move to the main room.
|
|
* @returns {Function}
|
|
*/
|
|
export function moveToRoom(roomId?: string) {
|
|
return async (dispatch: IStore['dispatch'], getState: IStore['getState']) => {
|
|
const mainRoomId = getMainRoom(getState)?.id;
|
|
let _roomId: string | undefined | String = roomId || mainRoomId;
|
|
|
|
// Check if we got a full JID.
|
|
if (_roomId && _roomId?.indexOf('@') !== -1) {
|
|
const [ id, ...domainParts ] = _roomId.split('@');
|
|
|
|
// On mobile we first store the room and the connection is created
|
|
// later, so let's attach the domain to the room String object as
|
|
// a little hack.
|
|
|
|
// eslint-disable-next-line no-new-wrappers
|
|
_roomId = new String(id);
|
|
|
|
// @ts-ignore
|
|
_roomId.domain = domainParts.join('@');
|
|
}
|
|
|
|
const roomIdStr = _roomId?.toString();
|
|
const goToMainRoom = roomIdStr === mainRoomId;
|
|
const rooms = getBreakoutRooms(getState);
|
|
const targetRoom = rooms[roomIdStr ?? ''];
|
|
|
|
if (!targetRoom) {
|
|
logger.warn(`Unknown room: ${targetRoom}`);
|
|
|
|
return;
|
|
}
|
|
|
|
dispatch({
|
|
type: _RESET_BREAKOUT_ROOMS
|
|
});
|
|
|
|
if (navigator.product === 'ReactNative') {
|
|
const conference = getCurrentConference(getState);
|
|
const { audio, video } = getState()['features/base/media'];
|
|
|
|
dispatch(conferenceWillLeave(conference));
|
|
|
|
try {
|
|
await conference.leave(CONFERENCE_LEAVE_REASONS.SWITCH_ROOM);
|
|
} catch (error) {
|
|
logger.warn('JitsiConference.leave() rejected with:', error);
|
|
|
|
dispatch(conferenceLeft(conference));
|
|
}
|
|
|
|
dispatch(clearNotifications());
|
|
|
|
// dispatch(setRoom(_roomId));
|
|
dispatch(createConference(_roomId?.toString()));
|
|
dispatch(setAudioMuted(audio.muted));
|
|
dispatch(setVideoMuted(Boolean(video.muted)));
|
|
dispatch(createDesiredLocalTracks());
|
|
} else {
|
|
const localTracks = getLocalTracks(getState()['features/base/tracks']);
|
|
const isAudioMuted = isLocalTrackMuted(localTracks, MEDIA_TYPE.AUDIO);
|
|
const isVideoMuted = isLocalCameraTrackMuted(localTracks);
|
|
|
|
try {
|
|
// all places we fire notifyConferenceLeft we pass the room name from APP.conference
|
|
await APP.conference.leaveRoom(false /* doDisconnect */, CONFERENCE_LEAVE_REASONS.SWITCH_ROOM).then(
|
|
() => APP.API.notifyConferenceLeft(APP.conference.roomName));
|
|
} catch (error) {
|
|
logger.warn('APP.conference.leaveRoom() rejected with:', error);
|
|
|
|
// TODO: revisit why we don't dispatch CONFERENCE_LEFT here.
|
|
}
|
|
|
|
APP.conference.joinRoom(_roomId, {
|
|
startWithAudioMuted: isAudioMuted,
|
|
startWithVideoMuted: isVideoMuted
|
|
});
|
|
}
|
|
|
|
if (goToMainRoom) {
|
|
dispatch(showNotification({
|
|
titleKey: 'breakoutRooms.notifications.joinedTitle',
|
|
descriptionKey: 'breakoutRooms.notifications.joinedMainRoom',
|
|
concatText: true,
|
|
maxLines: 2
|
|
}, NOTIFICATION_TIMEOUT_TYPE.MEDIUM));
|
|
} else {
|
|
dispatch(showNotification({
|
|
titleKey: 'breakoutRooms.notifications.joinedTitle',
|
|
descriptionKey: 'breakoutRooms.notifications.joined',
|
|
descriptionArguments: { name: targetRoom.name },
|
|
concatText: true,
|
|
maxLines: 2
|
|
}, NOTIFICATION_TIMEOUT_TYPE.MEDIUM));
|
|
}
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Finds a participant's connection JID given its ID.
|
|
*
|
|
* @param {Function} getState - The redux store state getter.
|
|
* @param {string} participantId - ID of the given participant.
|
|
* @returns {string|undefined} - The participant connection JID if found.
|
|
*/
|
|
function _findParticipantJid(getState: Function, participantId: string) {
|
|
const conference = getCurrentConference(getState);
|
|
|
|
if (!conference) {
|
|
return;
|
|
}
|
|
|
|
// Get the full JID of the participant. We could be getting the endpoint ID or
|
|
// a participant JID. We want to find the connection JID.
|
|
let _participantId = participantId;
|
|
let participantJid;
|
|
|
|
if (!participantId.includes('@')) {
|
|
const p = conference.getParticipantById(participantId);
|
|
|
|
_participantId = p?.getJid(); // This will be the room JID.
|
|
}
|
|
|
|
if (_participantId) {
|
|
const rooms = getBreakoutRooms(getState);
|
|
|
|
for (const room of Object.values(rooms)) {
|
|
const participants = room.participants || {};
|
|
const p = participants[_participantId]
|
|
|| Object.values(participants).find(item => item.jid === _participantId);
|
|
|
|
if (p) {
|
|
participantJid = p.jid;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return participantJid;
|
|
}
|