[iOS] Fix initial CallKit muted state

Turns out this was a bit more involved than I originally thought due to an
interesting (corner) case: IFF the user was never asked about microphone
permissions and the call starts with audio muted, unmuting from the CallKit
interface won't work (iOS won't show the prompt, it fails immediately) and we
need to sync the mute state back.
This commit is contained in:
Saúl Ibarra Corretgé 2018-01-24 15:41:48 +01:00 committed by Paweł Domas
parent 9e53d40b9c
commit 78fbfba573
1 changed files with 46 additions and 2 deletions

View File

@ -17,12 +17,18 @@ import {
} from '../../base/conference'; } from '../../base/conference';
import { getInviteURL } from '../../base/connection'; import { getInviteURL } from '../../base/connection';
import { import {
isVideoMutedByAudioOnly, MEDIA_TYPE,
SET_AUDIO_MUTED, SET_AUDIO_MUTED,
SET_VIDEO_MUTED, SET_VIDEO_MUTED,
VIDEO_MUTISM_AUTHORITY,
isVideoMutedByAudioOnly,
setAudioMuted setAudioMuted
} from '../../base/media'; } from '../../base/media';
import { MiddlewareRegistry } from '../../base/redux'; import { MiddlewareRegistry } from '../../base/redux';
import {
TRACK_CREATE_ERROR,
isLocalTrackMuted
} from '../../base/tracks';
import { _SET_CALLKIT_SUBSCRIPTIONS } from './actionTypes'; import { _SET_CALLKIT_SUBSCRIPTIONS } from './actionTypes';
import CallKit from './CallKit'; import CallKit from './CallKit';
@ -65,6 +71,9 @@ CallKit && MiddlewareRegistry.register(store => next => action => {
case SET_VIDEO_MUTED: case SET_VIDEO_MUTED:
return _setVideoMuted(store, next, action); return _setVideoMuted(store, next, action);
case TRACK_CREATE_ERROR:
return _trackCreateError(store, next, action);
} }
return next(action); return next(action);
@ -230,10 +239,14 @@ function _conferenceWillJoin({ getState }, next, action) {
.then(() => { .then(() => {
const { room } = state['features/base/conference']; const { room } = state['features/base/conference'];
const { callee } = state['features/base/jwt']; const { callee } = state['features/base/jwt'];
const tracks = state['features/base/tracks'];
const muted = isLocalTrackMuted(tracks, MEDIA_TYPE.AUDIO);
CallKit.updateCall( CallKit.updateCall(
conference.callUUID, conference.callUUID,
{ displayName: (callee && callee.name) || room }); { displayName: (callee && callee.name) || room });
CallKit.setMuted(conference.callUUID, muted);
}); });
return result; return result;
@ -280,7 +293,8 @@ function _onPerformSetMutedCallAction({ callUUID, muted: newValue }) {
const value = Boolean(newValue); const value = Boolean(newValue);
sendAnalytics(createTrackMutedEvent('audio', 'callkit', value)); sendAnalytics(createTrackMutedEvent('audio', 'callkit', value));
dispatch(setAudioMuted(value)); dispatch(setAudioMuted(
value, VIDEO_MUTISM_AUTHORITY.USER, /* ensureTrack */ true));
} }
} }
} }
@ -361,3 +375,33 @@ function _setVideoMuted({ getState }, next, action) {
return result; return result;
} }
/**
* Handles a track creation failure. This is relevant to us in the following
* (corner) case: if the user never gave their permission to use the microphone
* and try to unmute from the CallKit interface, this will fail, and we need
* to sync back the CallKit button state.
*
* @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 TRACK_CREARE_ERROR} which is
* being dispatched in the specified {@code store}.
* @private
* @returns {*}
*/
function _trackCreateError({ getState }, next, action) {
const result = next(action);
const state = getState();
const conference = getCurrentConference(state);
if (conference && conference.callUUID) {
const tracks = state['features/base/tracks'];
const muted = isLocalTrackMuted(tracks, MEDIA_TYPE.AUDIO);
CallKit.setMuted(conference.callUUID, muted);
}
return result;
}