// @flow import type { Dispatch } from 'redux'; import { getInviteURL } from '../base/connection'; import { inviteVideoRooms } from '../videosipgw'; import { getParticipants } from '../base/participants'; import { ADD_PENDING_INVITE_REQUEST, BEGIN_ADD_PEOPLE, REMOVE_PENDING_INVITE_REQUESTS, SET_CALLEE_INFO_VISIBLE, SET_DIAL_IN_SUMMARY_VISIBLE, SET_INVITE_DIALOG_VISIBLE, UPDATE_DIAL_IN_NUMBERS_FAILED, UPDATE_DIAL_IN_NUMBERS_SUCCESS } from './actionTypes'; import { getDialInConferenceID, getDialInNumbers, invitePeopleAndChatRooms } from './functions'; import logger from './logger'; /** * Creates a (redux) action to signal that a click/tap has been performed on * {@link InviteButton} and that the execution flow for adding/inviting people * to the current conference/meeting is to begin. * * @returns {{ * type: BEGIN_ADD_PEOPLE * }} */ export function beginAddPeople() { return { type: BEGIN_ADD_PEOPLE }; } /** * Invites (i.e. Sends invites to) an array of invitees (which may be a * combination of users, rooms, phone numbers, and video rooms. * * @param {Array} invitees - The recepients to send invites to. * @param {Array} showCalleeInfo - Indicates whether the * {@code CalleeInfo} should be displayed or not. * @returns {Promise>} A {@code Promise} resolving with an array * of invitees who were not invited (i.e. Invites were not sent to them). */ export function invite( invitees: Array, showCalleeInfo: boolean = false) { return ( dispatch: Dispatch, getState: Function): Promise> => { const state = getState(); const participants = getParticipants(state); const { calleeInfoVisible } = state['features/invite']; if (showCalleeInfo && !calleeInfoVisible && invitees.length === 1 && invitees[0].type === 'user' && participants.length === 1) { dispatch(setCalleeInfoVisible(true, invitees[0])); } const { conference } = state['features/base/conference']; if (typeof conference === 'undefined') { // Invite will fail before CONFERENCE_JOIN. The request will be // cached in order to be executed on CONFERENCE_JOIN. return new Promise(resolve => { dispatch(addPendingInviteRequest({ invitees, callback: failedInvitees => resolve(failedInvitees) })); }); } let allInvitePromises = []; let invitesLeftToSend = [ ...invitees ]; const { callFlowsEnabled, inviteServiceUrl, inviteServiceCallFlowsUrl } = state['features/base/config']; const inviteUrl = getInviteURL(state); const { jwt } = state['features/base/jwt']; // First create all promises for dialing out. const phoneNumbers = invitesLeftToSend.filter(({ type }) => type === 'phone'); // For each number, dial out. On success, remove the number from // {@link invitesLeftToSend}. const phoneInvitePromises = phoneNumbers.map(item => { const numberToInvite = item.number; return conference.dial(numberToInvite) .then(() => { invitesLeftToSend = invitesLeftToSend.filter( invitee => invitee !== item); }) .catch(error => logger.error('Error inviting phone number:', error)); }); allInvitePromises = allInvitePromises.concat(phoneInvitePromises); const usersAndRooms = invitesLeftToSend.filter( ({ type }) => type === 'user' || type === 'room'); if (usersAndRooms.length) { // Send a request to invite all the rooms and users. On success, // filter all rooms and users from {@link invitesLeftToSend}. const peopleInvitePromise = invitePeopleAndChatRooms( callFlowsEnabled ? inviteServiceCallFlowsUrl : inviteServiceUrl, inviteUrl, jwt, usersAndRooms) .then(() => { invitesLeftToSend = invitesLeftToSend.filter( ({ type }) => type !== 'user' && type !== 'room'); }) .catch(error => { dispatch(setCalleeInfoVisible(false)); logger.error('Error inviting people:', error); }); allInvitePromises.push(peopleInvitePromise); } // Sipgw calls are fire and forget. Invite them to the conference, then // immediately remove them from invitesLeftToSend. const vrooms = invitesLeftToSend.filter(({ type }) => type === 'videosipgw'); conference && vrooms.length > 0 && dispatch(inviteVideoRooms(conference, vrooms)); invitesLeftToSend = invitesLeftToSend.filter(({ type }) => type !== 'videosipgw'); return ( Promise.all(allInvitePromises) .then(() => invitesLeftToSend)); }; } /** * Sends AJAX requests for dial-in numbers and conference ID. * * @returns {Function} */ export function updateDialInNumbers() { return (dispatch: Dispatch, getState: Function) => { const state = getState(); const { dialInConfCodeUrl, dialInNumbersUrl, hosts } = state['features/base/config']; const mucURL = hosts && hosts.muc; if (!dialInConfCodeUrl || !dialInNumbersUrl || !mucURL) { // URLs for fetching dial in numbers not defined return; } const { room } = state['features/base/conference']; Promise.all([ getDialInNumbers(dialInNumbersUrl, room, mucURL), getDialInConferenceID(dialInConfCodeUrl, room, mucURL) ]) .then(([ dialInNumbers, { conference, id, message } ]) => { if (!conference || !id) { return Promise.reject(message); } dispatch({ type: UPDATE_DIAL_IN_NUMBERS_SUCCESS, conferenceID: id, dialInNumbers }); }) .catch(error => { dispatch({ type: UPDATE_DIAL_IN_NUMBERS_FAILED, error }); }); }; } /** * Sets the visibility of the invite dialog. * * @param {boolean} visible - The visibility to set. * @returns {{ * type: SET_INVITE_DIALOG_VISIBLE, * visible: boolean * }} */ export function setAddPeopleDialogVisible(visible: boolean) { return { type: SET_INVITE_DIALOG_VISIBLE, visible }; } /** * Sets the visibility of {@code CalleeInfo}. * * @param {boolean|undefined} [calleeInfoVisible] - If {@code CalleeInfo} is * to be displayed/visible, then {@code true}; otherwise, {@code false} or * {@code undefined}. * @param {Object|undefined} [initialCalleeInfo] - Callee information. * @returns {{ * type: SET_CALLEE_INFO_VISIBLE, * calleeInfoVisible: (boolean|undefined), * initialCalleeInfo * }} */ export function setCalleeInfoVisible( calleeInfoVisible: boolean, initialCalleeInfo: ?Object) { return { type: SET_CALLEE_INFO_VISIBLE, calleeInfoVisible, initialCalleeInfo }; } /** * Adds pending invite request. * * @param {Object} request - The request. * @returns {{ * type: ADD_PENDING_INVITE_REQUEST, * request: Object * }} */ export function addPendingInviteRequest( request: { invitees: Array, callback: Function }) { return { type: ADD_PENDING_INVITE_REQUEST, request }; } /** * Action to hide the dial in summary. * * @returns {showDialInSummary} */ export function hideDialInSummary() { return showDialInSummary(undefined); } /** * Removes all pending invite requests. * * @returns {{ * type: REMOVE_PENDING_INVITE_REQUEST * }} */ export function removePendingInviteRequests() { return { type: REMOVE_PENDING_INVITE_REQUESTS }; } /** * Action to set the dial in summary url (and show it). * * @param {?string} locationUrl - The location URL to show the dial in summary for. * @returns {{ * type: SET_DIAL_IN_SUMMARY_VISIBLE, * summaryUrl: ?string * }} */ export function showDialInSummary(locationUrl: ?string) { return { type: SET_DIAL_IN_SUMMARY_VISIBLE, summaryUrl: locationUrl }; }