123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239 |
- // @flow
-
- import i18next from 'i18next';
- import { NativeEventEmitter, NativeModules } from 'react-native';
-
- import { MiddlewareRegistry } from '../base/redux';
- import { APP_WILL_MOUNT, APP_WILL_UNMOUNT } from '../app';
-
- import { invite } from './actions';
- import {
- BEGIN_ADD_PEOPLE,
- _SET_EMITTER_SUBSCRIPTIONS
- } from './actionTypes';
- import {
- getInviteResultsForQuery,
- isAddPeopleEnabled,
- isDialOutEnabled
- } from './functions';
- import './middleware.any';
-
- /**
- * The react-native module of the feature invite.
- */
- const { Invite } = NativeModules;
-
- /**
- * The middleware of the feature invite specific to mobile/react-native.
- *
- * @param {Store} store - The redux store.
- * @returns {Function}
- */
- Invite && MiddlewareRegistry.register(store => next => action => {
- switch (action.type) {
- case _SET_EMITTER_SUBSCRIPTIONS:
- return _setEmitterSubscriptions(store, next, action);
-
- case APP_WILL_MOUNT:
- return _appWillMount(store, next, action);
-
- case APP_WILL_UNMOUNT: {
- const result = next(action);
-
- store.dispatch({
- type: _SET_EMITTER_SUBSCRIPTIONS,
- emitterSubscriptions: undefined
- });
-
- return result;
- }
-
- case BEGIN_ADD_PEOPLE:
- return _beginAddPeople(store, next, action);
- }
-
- return next(action);
- });
-
- /**
- * Notifies the feature jwt that the action {@link APP_WILL_MOUNT} is being
- * dispatched within a specific redux {@code store}.
- *
- * @param {Store} store - The redux store in which the specified {@code action}
- * is being dispatched.
- * @param {Dispatch} next - The redux dispatch function to dispatch the
- * specified {@code action} to the specified {@code store}.
- * @param {Action} action - The redux action {@code APP_WILL_MOUNT} which is
- * being dispatched in the specified {@code store}.
- * @private
- * @returns {*} The value returned by {@code next(action)}.
- */
- function _appWillMount({ dispatch, getState }, next, action) {
- const result = next(action);
-
- const emitter = new NativeEventEmitter(Invite);
- const context = {
- dispatch,
- getState
- };
-
- dispatch({
- type: _SET_EMITTER_SUBSCRIPTIONS,
- emitterSubscriptions: [
- emitter.addListener(
- 'org.jitsi.meet:features/invite#invite',
- _onInvite,
- context),
- emitter.addListener(
- 'org.jitsi.meet:features/invite#performQuery',
- _onPerformQuery,
- context)
- ]
- });
-
- return result;
- }
-
- /**
- * Notifies the feature invite that the action {@link BEGIN_ADD_PEOPLE} is being
- * dispatched within a specific redux {@code store}.
- *
- * @param {Store} store - The redux store in which the specified {@code action}
- * is being dispatched.
- * @param {Dispatch} next - The redux {@code dispatch} function to dispatch the
- * specified {@code action} to the specified {@code store}.
- * @param {Action} action - The redux action {@code BEGIN_ADD_PEOPLE} which is
- * being dispatched in the specified {@code store}.
- * @private
- * @returns {*} The value returned by {@code next(action)}.
- */
- function _beginAddPeople({ getState }, next, action) {
- const result = next(action);
-
- // The JavaScript App needs to provide uniquely identifying information to
- // the native Invite module so that the latter may match the former to the
- // native JitsiMeetView which hosts it.
- const { app } = getState()['features/app'];
-
- if (app) {
- const { externalAPIScope } = app.props;
-
- if (externalAPIScope) {
- Invite.beginAddPeople(externalAPIScope);
- }
- }
-
- return result;
- }
-
- /**
- * Handles the {@code invite} event of the feature invite i.e. invites specific
- * invitees to the current, ongoing conference.
- *
- * @param {Object} event - The details of the event.
- * @returns {void}
- */
- function _onInvite({ addPeopleControllerScope, externalAPIScope, invitees }) {
- const { dispatch, getState } = this; // eslint-disable-line no-invalid-this
-
- // If there are multiple JitsiMeetView instances alive, they will all get
- // the event, since there is a single bridge, so make sure we don't act if
- // the event is not for us.
- if (getState()['features/app'].app.props.externalAPIScope
- !== externalAPIScope) {
- return;
- }
-
- dispatch(invite(invitees))
- .then(failedInvitees =>
- Invite.inviteSettled(
- externalAPIScope,
- addPeopleControllerScope,
- failedInvitees));
- }
-
- /**
- * Handles the {@code performQuery} event of the feature invite i.e. queries for
- * invitees who may subsequently be invited to the current, ongoing conference.
- *
- * @param {Object} event - The details of the event.
- * @returns {void}
- */
- function _onPerformQuery(
- { addPeopleControllerScope, externalAPIScope, query }) {
- const { getState } = this; // eslint-disable-line no-invalid-this
- const state = getState();
-
- // If there are multiple JitsiMeetView instances alive, they will all get
- // the event, since there is a single bridge, so make sure we don't act if
- // the event is not for us.
- if (state['features/app'].app.props.externalAPIScope !== externalAPIScope) {
- return;
- }
-
- const {
- dialOutAuthUrl,
- peopleSearchQueryTypes,
- peopleSearchUrl
- } = state['features/base/config'];
- const options = {
- dialOutAuthUrl,
- addPeopleEnabled: isAddPeopleEnabled(state),
- dialOutEnabled: isDialOutEnabled(state),
- jwt: state['features/base/jwt'].jwt,
- peopleSearchQueryTypes,
- peopleSearchUrl
- };
-
- getInviteResultsForQuery(query, options)
- .catch(() => [])
- .then(results => {
- const translatedResults = results.map(result => {
- if (result.type === 'phone') {
- result.title = i18next.t('addPeople.telephone', {
- number: result.number
- });
-
- if (result.showCountryCodeReminder) {
- result.subtitle = i18next.t(
- 'addPeople.countryReminder'
- );
- }
- }
-
- return result;
- }).filter(result => result.type !== 'phone' || result.allowed);
-
- Invite.receivedResults(
- externalAPIScope,
- addPeopleControllerScope,
- query,
- translatedResults);
- });
- }
-
- /**
- * Notifies the feature invite that the action
- * {@link _SET_EMITTER_SUBSCRIPTIONS} is being dispatched within a specific
- * redux {@code store}.
- *
- * @param {Store} store - The redux store in which the specified {@code action}
- * is being dispatched.
- * @param {Dispatch} next - The redux dispatch function to dispatch the
- * specified {@code action} to the specified {@code store}.
- * @param {Action} action - The redux action {@code _SET_EMITTER_SUBSCRIPTIONS}
- * which is being dispatched in the specified {@code store}.
- * @private
- * @returns {*}
- */
- function _setEmitterSubscriptions({ getState }, next, action) {
- const { emitterSubscriptions } = getState()['features/invite'];
-
- if (emitterSubscriptions) {
- for (const subscription of emitterSubscriptions) {
- subscription.remove();
- }
- }
-
- return next(action);
- }
|