mirror of
https://github.com/danog/Telegram.git
synced 2024-12-03 09:57:46 +01:00
769 lines
25 KiB
Objective-C
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
|