2012-01-19 13 views
41

estoy capturar imágenes con este códigoios captura de imagen utilizando AVFramework

#pragma mark - image capture 

// Create and configure a capture session and start it running 
- (void)setupCaptureSession 
{ 
    NSError *error = nil; 

    // Create the session 
    AVCaptureSession *session = [[AVCaptureSession alloc] init]; 

    // Configure the session to produce lower resolution video frames, if your 
    // processing algorithm can cope. We'll specify medium quality for the 
    // chosen device. 
    session.sessionPreset = AVCaptureSessionPresetMedium; 

    // Find a suitable AVCaptureDevice 
    AVCaptureDevice *device = [AVCaptureDevice 
          defaultDeviceWithMediaType:AVMediaTypeVideo]; 

    // Create a device input with the device and add it to the session. 
    AVCaptureDeviceInput *input = [AVCaptureDeviceInput deviceInputWithDevice:device 
                    error:&error]; 
    if (!input) 
    { 
     NSLog(@"PANIC: no media input"); 
    } 
    [session addInput:input]; 

    // Create a VideoDataOutput and add it to the session 
    AVCaptureVideoDataOutput *output = [[AVCaptureVideoDataOutput alloc] init]; 
    [session addOutput:output]; 

    // Configure your output. 
    dispatch_queue_t queue = dispatch_queue_create("myQueue", NULL); 
    [output setSampleBufferDelegate:self queue:queue]; 
    dispatch_release(queue); 

    // Specify the pixel format 
    output.videoSettings = 
    [NSDictionary dictionaryWithObject: 
    [NSNumber numberWithInt:kCVPixelFormatType_32BGRA] 
          forKey:(id)kCVPixelBufferPixelFormatTypeKey]; 


    // If you wish to cap the frame rate to a known value, such as 15 fps, set 
    // minFrameDuration. 

    // Start the session running to start the flow of data 
    [session startRunning]; 

    // Assign session to an ivar. 
    [self setSession:session]; 
} 




// Delegate routine that is called when a sample buffer was written 
- (void)captureOutput:(AVCaptureOutput *)captureOutput 
didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer 
    fromConnection:(AVCaptureConnection *)connection 
{ 
    NSLog(@"captureOutput: didOutputSampleBufferFromConnection"); 

    // Create a UIImage from the sample buffer data 
    UIImage *image = [self imageFromSampleBuffer:sampleBuffer]; 

    //< Add your code here that uses the image > 
    [self.imageView setImage:image]; 
    [self.view setNeedsDisplay]; 
} 


// Create a UIImage from sample buffer data 
- (UIImage *) imageFromSampleBuffer:(CMSampleBufferRef) sampleBuffer 
{ 
    NSLog(@"imageFromSampleBuffer: called"); 
    // Get a CMSampleBuffer's Core Video image buffer for the media data 
    CVImageBufferRef imageBuffer = CMSampleBufferGetImageBuffer(sampleBuffer); 
    // Lock the base address of the pixel buffer 
    CVPixelBufferLockBaseAddress(imageBuffer, 0); 

    // Get the number of bytes per row for the pixel buffer 
    void *baseAddress = CVPixelBufferGetBaseAddress(imageBuffer); 

    // Get the number of bytes per row for the pixel buffer 
    size_t bytesPerRow = CVPixelBufferGetBytesPerRow(imageBuffer); 
    // Get the pixel buffer width and height 
    size_t width = CVPixelBufferGetWidth(imageBuffer); 
    size_t height = CVPixelBufferGetHeight(imageBuffer); 

    // Create a device-dependent RGB color space 
    CGColorSpaceRef colorSpace = CGColorSpaceCreateDeviceRGB(); 

    // Create a bitmap graphics context with the sample buffer data 
    CGContextRef context = CGBitmapContextCreate(baseAddress, width, height, 8, 
              bytesPerRow, colorSpace, kCGBitmapByteOrder32Little | kCGImageAlphaPremultipliedFirst); 
    // Create a Quartz image from the pixel data in the bitmap graphics context 
    CGImageRef quartzImage = CGBitmapContextCreateImage(context); 
    // Unlock the pixel buffer 
    CVPixelBufferUnlockBaseAddress(imageBuffer,0); 


    // Free up the context and color space 
    CGContextRelease(context); 
    CGColorSpaceRelease(colorSpace); 

    // Create an image object from the Quartz image 
    UIImage *image = [UIImage imageWithCGImage:quartzImage]; 

    // Release the Quartz image 
    CGImageRelease(quartzImage); 

    return (image); 
} 

-(void)setSession:(AVCaptureSession *)session 
{ 
    NSLog(@"setting session..."); 
    self.captureSession=session; 
} 

trabaja código de captura. ¡Pero! Necesito cambiar a cosas: - secuencia de video de la cámara en mi opinión. - obteniendo imágenes cada (por ejemplo 5 segundos) de ella. Ayúdeme por favor, ¿cómo se puede hacer?

+13

+1 para compartir el código detallado. Ayudó mucho en mi trabajo – zolio

Respuesta

18

Añadir la siguiente línea

output.minFrameDuration = CMTimeMake(5, 1); 

debajo del comentario

// If you wish to cap the frame rate to a known value, such as 15 fps, set 
// minFrameDuration. 

pero por encima del

[session startRunning]; 

Editar

Use el siguiente código para obtener una vista previa de la salida de la cámara.

AVCaptureVideoPreviewLayer *previewLayer = [AVCaptureVideoPreviewLayer layerWithSession:session]; 
UIView *aView = self.view; 
CGRect videoRect = CGRectMake(0.0, 0.0, 320.0, 150.0); 
previewLayer.frame = videoRect; // Assume you want the preview layer to fill the view. 
[aView.layer addSublayer:previewLayer]; 

Edición 2: bien Ok ..

Apple ha proporcionado una manera de establecer el minFrameDuration here

Así que ahora, utilice el siguiente código para establecer la duración de la trama

AVCaptureConnection *conn = [output connectionWithMediaType:AVMediaTypeVideo]; 

if (conn.supportsVideoMinFrameDuration) 
    conn.videoMinFrameDuration = CMTimeMake(5,1); 
if (conn.supportsVideoMaxFrameDuration) 
    conn.videoMaxFrameDuration = CMTimeMake(5,1); 
+0

Gracias. ¿Cómo puedo configurar la transmisión de video desde la cámara desde mi punto de vista? ¿Debo agregar otro AVCaptureOutput? – Oleg

+0

y, por cierto, este método está en desuso – Oleg

+2

Use 'AVCaptureVideoPreviewLayer' para reproducir la vista previa. Ver mi respuesta actualizada. – Ilanchezhian

15

Tenga cuidado: la devolución de llamada de AVCaptureOutput se publica en la cola de envío yo usted especificó Vi que realizaste actualizaciones de UI a partir de esta devolución de llamada, y eso está mal. Debe realizarlos solo en la cola principal. P.ej.

- (void)captureOutput:(AVCaptureOutput *)captureOutput 
    didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer 
     fromConnection:(AVCaptureConnection *)connection 
{ 
    NSLog(@"captureOutput: didOutputSampleBufferFromConnection"); 
    // Create a UIImage from the sample buffer data 
    UIImage *image = [self imageFromSampleBuffer:sampleBuffer]; 
    dispatch_async(dispatch_get_main_queue(), ^{ 
    //< Add your code here that uses the image > 
     [self.imageView setImage:image]; 
     [self.view setNeedsDisplay]; 
    } 
} 
+1

¡Eres un salvavidas! Esto impedía que mi vista de la imagen se actualizase desde la cámara, ¡gracias! –

+0

¡Increíble! ¡Gracias! –

4

Y aquí es una versión de la función Swift imageFromSampleBuffer:

func imageFromSampleBuffer(sampleBuffer:CMSampleBuffer!) -> UIImage { 
    let imageBuffer = CMSampleBufferGetImageBuffer(sampleBuffer)! 
    CVPixelBufferLockBaseAddress(imageBuffer, 0) 

    let baseAddress = CVPixelBufferGetBaseAddress(imageBuffer) 
    let bytesPerRow = CVPixelBufferGetBytesPerRow(imageBuffer) 
    let width = CVPixelBufferGetWidth(imageBuffer) 
    let height = CVPixelBufferGetHeight(imageBuffer) 

    let colorSpace = CGColorSpaceCreateDeviceRGB() 

    let bitmapInfo:CGBitmapInfo = [.ByteOrder32Little, CGBitmapInfo(rawValue: CGImageAlphaInfo.PremultipliedFirst.rawValue)] 
    let context = CGBitmapContextCreate(baseAddress, width, height, 8, bytesPerRow, colorSpace, bitmapInfo.rawValue) 

    let quartzImage = CGBitmapContextCreateImage(context) 
    CVPixelBufferUnlockBaseAddress(imageBuffer, 0) 

    let image = UIImage(CGImage: quartzImage!) 
    return image 
} 

Por encima de trabajar para mí con la siguiente configuración de vídeo:

videoDataOutput = AVCaptureVideoDataOutput() 
videoDataOutput?.videoSettings = [kCVPixelBufferPixelFormatTypeKey:Int(kCVPixelFormatType_32BGRA)] 
      videoDataOutput?.setSampleBufferDelegate(self, queue: queue) 
+0

gracias! esto me ahorró mucho tiempo – stanley

Cuestiones relacionadas