1
0
mirror of https://github.com/danog/Telegram.git synced 2024-12-03 09:57:46 +01:00
Telegram/thirdparty/PGPhotoEditor/PGCamera.m
2016-02-25 01:03:51 +01:00

769 lines
25 KiB
Objective-C

#import "PGCamera.h"
#import "PGCameraCaptureSession.h"
#import "PGCameraMovieWriter.h"
#import "PGCameraDeviceAngleSampler.h"
#import "SQueue.h"
#import "TGAccessChecker.h"
#import "TGCameraPreviewView.h"
NSString *const PGCameraFlashActiveKey = @"flashActive";
NSString *const PGCameraFlashAvailableKey = @"flashAvailable";
NSString *const PGCameraTorchActiveKey = @"torchActive";
NSString *const PGCameraTorchAvailableKey = @"torchAvailable";
NSString *const PGCameraAdjustingFocusKey = @"adjustingFocus";
@interface PGCamera ()
{
dispatch_queue_t cameraProcessingQueue;
dispatch_queue_t audioProcessingQueue;
AVCaptureDevice *_microphone;
AVCaptureVideoDataOutput *videoOutput;
AVCaptureAudioDataOutput *audioOutput;
PGCameraDeviceAngleSampler *_deviceAngleSampler;
bool _subscribedForCameraChanges;
bool _shownAudioAccessDisabled;
bool _invalidated;
bool _wasCapturingOnEnterBackground;
bool _capturing;
bool _moment;
TGCameraPreviewView *_previewView;
}
@end
@implementation PGCamera
- (instancetype)init
{
return [self initWithPosition:PGCameraPositionUndefined];
}
- (instancetype)initWithPosition:(PGCameraPosition)position
{
self = [super init];
if (self != nil)
{
_captureSession = [[PGCameraCaptureSession alloc] initWithPreferredPosition:position];
_deviceAngleSampler = [[PGCameraDeviceAngleSampler alloc] init];
[_deviceAngleSampler startMeasuring];
__weak PGCamera *weakSelf = self;
self.captureSession.requestPreviewIsMirrored = ^bool
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf == nil || strongSelf->_previewView == nil)
return false;
TGCameraPreviewView *previewView = strongSelf->_previewView;
return previewView.previewLayer.connection.videoMirrored;
};
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleEnteredBackground:) name:UIApplicationDidEnterBackgroundNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleEnteredForeground:) name:UIApplicationWillEnterForegroundNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleRuntimeError:) name:AVCaptureSessionRuntimeErrorNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(handleInterrupted:) name:AVCaptureSessionWasInterruptedNotification object:nil];
}
return self;
}
- (void)dealloc
{
TGLog(@"Camera: dealloc");
[_deviceAngleSampler stopMeasuring];
[self _unsubscribeFromCameraChanges];
self.captureSession.requestPreviewIsMirrored = nil;
[[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationDidEnterBackgroundNotification object:nil];
[[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationWillEnterForegroundNotification object:nil];
[[NSNotificationCenter defaultCenter] removeObserver:self name:AVCaptureSessionRuntimeErrorNotification object:nil];
[[NSNotificationCenter defaultCenter] removeObserver:self name:AVCaptureSessionWasInterruptedNotification object:nil];
}
- (void)handleEnteredBackground:(NSNotification *)__unused notification
{
if (self.isCapturing)
_wasCapturingOnEnterBackground = true;
[self stopCaptureForPause:true completion:nil];
}
- (void)handleEnteredForeground:(NSNotification *)__unused notification
{
if (_wasCapturingOnEnterBackground)
{
_wasCapturingOnEnterBackground = false;
[self startCaptureForResume:true completion:nil];
}
}
- (void)handleRuntimeError:(NSNotification *)notification
{
TGLog(@"ERROR: Camera runtime error: %@", notification.userInfo[AVCaptureSessionErrorKey]);
__weak PGCamera *weakSelf = self;
TGDispatchAfter(1.5f, [PGCamera cameraQueue]._dispatch_queue, ^
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf == nil || strongSelf->_invalidated)
return;
[strongSelf _unsubscribeFromCameraChanges];
for (AVCaptureInput *input in strongSelf.captureSession.inputs)
[strongSelf.captureSession removeInput:input];
for (AVCaptureOutput *output in strongSelf.captureSession.outputs)
[strongSelf.captureSession removeOutput:output];
[strongSelf.captureSession performInitialConfigurationWithCompletion:^
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf != nil)
[strongSelf _subscribeForCameraChanges];
}];
});
}
- (void)handleInterrupted:(NSNotification *)notification
{
if (iosMajorVersion() < 9)
return;
AVCaptureSessionInterruptionReason reason = [notification.userInfo[AVCaptureSessionInterruptionReasonKey] integerValue];
TGLog(@"WARNING: Camera was interrupted with reason %d", reason);
if (self.captureInterrupted != nil)
self.captureInterrupted(reason);
}
- (void)_subscribeForCameraChanges
{
if (_subscribedForCameraChanges)
return;
_subscribedForCameraChanges = true;
[self.captureSession.videoDevice addObserver:self forKeyPath:PGCameraFlashActiveKey options:NSKeyValueObservingOptionNew context:NULL];
[self.captureSession.videoDevice addObserver:self forKeyPath:PGCameraFlashAvailableKey options:NSKeyValueObservingOptionNew context:NULL];
[self.captureSession.videoDevice addObserver:self forKeyPath:PGCameraTorchActiveKey options:NSKeyValueObservingOptionNew context:NULL];
[self.captureSession.videoDevice addObserver:self forKeyPath:PGCameraTorchAvailableKey options:NSKeyValueObservingOptionNew context:NULL];
[self.captureSession.videoDevice addObserver:self forKeyPath:PGCameraAdjustingFocusKey options:NSKeyValueObservingOptionNew context:NULL];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(subjectAreaChanged:) name:AVCaptureDeviceSubjectAreaDidChangeNotification object:self.captureSession.videoDevice];
}
- (void)_unsubscribeFromCameraChanges
{
if (!_subscribedForCameraChanges)
return;
_subscribedForCameraChanges = false;
@try {
[self.captureSession.videoDevice removeObserver:self forKeyPath:PGCameraFlashActiveKey];
[self.captureSession.videoDevice removeObserver:self forKeyPath:PGCameraFlashAvailableKey];
[self.captureSession.videoDevice removeObserver:self forKeyPath:PGCameraTorchActiveKey];
[self.captureSession.videoDevice removeObserver:self forKeyPath:PGCameraTorchAvailableKey];
[self.captureSession.videoDevice removeObserver:self forKeyPath:PGCameraAdjustingFocusKey];
[[NSNotificationCenter defaultCenter] removeObserver:self name:AVCaptureDeviceSubjectAreaDidChangeNotification object:self.captureSession.videoDevice];
} @catch(NSException *e) { }
}
- (void)attachPreviewView:(TGCameraPreviewView *)previewView
{
TGCameraPreviewView *currentPreviewView = _previewView;
if (currentPreviewView != nil)
[currentPreviewView invalidate];
_previewView = previewView;
[previewView setupWithCamera:self];
__weak PGCamera *weakSelf = self;
[[PGCamera cameraQueue] dispatch:^
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf == nil || strongSelf->_invalidated)
return;
[strongSelf.captureSession performInitialConfigurationWithCompletion:^
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf != nil)
[strongSelf _subscribeForCameraChanges];
}];
}];
}
#pragma mark -
- (bool)isCapturing
{
return _capturing;
}
- (void)startCaptureForResume:(bool)resume completion:(void (^)(void))completion
{
if (_invalidated)
return;
[[PGCamera cameraQueue] dispatch:^
{
if (self.captureSession.isRunning)
return;
_capturing = true;
TGLog(@"Camera: start capture");
[self.captureSession startRunning];
TGDispatchOnMainThread(^
{
if (self.captureStarted != nil)
self.captureStarted(resume);
if (completion != nil)
completion();
});
}];
}
- (void)stopCaptureForPause:(bool)pause completion:(void (^)(void))completion
{
if (_invalidated)
return;
if (!pause)
_invalidated = true;
TGLog(@"Camera: stop capture");
[[PGCamera cameraQueue] dispatch:^
{
if (_invalidated)
{
[self.captureSession beginConfiguration];
[self.captureSession resetFlashMode];
TGLog(@"Camera: stop capture invalidated");
TGCameraPreviewView *previewView = _previewView;
if (previewView != nil)
[previewView invalidate];
for (AVCaptureInput *input in self.captureSession.inputs)
[self.captureSession removeInput:input];
for (AVCaptureOutput *output in self.captureSession.outputs)
[self.captureSession removeOutput:output];
[self.captureSession commitConfiguration];
}
TGLog(@"Camera: stop running");
[self.captureSession stopRunning];
_capturing = false;
TGDispatchOnMainThread(^
{
if (_invalidated)
_previewView = nil;
if (self.captureStopped != nil)
self.captureStopped(pause);
});
if (completion != nil)
completion();
}];
}
- (bool)isResetNeeded
{
return self.captureSession.isResetNeeded;
}
- (void)resetSynchronous:(bool)synchronous completion:(void (^)(void))completion
{
[self resetTerminal:false synchronous:synchronous completion:completion];
}
- (void)resetTerminal:(bool)__unused terminal synchronous:(bool)synchronous completion:(void (^)(void))completion
{
void (^block)(void) = ^
{
[self _unsubscribeFromCameraChanges];
[self.captureSession reset];
[self _subscribeForCameraChanges];
if (completion != nil)
completion();
};
if (synchronous)
[[PGCamera cameraQueue] dispatchSync:block];
else
[[PGCamera cameraQueue] dispatch:block];
}
#pragma mark -
- (void)captureNextFrameForVideoThumbnail:(bool)forVideoThumbnail completion:(void (^)(UIImage * image))completion
{
[self.captureSession captureNextFrameForVideoThumbnail:forVideoThumbnail completion:completion];
}
- (void)takePhotoWithCompletion:(void (^)(UIImage *result, PGCameraShotMetadata *metadata))completion
{
[[PGCamera cameraQueue] dispatch:^
{
if (!self.captureSession.isRunning || self.captureSession.imageOutput.isCapturingStillImage || _invalidated)
return;
TGCameraPreviewView *previewView = _previewView;
AVCaptureConnection *previewConnection = previewView.previewLayer.connection;
AVCaptureConnection *imageConnection = [self.captureSession.imageOutput connectionWithMediaType:AVMediaTypeVideo];
bool isMirrored = previewConnection.videoMirrored;
UIInterfaceOrientation orientation = UIInterfaceOrientationPortrait;
if (self.requestedCurrentInterfaceOrientation != nil)
orientation = self.requestedCurrentInterfaceOrientation(NULL);
[imageConnection setVideoOrientation:[PGCamera _videoOrientationForInterfaceOrientation:orientation]];
[self.captureSession.imageOutput captureStillImageAsynchronouslyFromConnection:self.captureSession.imageOutput.connections.firstObject completionHandler:^(CMSampleBufferRef imageDataSampleBuffer, NSError *error)
{
if (imageDataSampleBuffer != NULL && error == nil)
{
NSData *imageData = [AVCaptureStillImageOutput jpegStillImageNSDataRepresentation:imageDataSampleBuffer];
UIImage *image = [[UIImage alloc] initWithData:imageData];
if (self.cameraMode == PGCameraModeSquare)
{
CGFloat shorterSide = MIN(image.size.width, image.size.height);
CGFloat longerSide = MAX(image.size.width, image.size.height);
CGRect cropRect = CGRectMake(CGFloor((longerSide - shorterSide) / 2.0f), 0, shorterSide, shorterSide);
CGImageRef croppedCGImage = CGImageCreateWithImageInRect(image.CGImage, cropRect);
image = [UIImage imageWithCGImage:croppedCGImage scale:image.scale orientation:image.imageOrientation];
CGImageRelease(croppedCGImage);
}
PGCameraShotMetadata *metadata = [[PGCameraShotMetadata alloc] init];
metadata.frontal = isMirrored;
metadata.deviceAngle = [PGCameraShotMetadata relativeDeviceAngleFromAngle:_deviceAngleSampler.currentDeviceAngle orientation:orientation];
if (completion != nil)
completion(image, metadata);
}
}];
}];
}
- (void)startVideoRecordingForMoment:(bool)moment completion:(void (^)(NSURL *, CGAffineTransform transform, CGSize dimensions, NSTimeInterval duration, bool success))completion
{
[[PGCamera cameraQueue] dispatch:^
{
if (!self.captureSession.isRunning || _invalidated)
return;
UIInterfaceOrientation orientation = UIInterfaceOrientationPortrait;
bool mirrored = false;
if (self.requestedCurrentInterfaceOrientation != nil)
orientation = self.requestedCurrentInterfaceOrientation(&mirrored);
mirrored = false;
_moment = moment;
[self.captureSession startVideoRecordingWithOrientation:[PGCamera _videoOrientationForInterfaceOrientation:orientation] mirrored:mirrored completion:completion];
TGDispatchOnMainThread(^
{
if (self.beganVideoRecording != nil)
self.beganVideoRecording(moment);
});
}];
}
- (void)stopVideoRecording
{
[[PGCamera cameraQueue] dispatch:^
{
[self.captureSession stopVideoRecording];
TGDispatchOnMainThread(^
{
if (self.finishedVideoRecording != nil)
self.finishedVideoRecording(_moment);
});
}];
}
- (bool)isRecordingVideo
{
return self.captureSession.movieWriter.isRecording;
}
- (NSTimeInterval)videoRecordingDuration
{
return self.captureSession.movieWriter.currentDuration;
}
#pragma mark - Mode
- (PGCameraMode)cameraMode
{
return self.captureSession.currentMode;
}
- (void)setCameraMode:(PGCameraMode)cameraMode
{
if (self.disabled || self.captureSession.currentMode == cameraMode)
return;
__weak PGCamera *weakSelf = self;
void(^commitBlock)(void) = ^
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf == nil)
return;
[[PGCamera cameraQueue] dispatch:^
{
strongSelf.captureSession.currentMode = cameraMode;
if (strongSelf.finishedModeChange != nil)
strongSelf.finishedModeChange();
}];
};
if (self.beganModeChange != nil)
self.beganModeChange(cameraMode, commitBlock);
}
#pragma mark - Focus and Exposure
- (void)subjectAreaChanged:(NSNotification *)__unused notification
{
[self resetFocusPoint];
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)__unused object change:(NSDictionary *)__unused change context:(void *)__unused context
{
if ([keyPath isEqualToString:PGCameraAdjustingFocusKey])
{
bool adjustingFocus = [[change objectForKey:NSKeyValueChangeNewKey] isEqualToNumber:@YES];
if (adjustingFocus && self.beganAdjustingFocus != nil)
self.beganAdjustingFocus();
else if (!adjustingFocus && self.finishedAdjustingFocus != nil)
self.finishedAdjustingFocus();
}
else if ([keyPath isEqualToString:PGCameraFlashActiveKey] || [keyPath isEqualToString:PGCameraTorchActiveKey])
{
bool active = [[change objectForKey:NSKeyValueChangeNewKey] isEqualToNumber:@YES];
if (self.flashActivityChanged != nil)
self.flashActivityChanged(active);
}
else if ([keyPath isEqualToString:PGCameraFlashAvailableKey] || [keyPath isEqualToString:PGCameraTorchAvailableKey])
{
bool available = [[change objectForKey:NSKeyValueChangeNewKey] isEqualToNumber:@YES];
if (self.flashAvailabilityChanged != nil)
self.flashAvailabilityChanged(available);
}
}
- (bool)supportsExposurePOI
{
return [self.captureSession.videoDevice isExposurePointOfInterestSupported];
}
- (bool)supportsFocusPOI
{
return [self.captureSession.videoDevice isFocusPointOfInterestSupported];
}
- (void)resetFocusPoint
{
const CGPoint centerPoint = CGPointMake(0.5f, 0.5f);
[self _setFocusPoint:centerPoint focusMode:AVCaptureFocusModeContinuousAutoFocus exposureMode:AVCaptureExposureModeContinuousAutoExposure monitorSubjectAreaChange:false];
}
- (void)setFocusPoint:(CGPoint)point
{
[self _setFocusPoint:point focusMode:AVCaptureFocusModeAutoFocus exposureMode:AVCaptureExposureModeAutoExpose monitorSubjectAreaChange:true];
}
- (void)_setFocusPoint:(CGPoint)point focusMode:(AVCaptureFocusMode)focusMode exposureMode:(AVCaptureExposureMode)exposureMode monitorSubjectAreaChange:(bool)monitorSubjectAreaChange
{
[[PGCamera cameraQueue] dispatch:^
{
if (self.disabled)
return;
[self.captureSession setFocusPoint:point focusMode:focusMode exposureMode:exposureMode monitorSubjectAreaChange:monitorSubjectAreaChange];
}];
}
- (bool)supportsExposureTargetBias
{
return [self.captureSession.videoDevice respondsToSelector:@selector(setExposureTargetBias:completionHandler:)];
}
- (void)beginExposureTargetBiasChange
{
[[PGCamera cameraQueue] dispatch:^
{
if (self.disabled)
return;
[self.captureSession setFocusPoint:self.captureSession.focusPoint focusMode:AVCaptureFocusModeLocked exposureMode:AVCaptureExposureModeLocked monitorSubjectAreaChange:false];
}];
}
- (void)setExposureTargetBias:(CGFloat)bias
{
[[PGCamera cameraQueue] dispatch:^
{
if (self.disabled)
return;
[self.captureSession setExposureTargetBias:bias];
}];
}
- (void)endExposureTargetBiasChange
{
[[PGCamera cameraQueue] dispatch:^
{
if (self.disabled)
return;
[self.captureSession setFocusPoint:self.captureSession.focusPoint focusMode:AVCaptureFocusModeAutoFocus exposureMode:AVCaptureExposureModeAutoExpose monitorSubjectAreaChange:true];
}];
}
#pragma mark - Flash
- (bool)hasFlash
{
return self.captureSession.videoDevice.hasFlash;
}
- (bool)flashActive
{
if (self.cameraMode == PGCameraModeVideo || self.cameraMode == PGCameraModeClip)
return self.captureSession.videoDevice.torchActive;
return self.captureSession.videoDevice.flashActive;
}
- (bool)flashAvailable
{
if (self.cameraMode == PGCameraModeVideo || self.cameraMode == PGCameraModeClip)
return self.captureSession.videoDevice.torchAvailable;
return self.captureSession.videoDevice.flashAvailable;
}
- (PGCameraFlashMode)flashMode
{
return self.captureSession.currentFlashMode;
}
- (void)setFlashMode:(PGCameraFlashMode)flashMode
{
[[PGCamera cameraQueue] dispatch:^
{
self.captureSession.currentFlashMode = flashMode;
}];
}
#pragma mark - Position
- (void)togglePosition
{
if ([AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo].count < 2 || self.disabled)
return;
[self _unsubscribeFromCameraChanges];
PGCameraPosition targetCameraPosition = PGCameraPositionFront;
if (self.captureSession.currentCameraPosition == PGCameraPositionFront)
targetCameraPosition = PGCameraPositionRear;
AVCaptureDevice *targetDevice = [PGCameraCaptureSession _deviceWithCameraPosition:targetCameraPosition];
__weak PGCamera *weakSelf = self;
void(^commitBlock)(void) = ^
{
__strong PGCamera *strongSelf = weakSelf;
if (strongSelf == nil)
return;
[[PGCamera cameraQueue] dispatch:^
{
[strongSelf.captureSession setCurrentCameraPosition:targetCameraPosition];
if (strongSelf.finishedPositionChange != nil)
strongSelf.finishedPositionChange();
[strongSelf setZoomLevel:0.0f];
[strongSelf _subscribeForCameraChanges];
}];
};
if (self.beganPositionChange != nil)
self.beganPositionChange(targetDevice.hasFlash, [PGCameraCaptureSession _isZoomAvailableForDevice:targetDevice], commitBlock);
}
#pragma mark - Zoom
- (bool)isZoomAvailable
{
return self.captureSession.isZoomAvailable;
}
- (CGFloat)zoomLevel
{
return self.captureSession.zoomLevel;
}
- (void)setZoomLevel:(CGFloat)zoomLevel
{
zoomLevel = MAX(0.0f, MIN(1.0f, zoomLevel));
[[PGCamera cameraQueue] dispatch:^
{
if (self.disabled)
return;
[self.captureSession setZoomLevel:zoomLevel];
}];
}
#pragma mark - Device Angle
- (void)startDeviceAngleMeasuring
{
[_deviceAngleSampler startMeasuring];
}
- (void)stopDeviceAngleMeasuring
{
[_deviceAngleSampler stopMeasuring];
}
#pragma mark - Availability
+ (bool)cameraAvailable
{
#if TARGET_IPHONE_SIMULATOR
return true;
#endif
return [UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeCamera];
}
+ (bool)hasRearCamera
{
return ([PGCameraCaptureSession _deviceWithCameraPosition:PGCameraPositionRear] != nil);
}
+ (bool)hasFrontCamera
{
return ([PGCameraCaptureSession _deviceWithCameraPosition:PGCameraPositionFront] != nil);
}
+ (SQueue *)cameraQueue
{
static dispatch_once_t onceToken;
static SQueue *queue = nil;
dispatch_once(&onceToken, ^
{
queue = [[SQueue alloc] init];
});
return queue;
}
+ (AVCaptureVideoOrientation)_videoOrientationForInterfaceOrientation:(UIInterfaceOrientation)deviceOrientation
{
switch (deviceOrientation)
{
case UIInterfaceOrientationPortraitUpsideDown:
return AVCaptureVideoOrientationPortraitUpsideDown;
case UIInterfaceOrientationLandscapeLeft:
return AVCaptureVideoOrientationLandscapeLeft;
case UIInterfaceOrientationLandscapeRight:
return AVCaptureVideoOrientationLandscapeRight;
default:
return AVCaptureVideoOrientationPortrait;
}
}
+ (PGCameraAuthorizationStatus)cameraAuthorizationStatus
{
if ([AVCaptureDevice respondsToSelector:@selector(authorizationStatusForMediaType:)])
return [PGCamera _cameraAuthorizationStatusForAuthorizationStatus:[AVCaptureDevice authorizationStatusForMediaType:AVMediaTypeVideo]];
return PGCameraAuthorizationStatusAuthorized;
}
+ (PGMicrophoneAuthorizationStatus)microphoneAuthorizationStatus
{
if ([AVCaptureDevice respondsToSelector:@selector(authorizationStatusForMediaType:)])
return [PGCamera _microphoneAuthorizationStatusForAuthorizationStatus:[AVCaptureDevice authorizationStatusForMediaType:AVMediaTypeAudio]];
return PGMicrophoneAuthorizationStatusAuthorized;
}
+ (PGCameraAuthorizationStatus)_cameraAuthorizationStatusForAuthorizationStatus:(AVAuthorizationStatus)authorizationStatus
{
switch (authorizationStatus)
{
case AVAuthorizationStatusRestricted:
return PGCameraAuthorizationStatusRestricted;
case AVAuthorizationStatusDenied:
return PGCameraAuthorizationStatusDenied;
case AVAuthorizationStatusAuthorized:
return PGCameraAuthorizationStatusAuthorized;
default:
return PGCameraAuthorizationStatusNotDetermined;
}
}
+ (PGMicrophoneAuthorizationStatus)_microphoneAuthorizationStatusForAuthorizationStatus:(AVAuthorizationStatus)authorizationStatus
{
switch (authorizationStatus)
{
case AVAuthorizationStatusRestricted:
return PGMicrophoneAuthorizationStatusRestricted;
case AVAuthorizationStatusDenied:
return PGMicrophoneAuthorizationStatusDenied;
case AVAuthorizationStatusAuthorized:
return PGMicrophoneAuthorizationStatusAuthorized;
default:
return PGMicrophoneAuthorizationStatusNotDetermined;
}
}
@end