Problemas con AVCaptureSession en modo horizontal en iPad
-
27-12-2019 - |
Pregunta
He estado intentando capturar fotogramas desde la cámara frontal y presentarlos en la vista.Aquí está mi código.
_session = [[AVCaptureSession alloc] init];
_session.sessionPreset = AVCaptureSessionPreset640x480;
AVCaptureDevice *device = [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
NSError *error = nil;
AVCaptureDeviceInput *deviceInput = [[AVCaptureDeviceInput alloc] initWithDevice:device error:&error];
if (deviceInput) {
[_session addInput:deviceInput];
}
else {
DDLogInfo(@"Some wierd shit happened");
return;
}
// Session output
/*
AVCaptureMetadataOutput *output = [[AVCaptureMetadataOutput alloc] init];
[output setMetadataObjectsDelegate:self queue:dispatch_get_main_queue()];
[_session addOutput:output];
output.metadataObjectTypes = @[AVMetadataObjectTypeFace];
AVCaptureConnection *connection = [output connectionWithMediaType:AVMediaTypeMetadata];
connection.videoOrientation = AVCaptureVideoOrientationLandscapeLeft;
*/
// Session output
AVCaptureMovieFileOutput *videoOutput = [[AVCaptureMovieFileOutput alloc] init];
[_session addOutput:videoOutput];
AVCaptureConnection *connection = [videoOutput connectionWithMediaType:AVMediaTypeVideo];
connection.videoOrientation = AVCaptureVideoOrientationPortrait;
// Preview layer
_previewLayer = [AVCaptureVideoPreviewLayer layerWithSession:_session];
_previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill;
_previewLayer.frame = CGRectMake(0, 0, self.middleView.frame.size.width, self.middleView.frame.size.height);
[self.middleView.layer addSublayer:_previewLayer];
[_session startRunning];
El problema con esto es que, cuando mi iPad está en posición horizontal, las imágenes en la capa de presentación se giran 90 grados.¿Cómo puedo solucionar esto?He estado navegando por el desbordamiento de pila, intentando hacer algo con AVCaptureConnection
, pero lamentablemente fue en vano.
Solución
Lo solucionó, ahora está trabajando en todas las orientaciones.Necesitamos establecer este
_previewLayer.connection.videoOrientation = [self videoOrientationFromCurrentDeviceOrientation];
donde el método es:
- (AVCaptureVideoOrientation) videoOrientationFromCurrentDeviceOrientation {
switch (self.interfaceOrientation) {
case UIInterfaceOrientationPortrait: {
return AVCaptureVideoOrientationPortrait;
}
case UIInterfaceOrientationLandscapeLeft: {
return AVCaptureVideoOrientationLandscapeLeft;
}
case UIInterfaceOrientationLandscapeRight: {
return AVCaptureVideoOrientationLandscapeRight;
}
case UIInterfaceOrientationPortraitUpsideDown: {
return AVCaptureVideoOrientationPortraitUpsideDown;
}
}
}
También en la salida de captura necesitamos establecer:
AVCaptureConnection *output2VideoConnection = [videoOutput connectionWithMediaType:AVMediaTypeVideo];
output2VideoConnection.videoOrientation = [self videoOrientationFromCurrentDeviceOrientation];
Otros consejos
En veloz 3:
Primero tenemos que configurar previewLayer.connection.videoOrientation
como lo hace megamanx
previewLayer.connection.videoOrientation = self.videoOrientationFromCurrentDeviceOrientation()
donde el método es:
func videoOrientationFromCurrentDeviceOrientation() -> AVCaptureVideoOrientation {
switch UIApplication.shared.statusBarOrientation {
case .portrait:
return AVCaptureVideoOrientation.portrait
case .landscapeLeft:
return AVCaptureVideoOrientation.landscapeLeft
case .landscapeRight:
return AVCaptureVideoOrientation.landscapeRight
case .portraitUpsideDown:
return AVCaptureVideoOrientation.portraitUpsideDown
default:
// Can this happen?
return AVCaptureVideoOrientation.portrait
}
}
Entonces también tenemos que manejar el videoOrientation
cuando el usuario gira el dispositivo (el método proviene de aquí)
override func viewWillTransition(to size: CGSize, with coordinator: UIViewControllerTransitionCoordinator) {
coordinator.animate(alongsideTransition: { (UIViewControllerTransitionCoordinatorContext) -> Void in
self.previewLayer.connection.videoOrientation = self.videoOrientationFromCurrentDeviceOrientation()
}, completion: { (UIViewControllerTransitionCoordinatorContext) -> Void in
// Finish Rotation
})
super.viewWillTransition(to: size, with: coordinator)
}