[RN][iOS] Default to speaker for video conferences
This commit is contained in:
parent
b1b5f3e6f0
commit
6c12681b9c
|
@ -0,0 +1,14 @@
|
|||
#import <AVFoundation/AVFoundation.h>
|
||||
#import <Foundation/Foundation.h>
|
||||
|
||||
#import "RCTBridgeModule.h"
|
||||
|
||||
|
||||
@interface AudioMode : NSObject<RCTBridgeModule>
|
||||
|
||||
@property (nonatomic, readonly) AVAudioSession *session;
|
||||
@property (nonatomic, readonly) NSString *category;
|
||||
@property (nonatomic, readonly) NSString *mode;
|
||||
@property (nonatomic, readonly) BOOL initialized;
|
||||
|
||||
@end
|
|
@ -0,0 +1,130 @@
|
|||
#import "AudioMode.h"
|
||||
#import "RCTLog.h"
|
||||
|
||||
|
||||
@implementation AudioMode
|
||||
|
||||
RCT_EXPORT_MODULE();
|
||||
|
||||
typedef enum {
|
||||
kAudioModeDefault,
|
||||
kAudioModeAudioCall,
|
||||
kAudioModeVideoCall
|
||||
} JitsiMeetAudioMode;
|
||||
|
||||
- (instancetype)init
|
||||
{
|
||||
self = [super init];
|
||||
if (self) {
|
||||
_initialized = NO;
|
||||
_category = nil;
|
||||
_mode = nil;
|
||||
_session = [AVAudioSession sharedInstance];
|
||||
}
|
||||
return self;
|
||||
}
|
||||
|
||||
- (dispatch_queue_t)methodQueue
|
||||
{
|
||||
// Make sure all our methods run in the main thread. The route change
|
||||
// notification runs there so this will make sure it will only be fired
|
||||
// after our changes have been applied (when we cause them, that is).
|
||||
return dispatch_get_main_queue();
|
||||
}
|
||||
|
||||
- (void)routeChanged:(NSNotification*)notification {
|
||||
NSDictionary *dict = notification.userInfo;
|
||||
NSInteger reason = [[dict valueForKey:AVAudioSessionRouteChangeReasonKey]
|
||||
integerValue];
|
||||
switch (reason) {
|
||||
case AVAudioSessionRouteChangeReasonCategoryChange: {
|
||||
// The category has changed, check if it's the one we want and adjust
|
||||
// as needed.
|
||||
BOOL success;
|
||||
NSError *error;
|
||||
|
||||
if (_session.category != _category) {
|
||||
success = [_session setCategory: _category error: &error];
|
||||
if (!success || error) {
|
||||
RCTLogInfo(@"Error overriding the desired session category");
|
||||
}
|
||||
}
|
||||
|
||||
if (_session.mode != _mode) {
|
||||
success = [_session setMode: _mode error: &error];
|
||||
if (!success || error) {
|
||||
RCTLogInfo(@"Error overriding the desired session mode");
|
||||
}
|
||||
}
|
||||
}
|
||||
default:
|
||||
// Do nothing
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
- (NSDictionary *)constantsToExport
|
||||
{
|
||||
return @{ @"AUDIO_CALL" : [NSNumber numberWithInt: kAudioModeAudioCall],
|
||||
@"VIDEO_CALL" : [NSNumber numberWithInt: kAudioModeVideoCall],
|
||||
@"DEFAULT" : [NSNumber numberWithInt: kAudioModeDefault]
|
||||
};
|
||||
};
|
||||
|
||||
RCT_EXPORT_METHOD(setMode:(int)mode
|
||||
resolve:(RCTPromiseResolveBlock)resolve
|
||||
reject:(RCTPromiseRejectBlock)reject) {
|
||||
NSError *error;
|
||||
BOOL success;
|
||||
NSString *avCategory;
|
||||
NSString *avMode;
|
||||
|
||||
switch (mode) {
|
||||
case kAudioModeAudioCall:
|
||||
avCategory = AVAudioSessionCategoryPlayAndRecord;
|
||||
avMode = AVAudioSessionModeVoiceChat;
|
||||
break;
|
||||
case kAudioModeVideoCall:
|
||||
avCategory = AVAudioSessionCategoryPlayAndRecord;
|
||||
avMode = AVAudioSessionModeVideoChat;
|
||||
break;
|
||||
case kAudioModeDefault:
|
||||
avCategory = AVAudioSessionCategorySoloAmbient;
|
||||
avMode = AVAudioSessionModeDefault;
|
||||
break;
|
||||
default:
|
||||
reject(@"setMode", @"Invalid mode", nil);
|
||||
return;
|
||||
}
|
||||
|
||||
// Configure AVAudioSession category
|
||||
success = [_session setCategory: avCategory error: &error];
|
||||
if (!success || error) {
|
||||
reject(@"setMode", error.localizedDescription, error);
|
||||
return;
|
||||
}
|
||||
|
||||
// Configure AVAudioSession mode
|
||||
success = [_session setMode: avMode error: &error];
|
||||
if (!success || error) {
|
||||
reject(@"setMode", error.localizedDescription, error);
|
||||
return;
|
||||
}
|
||||
|
||||
// Save the desired mode and category
|
||||
_category = avCategory;
|
||||
_mode = avMode;
|
||||
|
||||
// Initialize audio route changes observer if needed
|
||||
if (!_initialized) {
|
||||
[[NSNotificationCenter defaultCenter] addObserver: self
|
||||
selector: @selector(routeChanged:)
|
||||
name: AVAudioSessionRouteChangeNotification
|
||||
object: nil];
|
||||
_initialized = YES;
|
||||
}
|
||||
|
||||
resolve(nil);
|
||||
}
|
||||
|
||||
@end
|
|
@ -13,6 +13,7 @@
|
|||
00C302E81ABCBA2D00DB3ED1 /* libRCTImage.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302C01ABCB91800DB3ED1 /* libRCTImage.a */; };
|
||||
00C302E91ABCBA2D00DB3ED1 /* libRCTNetwork.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302DC1ABCB9D200DB3ED1 /* libRCTNetwork.a */; };
|
||||
00C302EA1ABCBA2D00DB3ED1 /* libRCTVibration.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 00C302E41ABCB9EE00DB3ED1 /* libRCTVibration.a */; };
|
||||
0B42DFAE1E2FD90700111B12 /* AudioMode.m in Sources */ = {isa = PBXBuildFile; fileRef = 0B42DFAD1E2FD90700111B12 /* AudioMode.m */; };
|
||||
133E29F31AD74F7200F7D852 /* libRCTLinking.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 78C398B91ACF4ADC00677621 /* libRCTLinking.a */; };
|
||||
139105C61AF99C1200B5F7CC /* libRCTSettings.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139105C11AF99BAD00B5F7CC /* libRCTSettings.a */; };
|
||||
139FDEF61B0652A700C62182 /* libRCTWebSocket.a in Frameworks */ = {isa = PBXBuildFile; fileRef = 139FDEF41B06529B00C62182 /* libRCTWebSocket.a */; };
|
||||
|
@ -205,6 +206,8 @@
|
|||
00C302BB1ABCB91800DB3ED1 /* RCTImage.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTImage.xcodeproj; path = "../node_modules/react-native/Libraries/Image/RCTImage.xcodeproj"; sourceTree = "<group>"; };
|
||||
00C302D31ABCB9D200DB3ED1 /* RCTNetwork.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTNetwork.xcodeproj; path = "../node_modules/react-native/Libraries/Network/RCTNetwork.xcodeproj"; sourceTree = "<group>"; };
|
||||
00C302DF1ABCB9EE00DB3ED1 /* RCTVibration.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTVibration.xcodeproj; path = "../node_modules/react-native/Libraries/Vibration/RCTVibration.xcodeproj"; sourceTree = "<group>"; };
|
||||
0B42DFAC1E2FD90700111B12 /* AudioMode.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; name = AudioMode.h; path = app/AudioMode.h; sourceTree = "<group>"; };
|
||||
0B42DFAD1E2FD90700111B12 /* AudioMode.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; name = AudioMode.m; path = app/AudioMode.m; sourceTree = "<group>"; };
|
||||
0EA8C046B2BF46279796F07D /* libKCKeepAwake.a */ = {isa = PBXFileReference; explicitFileType = undefined; fileEncoding = 9; includeInIndex = 0; lastKnownFileType = archive.ar; path = libKCKeepAwake.a; sourceTree = "<group>"; };
|
||||
139105B61AF99BAD00B5F7CC /* RCTSettings.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTSettings.xcodeproj; path = "../node_modules/react-native/Libraries/Settings/RCTSettings.xcodeproj"; sourceTree = "<group>"; };
|
||||
139FDEE61B06529A00C62182 /* RCTWebSocket.xcodeproj */ = {isa = PBXFileReference; lastKnownFileType = "wrapper.pb-project"; name = RCTWebSocket.xcodeproj; path = "../node_modules/react-native/Libraries/WebSocket/RCTWebSocket.xcodeproj"; sourceTree = "<group>"; };
|
||||
|
@ -348,6 +351,8 @@
|
|||
children = (
|
||||
13B07FAF1A68108700A75B9A /* AppDelegate.h */,
|
||||
13B07FB01A68108700A75B9A /* AppDelegate.m */,
|
||||
0B42DFAC1E2FD90700111B12 /* AudioMode.h */,
|
||||
0B42DFAD1E2FD90700111B12 /* AudioMode.m */,
|
||||
13B07FB51A68108700A75B9A /* Images.xcassets */,
|
||||
13B07FB61A68108700A75B9A /* Info.plist */,
|
||||
13B07FB11A68108700A75B9A /* LaunchScreen.xib */,
|
||||
|
@ -778,6 +783,7 @@
|
|||
buildActionMask = 2147483647;
|
||||
files = (
|
||||
B3A9D0251E0481E10009343D /* POSIX.m in Sources */,
|
||||
0B42DFAE1E2FD90700111B12 /* AudioMode.m in Sources */,
|
||||
13B07FBC1A68108700A75B9A /* AppDelegate.m in Sources */,
|
||||
13B07FC11A68108700A75B9A /* main.m in Sources */,
|
||||
);
|
||||
|
|
|
@ -3,6 +3,7 @@
|
|||
import { Linking } from 'react-native';
|
||||
|
||||
import { Platform } from '../../base/react';
|
||||
import '../../audio-mode';
|
||||
import '../../wake-lock';
|
||||
|
||||
import { AbstractApp } from './AbstractApp';
|
||||
|
|
|
@ -1,3 +1,4 @@
|
|||
export * from './actions';
|
||||
export * from './actionTypes';
|
||||
export * from './components';
|
||||
export * from './functions';
|
||||
|
|
|
@ -0,0 +1 @@
|
|||
import './middleware';
|
|
@ -0,0 +1,56 @@
|
|||
import { AudioMode } from '../base/react-native';
|
||||
|
||||
import { APP_WILL_MOUNT } from '../app';
|
||||
import {
|
||||
CONFERENCE_FAILED,
|
||||
CONFERENCE_LEFT,
|
||||
CONFERENCE_WILL_JOIN
|
||||
} from '../base/conference';
|
||||
|
||||
import { MiddlewareRegistry } from '../base/redux';
|
||||
|
||||
/**
|
||||
* Middleware that captures conference actions and sets the correct audio
|
||||
* mode based on the type of conference. Audio-only conferences don't
|
||||
* use the speaker by default, and video conferences do.
|
||||
*
|
||||
* @param {Store} store - Redux store.
|
||||
* @returns {Function}
|
||||
*/
|
||||
MiddlewareRegistry.register(store => next => action => {
|
||||
switch (action.type) {
|
||||
case APP_WILL_MOUNT: {
|
||||
AudioMode.setMode(AudioMode.DEFAULT)
|
||||
.catch(err => {
|
||||
console.warn(`Error setting audio mode: ${err}`);
|
||||
});
|
||||
break;
|
||||
}
|
||||
case CONFERENCE_WILL_JOIN: {
|
||||
let mode;
|
||||
const state = store.getState()['features/base/conference'];
|
||||
|
||||
if (state.audioOnly) {
|
||||
// TODO(saghul): Implement audio-only mode
|
||||
mode = AudioMode.AUDIO_CALL;
|
||||
} else {
|
||||
mode = AudioMode.VIDEO_CALL;
|
||||
}
|
||||
|
||||
AudioMode.setMode(mode)
|
||||
.catch(err => {
|
||||
console.warn(`Error setting audio mode: ${err}`);
|
||||
});
|
||||
break;
|
||||
}
|
||||
case CONFERENCE_FAILED:
|
||||
case CONFERENCE_LEFT:
|
||||
AudioMode.setMode(AudioMode.DEFAULT)
|
||||
.catch(err => {
|
||||
console.warn(`Error setting audio mode: ${err}`);
|
||||
});
|
||||
break;
|
||||
}
|
||||
|
||||
return next(action);
|
||||
});
|
|
@ -0,0 +1,20 @@
|
|||
import { NativeModules } from 'react-native';
|
||||
import { Platform } from '../react';
|
||||
|
||||
let AudioMode;
|
||||
|
||||
if (Platform.OS === 'ios') {
|
||||
AudioMode = NativeModules.AudioMode;
|
||||
} else {
|
||||
// TODO(saghul): Implement for Android
|
||||
AudioMode = {
|
||||
DEFAULT: 0,
|
||||
AUDIO_CALL: 1,
|
||||
VIDEO_CALL: 2,
|
||||
setMode() {
|
||||
return Promise.resolve(null);
|
||||
}
|
||||
};
|
||||
}
|
||||
|
||||
export default AudioMode;
|
|
@ -1 +1,2 @@
|
|||
export { default as AudioMode } from './AudioMode';
|
||||
export { default as POSIX } from './POSIX';
|
||||
|
|
Loading…
Reference in New Issue