自定義相機碰到的問題,比如常見的拍照錄製影片方向,鏡像左右顛倒等問題
iOS開發了好幾年了,自定義相機都碰到過很多次,每次都是從網上copy程式碼使用,但是很多時候都會有方向等問題,從來沒有真正研究過,現在在這裡記錄一下自定義相機碰到的問題,以防忘記
問題一:橫向拍照/錄製影片,得到的影片也需要橫屏。
要實現這個功能,就需要獲取到設備的方向,這裡有兩種方法獲取方向
方法一:根據設備方向開關來獲取方向,[UIDevice currentDevice].orientation
方法二:根據重力感應獲取方向
DN_WEAK_SELF if([self.cmmotionManager isDeviceMotionAvailable]) { [self.cmmotionManager startAccelerometerUpdatesToQueue:[NSOperationQueue currentQueue] withHandler:^(CMAccelerometerData * _Nullable accelerometerData, NSError * _Nullable error) { DN_STRONG_SELF double x = accelerometerData.acceleration.x; double y = accelerometerData.acceleration.y; if (fabs(y) >= fabs(x)) { if (y >= 0) { //Down self.deviceOrientation = UIDeviceOrientationPortraitUpsideDown; } else { //Portrait self.deviceOrientation = UIDeviceOrientationPortrait; } } else { if (x >= 0) { //Righ self.deviceOrientation = UIDeviceOrientationLandscapeRight; } else { //Left self.deviceOrientation = UIDeviceOrientationLandscapeLeft; } } }]; }
獲取到方向後,在拍照前設置方向
AVCaptureConnection *myVideoConnection = [self.ImageOutPut connectionWithMediaType:AVMediaTypeVideo]; if ([myVideoConnection isVideoOrientationSupported]) { myVideoConnection.videoOrientation = [self getCaptureVideoOrientation:self.deviceOrientation]; }
在錄製影片前設置方向
AVCaptureConnection *connection = [self.captureMovieFileOutput connectionWithMediaType:AVMediaTypeVideo]; if ([connection isVideoOrientationSupported]) { connection.videoOrientation = [self getCaptureVideoOrientation:self.deviceOrientation]; }
問題二: 前置攝影機拍照時,照片和影片左右顛倒(類似微信)
這個問題其實很簡單,只要設置一下輸出對象的鏡像開關就可以了
AVCaptureSession *session = (AVCaptureSession *)self.session; for (AVCaptureVideoDataOutput* output in session.outputs) { for (AVCaptureConnection * av in output.connections) { //判斷是否是前置攝影機狀態 if ([self.input device].position == AVCaptureDevicePositionFront) { if (av.supportsVideoMirroring) { //鏡像設置 av.videoMirrored = YES; } } } }
錄製影片的話,解決左右顛倒 只能用以上的方法
而拍照的話,解決左右顛倒,還有一種方法,在拍完照片後,設置圖片方向
UIImageOrientation imgOrientation; //拍攝後獲取的的影像方向 if ([self.input device].position == AVCaptureDevicePositionFront && self.isFixLeftRightProblem) { NSLog(@"前置攝影機"); // 前置攝影機影像方向 UIImageOrientationLeftMirrored // IOS前置攝影機左右成像 imgOrientation = UIImageOrientationLeftMirrored; if (image.imageOrientation == UIImageOrientationRight) { imgOrientation = UIImageOrientationLeftMirrored; }else if (image.imageOrientation == UIImageOrientationLeft) { imgOrientation = UIImageOrientationRightMirrored; }else if (image.imageOrientation == UIImageOrientationDown) { imgOrientation = UIImageOrientationDownMirrored; }else if (image.imageOrientation == UIImageOrientationUp) { imgOrientation = UIImageOrientationUpMirrored; } image = [[UIImage alloc]initWithCGImage:image.CGImage scale:1.0f orientation:imgOrientation]; }
問題三:自定義相機拍照後,在其他端查看時,方向不正確
這時候就需要矯正方向了
/*! ** 矯正圖片方向 參考 //www.cocoachina.com/articles/12021 */ - (UIImage *)fixOrientation { // No-op if the orientation is already correct if (self.imageOrientation == UIImageOrientationUp) return self; // We need to calculate the proper transformation to make the image upright. // We do it in 2 steps: Rotate if Left/Right/Down, and then flip if Mirrored. CGAffineTransform transform = CGAffineTransformIdentity; switch (self.imageOrientation) { case UIImageOrientationDown: case UIImageOrientationDownMirrored: transform = CGAffineTransformTranslate(transform, self.size.width, self.size.height); transform = CGAffineTransformRotate(transform, M_PI); break; case UIImageOrientationLeft: case UIImageOrientationLeftMirrored: transform = CGAffineTransformTranslate(transform, self.size.width, 0); transform = CGAffineTransformRotate(transform, M_PI_2); break; case UIImageOrientationRight: case UIImageOrientationRightMirrored: transform = CGAffineTransformTranslate(transform, 0, self.size.height); transform = CGAffineTransformRotate(transform, -M_PI_2); break; case UIImageOrientationUp: case UIImageOrientationUpMirrored: break; } switch (self.imageOrientation) { case UIImageOrientationUpMirrored: case UIImageOrientationDownMirrored: transform = CGAffineTransformTranslate(transform, self.size.width, 0); transform = CGAffineTransformScale(transform, -1, 1); break; case UIImageOrientationLeftMirrored: case UIImageOrientationRightMirrored: transform = CGAffineTransformTranslate(transform, self.size.height, 0); transform = CGAffineTransformScale(transform, -1, 1); break; case UIImageOrientationUp: case UIImageOrientationDown: case UIImageOrientationLeft: case UIImageOrientationRight: break; } // Now we draw the underlying CGImage into a new context, applying the transform // calculated above. CGContextRef ctx = CGBitmapContextCreate(NULL, self.size.width, self.size.height, CGImageGetBitsPerComponent(self.CGImage), 0, CGImageGetColorSpace(self.CGImage), CGImageGetBitmapInfo(self.CGImage)); CGContextConcatCTM(ctx, transform); switch (self.imageOrientation) { case UIImageOrientationLeft: case UIImageOrientationLeftMirrored: case UIImageOrientationRight: case UIImageOrientationRightMirrored: // Grr... CGContextDrawImage(ctx, CGRectMake(0,0,self.size.height,self.size.width), self.CGImage); break; default: CGContextDrawImage(ctx, CGRectMake(0,0,self.size.width,self.size.height), self.CGImage); break; } // And now we just create a new UIImage from the drawing context CGImageRef cgimg = CGBitmapContextCreateImage(ctx); UIImage *img = [UIImage imageWithCGImage:cgimg]; CGContextRelease(ctx); CGImageRelease(cgimg); return img; }
問題四:自定義相機打開後,這時候來電話了,再切回到相機不能拍照了。
這裡就需要監聽音頻中斷的通知AVCaptureSessionWasInterruptedNotification,中斷結束的通知AVCaptureSessionInterruptionEndedNotification
還要監聽系統電話來電 CTCallCenter callEventHandler,在中斷通知AVCaptureSessionWasInterruptedNotification中,將音頻輸入設備從session中移除,等中斷通知結束後,判斷是否時系統電話造成的,如果是則將音頻輸入設備加入session
監聽來電的程式碼 <CoreTelephony/CTCallCenter.h> <CoreTelephony/CTCall.h>
DN_WEAK_SELF self.callCenter = [[CTCallCenter alloc] init]; self.callCenter.callEventHandler = ^(CTCall * call) { DN_STRONG_SELF if([call.callState isEqualToString:CTCallStateDisconnected]) { NSLog(@"Call has been disconnected");//電話被掛斷(我們用的這個) self.isSystemCall = NO; } else if([call.callState isEqualToString:CTCallStateConnected]) { NSLog(@"Call has been connected");//電話被接聽 } else if([call.callState isEqualToString:CTCallStateIncoming]) { NSLog(@"Call is incoming");//來電話了 self.isSystemCall = YES; [self endCall]; } else if([call.callState isEqualToString:CTCallStateDialing]) { NSLog(@"Call is Dialing");//撥號 self.isSystemCall = YES; [self endCall]; } else { NSLog(@"Nothing is done"); } };
中斷音頻程式碼
- (void)initNotification { [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(AVCaptureSessionWasInterruptedNotification:) name:AVCaptureSessionWasInterruptedNotification object:self.session]; [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(AVCaptureSessionInterruptionEndedNotification:) name:AVCaptureSessionInterruptionEndedNotification object:self.session]; } #pragma mark - 通知 - (void)AVCaptureSessionWasInterruptedNotification:(NSNotification *)notification { AVCaptureSessionInterruptionReason reason = [notification.userInfo[AVCaptureSessionInterruptionReasonKey] integerValue]; NSLog(@"fanshijian 中斷 : %d reason : %ld",self.session.interrupted,(long)reason); if (reason == AVCaptureSessionInterruptionReasonAudioDeviceInUseByAnotherClient) { [self removeAudioInput]; } } - (void)AVCaptureSessionInterruptionEndedNotification:(NSNotification *)notification { NSLog(@"fanshijian 中斷結束 : %d",self.session.interrupted); if (![DNVideoChatConfig sharedConfig].isSystemCall) { [self addAudioInput]; } } - (void)addAudioInput { if (!_audioCaptureDeviceInput) { //添加一個音頻輸入設備 AVCaptureDevice *audioCaptureDevice = [[AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio] firstObject]; //添加音頻 NSError *error = nil; AVCaptureDeviceInput *audioCaptureDeviceInput = [[AVCaptureDeviceInput alloc]initWithDevice:audioCaptureDevice error:&error]; if (error) { NSLog(@"取得設備輸入對象時出錯,錯誤原因:%@",error.localizedDescription); return; } self.audioCaptureDeviceInput = audioCaptureDeviceInput; } if ([self.session canAddInput:self.audioCaptureDeviceInput]) { [self.session addInput:self.audioCaptureDeviceInput]; } } - (void)removeAudioInput { [self.session removeInput:self.audioCaptureDeviceInput]; }
可以參考demo //github.com/smallMas/FSEditVideo.git 類FSLiveWindow