| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398 | 
							- // @flow
 - 
 - import uuid from 'uuid';
 - 
 - import { createTrackMutedEvent, sendAnalytics } from '../../analytics';
 - import {
 -     APP_WILL_MOUNT,
 -     APP_WILL_UNMOUNT,
 -     appNavigate,
 -     getName
 - } from '../../app';
 - import {
 -     CONFERENCE_FAILED,
 -     CONFERENCE_LEFT,
 -     CONFERENCE_WILL_JOIN,
 -     CONFERENCE_JOINED,
 -     SET_AUDIO_ONLY,
 -     getCurrentConference
 - } from '../../base/conference';
 - import { getInviteURL } from '../../base/connection';
 - import {
 -     MEDIA_TYPE,
 -     isVideoMutedByAudioOnly,
 -     setAudioMuted
 - } from '../../base/media';
 - import { MiddlewareRegistry } from '../../base/redux';
 - import {
 -     TRACK_ADDED,
 -     TRACK_REMOVED,
 -     TRACK_UPDATED,
 -     isLocalTrackMuted
 - } from '../../base/tracks';
 - 
 - import { _SET_CALLKIT_SUBSCRIPTIONS } from './actionTypes';
 - import CallKit from './CallKit';
 - 
 - /**
 -  * Middleware that captures system actions and hooks up CallKit.
 -  *
 -  * @param {Store} store - The redux store.
 -  * @returns {Function}
 -  */
 - CallKit && MiddlewareRegistry.register(store => next => action => {
 -     switch (action.type) {
 -     case _SET_CALLKIT_SUBSCRIPTIONS:
 -         return _setCallKitSubscriptions(store, next, action);
 - 
 -     case APP_WILL_MOUNT:
 -         return _appWillMount(store, next, action);
 - 
 -     case APP_WILL_UNMOUNT:
 -         store.dispatch({
 -             type: _SET_CALLKIT_SUBSCRIPTIONS,
 -             subscriptions: undefined
 -         });
 -         break;
 - 
 -     case CONFERENCE_FAILED:
 -         return _conferenceFailed(store, next, action);
 - 
 -     case CONFERENCE_JOINED:
 -         return _conferenceJoined(store, next, action);
 - 
 -     case CONFERENCE_LEFT:
 -         return _conferenceLeft(store, next, action);
 - 
 -     case CONFERENCE_WILL_JOIN:
 -         return _conferenceWillJoin(store, next, action);
 - 
 -     case SET_AUDIO_ONLY:
 -         return _setAudioOnly(store, next, action);
 - 
 -     case TRACK_ADDED:
 -     case TRACK_REMOVED:
 -     case TRACK_UPDATED:
 -         return _syncTrackState(store, next, action);
 -     }
 - 
 -     return next(action);
 - });
 - 
 - /**
 -  * Notifies the feature callkit 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 {@code dispatch} function to dispatch the
 -  * specified {@code action} in 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);
 - 
 -     CallKit.setProviderConfiguration({
 -         iconTemplateImageName: 'CallKitIcon',
 -         localizedName: getName()
 -     });
 - 
 -     const context = {
 -         dispatch,
 -         getState
 -     };
 -     const subscriptions = [
 -         CallKit.addListener(
 -             'performEndCallAction',
 -             _onPerformEndCallAction,
 -             context),
 -         CallKit.addListener(
 -             'performSetMutedCallAction',
 -             _onPerformSetMutedCallAction,
 -             context),
 - 
 -         // According to CallKit's documentation, when the system resets we
 -         // should terminate all calls. Hence, providerDidReset is the same to us
 -         // as performEndCallAction.
 -         CallKit.addListener(
 -             'providerDidReset',
 -             _onPerformEndCallAction,
 -             context)
 -     ];
 - 
 -     dispatch({
 -         type: _SET_CALLKIT_SUBSCRIPTIONS,
 -         subscriptions
 -     });
 - 
 -     return result;
 - }
 - 
 - /**
 -  * Notifies the feature callkit that the action {@link CONFERENCE_FAILED} 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} in the specified {@code store}.
 -  * @param {Action} action - The redux action {@code CONFERENCE_FAILED} which is
 -  * being dispatched in the specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _conferenceFailed(store, next, action) {
 -     const result = next(action);
 - 
 -     // XXX Certain CONFERENCE_FAILED errors are recoverable i.e. they have
 -     // prevented the user from joining a specific conference but the app may be
 -     // able to eventually join the conference.
 -     if (!action.error.recoverable) {
 -         const { callUUID } = action.conference;
 - 
 -         if (callUUID) {
 -             CallKit.reportCallFailed(callUUID);
 -         }
 -     }
 - 
 -     return result;
 - }
 - 
 - /**
 -  * Notifies the feature callkit that the action {@link CONFERENCE_JOINED} 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} in the specified {@code store}.
 -  * @param {Action} action - The redux action {@code CONFERENCE_JOINED} which is
 -  * being dispatched in the specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _conferenceJoined(store, next, action) {
 -     const result = next(action);
 - 
 -     const { callUUID } = action.conference;
 - 
 -     if (callUUID) {
 -         CallKit.reportConnectedOutgoingCall(callUUID);
 -     }
 - 
 -     return result;
 - }
 - 
 - /**
 -  * Notifies the feature callkit that the action {@link CONFERENCE_LEFT} 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} in the specified {@code store}.
 -  * @param {Action} action - The redux action {@code CONFERENCE_LEFT} which is
 -  * being dispatched in the specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _conferenceLeft(store, next, action) {
 -     const result = next(action);
 - 
 -     const { callUUID } = action.conference;
 - 
 -     if (callUUID) {
 -         CallKit.endCall(callUUID);
 -     }
 - 
 -     return result;
 - }
 - 
 - /**
 -  * Notifies the feature callkit that the action {@link CONFERENCE_WILL_JOIN} 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} in the specified {@code store}.
 -  * @param {Action} action - The redux action {@code CONFERENCE_WILL_JOIN} which
 -  * is being dispatched in the specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _conferenceWillJoin({ getState }, next, action) {
 -     const result = next(action);
 - 
 -     const { conference } = action;
 -     const state = getState();
 -     const { callHandle, callUUID } = state['features/base/config'];
 -     const url = getInviteURL(state);
 -     const handle = callHandle || url.toString();
 -     const hasVideo = !isVideoMutedByAudioOnly(state);
 - 
 -     // When assigning the call UUID, do so in upper case, since iOS will return
 -     // it upper cased.
 -     conference.callUUID = (callUUID || uuid.v4()).toUpperCase();
 - 
 -     CallKit.startCall(conference.callUUID, handle, hasVideo)
 -         .then(() => {
 -             const { callee } = state['features/base/jwt'];
 -             const displayName
 -                  = state['features/base/config'].callDisplayName
 -                      || (callee && callee.name)
 -                      || state['features/base/conference'].room;
 - 
 -             const muted
 -                 = isLocalTrackMuted(
 -                     state['features/base/tracks'],
 -                     MEDIA_TYPE.AUDIO);
 - 
 -             // eslint-disable-next-line object-property-newline
 -             CallKit.updateCall(conference.callUUID, { displayName, hasVideo });
 -             CallKit.setMuted(conference.callUUID, muted);
 -         });
 - 
 -     return result;
 - }
 - 
 - /**
 -  * Handles CallKit's event {@code performEndCallAction}.
 -  *
 -  * @param {Object} event - The details of the CallKit event
 -  * {@code performEndCallAction}.
 -  * @returns {void}
 -  */
 - function _onPerformEndCallAction({ callUUID }) {
 -     const { dispatch, getState } = this; // eslint-disable-line no-invalid-this
 -     const conference = getCurrentConference(getState);
 - 
 -     if (conference && conference.callUUID === callUUID) {
 -         // We arrive here when a call is ended by the system, for example, when
 -         // another incoming call is received and the user selects "End &
 -         // Accept".
 -         delete conference.callUUID;
 -         dispatch(appNavigate(undefined));
 -     }
 - }
 - 
 - /**
 -  * Handles CallKit's event {@code performSetMutedCallAction}.
 -  *
 -  * @param {Object} event - The details of the CallKit event
 -  * {@code performSetMutedCallAction}.
 -  * @returns {void}
 -  */
 - function _onPerformSetMutedCallAction({ callUUID, muted: newValue }) {
 -     const { dispatch, getState } = this; // eslint-disable-line no-invalid-this
 -     const conference = getCurrentConference(getState);
 - 
 -     if (conference && conference.callUUID === callUUID) {
 -         // Break the loop. Audio can be muted from both CallKit and Jitsi Meet.
 -         // We must keep them in sync, but at some point the loop needs to be
 -         // broken. We are doing it here, on the CallKit handler.
 -         const tracks = getState()['features/base/tracks'];
 -         const oldValue = isLocalTrackMuted(tracks, MEDIA_TYPE.AUDIO);
 - 
 -         newValue = Boolean(newValue); // eslint-disable-line no-param-reassign
 - 
 -         if (oldValue !== newValue) {
 -             sendAnalytics(createTrackMutedEvent('audio', 'callkit', newValue));
 -             dispatch(setAudioMuted(newValue, /* ensureTrack */ true));
 -         }
 -     }
 - }
 - 
 - /**
 -  * Update CallKit with the audio only state of the conference. When a conference
 -  * is in audio only mode we will tell CallKit the call has no video. This
 -  * affects how the call is saved in the recent calls list.
 -  *
 -  * XXX: Note that here we are taking the `audioOnly` value straight from the
 -  * action, instead of examining the state. This is intentional, as setting the
 -  * audio only involves multiple actions which will be reflected in the state
 -  * later, but we are just interested in knowing if the mode is going to be
 -  * set or not.
 -  *
 -  * @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} in the specified {@code store}.
 -  * @param {Action} action - The redux action which is being dispatched in the
 -  * specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _setAudioOnly({ getState }, next, action) {
 -     const result = next(action);
 -     const state = getState();
 -     const conference = getCurrentConference(state);
 - 
 -     if (conference && conference.callUUID) {
 -         CallKit.updateCall(
 -             conference.callUUID,
 -             { hasVideo: !action.audioOnly });
 -     }
 - 
 -     return result;
 - }
 - 
 - /**
 -  * Notifies the feature callkit that the action
 -  * {@link _SET_CALLKIT_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 {@code dispatch} function to dispatch the
 -  * specified {@code action} in the specified {@code store}.
 -  * @param {Action} action - The redux action {@code _SET_CALLKIT_SUBSCRIPTIONS}
 -  * which is being dispatched in the specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _setCallKitSubscriptions({ getState }, next, action) {
 -     const { subscriptions } = getState()['features/callkit'];
 - 
 -     if (subscriptions) {
 -         for (const subscription of subscriptions) {
 -             subscription.remove();
 -         }
 -     }
 - 
 -     return next(action);
 - }
 - 
 - /**
 -  * Synchronize the muted state of tracks with CallKit.
 -  *
 -  * @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} in the specified {@code store}.
 -  * @param {Action} action - The redux action which is being dispatched in the
 -  * specified {@code store}.
 -  * @private
 -  * @returns {*} The value returned by {@code next(action)}.
 -  */
 - function _syncTrackState({ getState }, next, action) {
 -     const result = next(action);
 -     const { jitsiTrack } = action.track;
 -     const state = getState();
 -     const conference = getCurrentConference(state);
 - 
 -     if (jitsiTrack.isLocal() && conference && conference.callUUID) {
 -         const tracks = state['features/base/tracks'];
 -         const muted = isLocalTrackMuted(tracks, MEDIA_TYPE.AUDIO);
 - 
 -         CallKit.setMuted(conference.callUUID, muted);
 -         CallKit.updateCall(
 -             conference.callUUID,
 -             { hasVideo: !isVideoMutedByAudioOnly(state) });
 -     }
 - 
 -     return result;
 - }
 
 
  |