2012-08-20 23:36:36 +08:00
|
|
|
/*
|
|
|
|
* cap_ios_video_camera.mm
|
|
|
|
* For iOS video I/O
|
|
|
|
* by Eduard Feicho on 29/07/12
|
2013-06-18 11:02:09 +08:00
|
|
|
* by Alexander Shishkov on 17/07/13
|
2012-08-20 23:36:36 +08:00
|
|
|
* Copyright 2012. All rights reserved.
|
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions are met:
|
|
|
|
*
|
|
|
|
* 1. Redistributions of source code must retain the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer.
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
|
|
* and/or other materials provided with the distribution.
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
|
|
|
* derived from this software without specific prior written permission.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR "AS IS" AND ANY EXPRESS OR IMPLIED
|
|
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
|
|
|
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
|
|
|
* EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
|
|
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
|
|
|
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
|
|
|
* OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
|
|
|
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2014-07-10 22:27:32 +08:00
|
|
|
#import "opencv2/videoio/cap_ios.h"
|
2012-08-20 23:36:36 +08:00
|
|
|
#include "precomp.hpp"
|
2017-10-06 01:22:56 +08:00
|
|
|
#import <UIKit/UIKit.h>
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
|
2014-01-18 05:30:29 +08:00
|
|
|
static CGFloat DegreesToRadians(CGFloat degrees) {return degrees * M_PI / 180;}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
#pragma mark - Private Interface
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2015-09-19 20:05:38 +08:00
|
|
|
@interface CvVideoCamera () {
|
|
|
|
int recordingCountDown;
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
- (void)createVideoDataOutput;
|
|
|
|
- (void)createVideoFileOutput;
|
|
|
|
|
|
|
|
|
2016-08-26 02:40:16 +08:00
|
|
|
@property (nonatomic, strong) CALayer *customPreviewLayer;
|
|
|
|
@property (nonatomic, strong) AVCaptureVideoDataOutput *videoDataOutput;
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark - Implementation
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@implementation CvVideoCamera
|
2016-09-14 19:48:41 +08:00
|
|
|
{
|
|
|
|
id<CvVideoCameraDelegate> _delegate;
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@synthesize grayscaleMode;
|
|
|
|
|
|
|
|
@synthesize customPreviewLayer;
|
|
|
|
@synthesize videoDataOutput;
|
|
|
|
|
|
|
|
@synthesize recordVideo;
|
2013-06-14 19:10:25 +08:00
|
|
|
@synthesize rotateVideo;
|
2012-08-20 23:36:36 +08:00
|
|
|
//@synthesize videoFileOutput;
|
|
|
|
@synthesize recordAssetWriterInput;
|
|
|
|
@synthesize recordPixelBufferAdaptor;
|
|
|
|
@synthesize recordAssetWriter;
|
|
|
|
|
2016-09-14 19:48:41 +08:00
|
|
|
- (void)setDelegate:(id<CvVideoCameraDelegate>)newDelegate {
|
|
|
|
_delegate = newDelegate;
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
2016-09-14 19:48:41 +08:00
|
|
|
- (id<CvVideoCameraDelegate>)delegate {
|
|
|
|
return _delegate;
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
#pragma mark - Constructors
|
|
|
|
|
|
|
|
- (id)initWithParentView:(UIView*)parent;
|
|
|
|
{
|
2012-10-17 07:18:30 +08:00
|
|
|
self = [super initWithParentView:parent];
|
|
|
|
if (self) {
|
|
|
|
self.useAVCaptureVideoPreviewLayer = NO;
|
|
|
|
self.recordVideo = NO;
|
2013-06-14 19:10:25 +08:00
|
|
|
self.rotateVideo = NO;
|
2012-10-17 07:18:30 +08:00
|
|
|
}
|
|
|
|
return self;
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#pragma mark - Public interface
|
|
|
|
|
|
|
|
|
|
|
|
- (void)start;
|
|
|
|
{
|
2016-08-03 05:52:02 +08:00
|
|
|
if (self.running == YES) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-09-19 20:05:38 +08:00
|
|
|
recordingCountDown = 10;
|
2012-08-20 23:36:36 +08:00
|
|
|
[super start];
|
2012-10-17 07:18:30 +08:00
|
|
|
|
|
|
|
if (self.recordVideo == YES) {
|
2014-10-17 19:45:13 +08:00
|
|
|
NSError* error = nil;
|
2012-08-22 17:33:13 +08:00
|
|
|
if ([[NSFileManager defaultManager] fileExistsAtPath:[self videoFileString]]) {
|
2012-10-17 07:18:30 +08:00
|
|
|
[[NSFileManager defaultManager] removeItemAtPath:[self videoFileString] error:&error];
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
if (error == nil) {
|
2012-08-22 17:33:13 +08:00
|
|
|
NSLog(@"[Camera] Delete file %@", [self videoFileString]);
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
- (void)stop;
|
|
|
|
{
|
2016-10-28 15:50:37 +08:00
|
|
|
if (self.running == YES) {
|
|
|
|
[super stop];
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2016-10-28 15:50:37 +08:00
|
|
|
[videoDataOutput release];
|
|
|
|
if (videoDataOutputQueue) {
|
|
|
|
dispatch_release(videoDataOutputQueue);
|
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2016-10-28 15:50:37 +08:00
|
|
|
if (self.recordVideo == YES) {
|
|
|
|
if (self.recordAssetWriter) {
|
|
|
|
if (self.recordAssetWriter.status == AVAssetWriterStatusWriting) {
|
|
|
|
[self.recordAssetWriter finishWriting];
|
|
|
|
NSLog(@"[Camera] recording stopped");
|
|
|
|
} else {
|
|
|
|
NSLog(@"[Camera] Recording Error: asset writer status is not writing");
|
|
|
|
}
|
|
|
|
[recordAssetWriter release];
|
2016-08-05 00:44:07 +08:00
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2016-10-28 15:50:37 +08:00
|
|
|
[recordAssetWriterInput release];
|
|
|
|
[recordPixelBufferAdaptor release];
|
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2016-10-28 15:50:37 +08:00
|
|
|
if (self.customPreviewLayer) {
|
|
|
|
[self.customPreviewLayer removeFromSuperlayer];
|
|
|
|
self.customPreviewLayer = nil;
|
|
|
|
}
|
2016-08-05 00:44:07 +08:00
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO fix
|
|
|
|
- (void)adjustLayoutToInterfaceOrientation:(UIInterfaceOrientation)interfaceOrientation;
|
2012-10-17 07:18:30 +08:00
|
|
|
{
|
|
|
|
|
|
|
|
NSLog(@"layout preview layer");
|
|
|
|
if (self.parentView != nil) {
|
|
|
|
|
|
|
|
CALayer* layer = self.customPreviewLayer;
|
|
|
|
CGRect bounds = self.customPreviewLayer.bounds;
|
|
|
|
int rotation_angle = 0;
|
|
|
|
bool flip_bounds = false;
|
|
|
|
|
|
|
|
switch (interfaceOrientation) {
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIInterfaceOrientationPortrait:
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"to Portrait");
|
2012-08-20 23:36:36 +08:00
|
|
|
rotation_angle = 270;
|
|
|
|
break;
|
|
|
|
case UIInterfaceOrientationPortraitUpsideDown:
|
|
|
|
rotation_angle = 90;
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"to UpsideDown");
|
|
|
|
break;
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIInterfaceOrientationLandscapeLeft:
|
|
|
|
rotation_angle = 0;
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"to LandscapeLeft");
|
|
|
|
break;
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIInterfaceOrientationLandscapeRight:
|
|
|
|
rotation_angle = 180;
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"to LandscapeRight");
|
|
|
|
break;
|
2012-08-20 23:36:36 +08:00
|
|
|
default:
|
|
|
|
break; // leave the layer in its last known orientation
|
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2016-08-27 06:10:17 +08:00
|
|
|
switch (self.defaultAVCaptureVideoOrientation) {
|
2012-10-17 07:18:30 +08:00
|
|
|
case AVCaptureVideoOrientationLandscapeRight:
|
|
|
|
rotation_angle += 180;
|
|
|
|
break;
|
|
|
|
case AVCaptureVideoOrientationPortraitUpsideDown:
|
|
|
|
rotation_angle += 270;
|
|
|
|
break;
|
|
|
|
case AVCaptureVideoOrientationPortrait:
|
|
|
|
rotation_angle += 90;
|
|
|
|
case AVCaptureVideoOrientationLandscapeLeft:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
rotation_angle = rotation_angle % 360;
|
|
|
|
|
|
|
|
if (rotation_angle == 90 || rotation_angle == 270) {
|
|
|
|
flip_bounds = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flip_bounds) {
|
|
|
|
NSLog(@"flip bounds");
|
|
|
|
bounds = CGRectMake(0, 0, bounds.size.height, bounds.size.width);
|
|
|
|
}
|
|
|
|
|
|
|
|
layer.position = CGPointMake(self.parentView.frame.size.width/2., self.parentView.frame.size.height/2.);
|
|
|
|
self.customPreviewLayer.bounds = CGRectMake(0, 0, self.parentView.frame.size.width, self.parentView.frame.size.height);
|
|
|
|
|
|
|
|
layer.affineTransform = CGAffineTransformMakeRotation( DegreesToRadians(rotation_angle) );
|
|
|
|
layer.bounds = bounds;
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO fix
|
|
|
|
- (void)layoutPreviewLayer;
|
|
|
|
{
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"layout preview layer");
|
|
|
|
if (self.parentView != nil) {
|
|
|
|
|
|
|
|
CALayer* layer = self.customPreviewLayer;
|
|
|
|
CGRect bounds = self.customPreviewLayer.bounds;
|
|
|
|
int rotation_angle = 0;
|
|
|
|
bool flip_bounds = false;
|
|
|
|
|
|
|
|
switch (currentDeviceOrientation) {
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIDeviceOrientationPortrait:
|
|
|
|
rotation_angle = 270;
|
|
|
|
break;
|
|
|
|
case UIDeviceOrientationPortraitUpsideDown:
|
|
|
|
rotation_angle = 90;
|
2012-10-17 07:18:30 +08:00
|
|
|
break;
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIDeviceOrientationLandscapeLeft:
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"left");
|
2012-08-20 23:36:36 +08:00
|
|
|
rotation_angle = 180;
|
2012-10-17 07:18:30 +08:00
|
|
|
break;
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIDeviceOrientationLandscapeRight:
|
2012-10-17 07:18:30 +08:00
|
|
|
NSLog(@"right");
|
2012-08-20 23:36:36 +08:00
|
|
|
rotation_angle = 0;
|
2012-10-17 07:18:30 +08:00
|
|
|
break;
|
2012-08-20 23:36:36 +08:00
|
|
|
case UIDeviceOrientationFaceUp:
|
|
|
|
case UIDeviceOrientationFaceDown:
|
|
|
|
default:
|
|
|
|
break; // leave the layer in its last known orientation
|
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2016-08-27 06:10:17 +08:00
|
|
|
switch (self.defaultAVCaptureVideoOrientation) {
|
2012-10-17 07:18:30 +08:00
|
|
|
case AVCaptureVideoOrientationLandscapeRight:
|
|
|
|
rotation_angle += 180;
|
|
|
|
break;
|
|
|
|
case AVCaptureVideoOrientationPortraitUpsideDown:
|
|
|
|
rotation_angle += 270;
|
|
|
|
break;
|
|
|
|
case AVCaptureVideoOrientationPortrait:
|
|
|
|
rotation_angle += 90;
|
|
|
|
case AVCaptureVideoOrientationLandscapeLeft:
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
rotation_angle = rotation_angle % 360;
|
|
|
|
|
|
|
|
if (rotation_angle == 90 || rotation_angle == 270) {
|
|
|
|
flip_bounds = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (flip_bounds) {
|
|
|
|
NSLog(@"flip bounds");
|
|
|
|
bounds = CGRectMake(0, 0, bounds.size.height, bounds.size.width);
|
|
|
|
}
|
|
|
|
|
|
|
|
layer.position = CGPointMake(self.parentView.frame.size.width/2., self.parentView.frame.size.height/2.);
|
|
|
|
layer.affineTransform = CGAffineTransformMakeRotation( DegreesToRadians(rotation_angle) );
|
|
|
|
layer.bounds = bounds;
|
|
|
|
}
|
|
|
|
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
#pragma mark - Private Interface
|
|
|
|
|
|
|
|
- (void)createVideoDataOutput;
|
|
|
|
{
|
2012-10-17 07:18:30 +08:00
|
|
|
// Make a video data output
|
|
|
|
self.videoDataOutput = [AVCaptureVideoDataOutput new];
|
|
|
|
|
|
|
|
// In grayscale mode we want YUV (YpCbCr 4:2:0) so we can directly access the graylevel intensity values (Y component)
|
|
|
|
// In color mode we, BGRA format is used
|
|
|
|
OSType format = self.grayscaleMode ? kCVPixelFormatType_420YpCbCr8BiPlanarFullRange : kCVPixelFormatType_32BGRA;
|
|
|
|
|
|
|
|
self.videoDataOutput.videoSettings = [NSDictionary dictionaryWithObject:[NSNumber numberWithUnsignedInt:format]
|
2012-08-20 23:36:36 +08:00
|
|
|
forKey:(id)kCVPixelBufferPixelFormatTypeKey];
|
2012-10-17 07:18:30 +08:00
|
|
|
|
|
|
|
// discard if the data output queue is blocked (as we process the still image)
|
|
|
|
[self.videoDataOutput setAlwaysDiscardsLateVideoFrames:YES];
|
|
|
|
|
|
|
|
if ( [self.captureSession canAddOutput:self.videoDataOutput] ) {
|
|
|
|
[self.captureSession addOutput:self.videoDataOutput];
|
|
|
|
}
|
|
|
|
[[self.videoDataOutput connectionWithMediaType:AVMediaTypeVideo] setEnabled:YES];
|
|
|
|
|
|
|
|
|
|
|
|
// set default FPS
|
2015-08-01 19:55:28 +08:00
|
|
|
AVCaptureDeviceInput *currentInput = [self.captureSession.inputs objectAtIndex:0];
|
|
|
|
AVCaptureDevice *device = currentInput.device;
|
|
|
|
|
|
|
|
NSError *error = nil;
|
|
|
|
[device lockForConfiguration:&error];
|
|
|
|
|
|
|
|
float maxRate = ((AVFrameRateRange*) [device.activeFormat.videoSupportedFrameRateRanges objectAtIndex:0]).maxFrameRate;
|
|
|
|
if (maxRate > self.defaultFPS - 1 && error == nil) {
|
|
|
|
[device setActiveVideoMinFrameDuration:CMTimeMake(1, self.defaultFPS)];
|
|
|
|
[device setActiveVideoMaxFrameDuration:CMTimeMake(1, self.defaultFPS)];
|
|
|
|
NSLog(@"[Camera] FPS set to %d", self.defaultFPS);
|
|
|
|
} else {
|
|
|
|
NSLog(@"[Camera] unable to set defaultFPS at %d FPS, max is %f FPS", self.defaultFPS, maxRate);
|
2012-10-17 07:18:30 +08:00
|
|
|
}
|
2015-08-01 19:55:28 +08:00
|
|
|
|
|
|
|
if (error != nil) {
|
|
|
|
NSLog(@"[Camera] unable to set defaultFPS: %@", error);
|
2012-10-17 07:18:30 +08:00
|
|
|
}
|
|
|
|
|
2015-08-01 19:55:28 +08:00
|
|
|
[device unlockForConfiguration];
|
|
|
|
|
2012-10-17 07:18:30 +08:00
|
|
|
// set video mirroring for front camera (more intuitive)
|
|
|
|
if ([self.videoDataOutput connectionWithMediaType:AVMediaTypeVideo].supportsVideoMirroring) {
|
|
|
|
if (self.defaultAVCaptureDevicePosition == AVCaptureDevicePositionFront) {
|
|
|
|
[self.videoDataOutput connectionWithMediaType:AVMediaTypeVideo].videoMirrored = YES;
|
|
|
|
} else {
|
|
|
|
[self.videoDataOutput connectionWithMediaType:AVMediaTypeVideo].videoMirrored = NO;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// set default video orientation
|
|
|
|
if ([self.videoDataOutput connectionWithMediaType:AVMediaTypeVideo].supportsVideoOrientation) {
|
|
|
|
[self.videoDataOutput connectionWithMediaType:AVMediaTypeVideo].videoOrientation = self.defaultAVCaptureVideoOrientation;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// create a custom preview layer
|
|
|
|
self.customPreviewLayer = [CALayer layer];
|
|
|
|
self.customPreviewLayer.bounds = CGRectMake(0, 0, self.parentView.frame.size.width, self.parentView.frame.size.height);
|
2017-11-21 12:56:23 +08:00
|
|
|
self.customPreviewLayer.position = CGPointMake(self.parentView.frame.size.width/2., self.parentView.frame.size.height/2.);
|
|
|
|
[self updateOrientation];
|
2012-10-17 07:18:30 +08:00
|
|
|
|
|
|
|
// create a serial dispatch queue used for the sample buffer delegate as well as when a still image is captured
|
|
|
|
// a serial dispatch queue must be used to guarantee that video frames will be delivered in order
|
|
|
|
// see the header doc for setSampleBufferDelegate:queue: for more information
|
|
|
|
videoDataOutputQueue = dispatch_queue_create("VideoDataOutputQueue", DISPATCH_QUEUE_SERIAL);
|
|
|
|
[self.videoDataOutput setSampleBufferDelegate:self queue:videoDataOutputQueue];
|
|
|
|
|
|
|
|
|
2015-08-01 19:55:28 +08:00
|
|
|
NSLog(@"[Camera] created AVCaptureVideoDataOutput");
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
- (void)createVideoFileOutput;
|
|
|
|
{
|
2012-10-17 07:18:30 +08:00
|
|
|
/* Video File Output in H.264, via AVAsserWriter */
|
2012-08-20 23:36:36 +08:00
|
|
|
NSLog(@"Create Video with dimensions %dx%d", self.imageWidth, self.imageHeight);
|
2012-10-17 07:18:30 +08:00
|
|
|
|
|
|
|
NSDictionary *outputSettings
|
|
|
|
= [NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithInt:self.imageWidth], AVVideoWidthKey,
|
|
|
|
[NSNumber numberWithInt:self.imageHeight], AVVideoHeightKey,
|
|
|
|
AVVideoCodecH264, AVVideoCodecKey,
|
|
|
|
nil
|
|
|
|
];
|
|
|
|
|
|
|
|
|
|
|
|
self.recordAssetWriterInput = [AVAssetWriterInput assetWriterInputWithMediaType:AVMediaTypeVideo outputSettings:outputSettings];
|
|
|
|
|
|
|
|
|
|
|
|
int pixelBufferFormat = (self.grayscaleMode == YES) ? kCVPixelFormatType_420YpCbCr8BiPlanarFullRange : kCVPixelFormatType_32BGRA;
|
|
|
|
|
|
|
|
self.recordPixelBufferAdaptor =
|
|
|
|
[[AVAssetWriterInputPixelBufferAdaptor alloc]
|
|
|
|
initWithAssetWriterInput:self.recordAssetWriterInput
|
|
|
|
sourcePixelBufferAttributes:[NSDictionary dictionaryWithObjectsAndKeys:[NSNumber numberWithInt:pixelBufferFormat], kCVPixelBufferPixelFormatTypeKey, nil]];
|
|
|
|
|
|
|
|
NSError* error = nil;
|
2012-08-22 17:33:13 +08:00
|
|
|
NSLog(@"Create AVAssetWriter with url: %@", [self videoFileURL]);
|
2012-10-17 07:18:30 +08:00
|
|
|
self.recordAssetWriter = [AVAssetWriter assetWriterWithURL:[self videoFileURL]
|
2012-08-20 23:36:36 +08:00
|
|
|
fileType:AVFileTypeMPEG4
|
|
|
|
error:&error];
|
2012-10-17 07:18:30 +08:00
|
|
|
if (error != nil) {
|
|
|
|
NSLog(@"[Camera] Unable to create AVAssetWriter: %@", error);
|
|
|
|
}
|
|
|
|
|
|
|
|
[self.recordAssetWriter addInput:self.recordAssetWriterInput];
|
|
|
|
self.recordAssetWriterInput.expectsMediaDataInRealTime = YES;
|
|
|
|
|
2012-08-20 23:36:36 +08:00
|
|
|
NSLog(@"[Camera] created AVAssetWriter");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
- (void)createCaptureOutput;
|
|
|
|
{
|
2012-10-17 07:18:30 +08:00
|
|
|
[self createVideoDataOutput];
|
|
|
|
if (self.recordVideo == YES) {
|
|
|
|
[self createVideoFileOutput];
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)createCustomVideoPreview;
|
|
|
|
{
|
2012-10-17 07:18:30 +08:00
|
|
|
[self.parentView.layer addSublayer:self.customPreviewLayer];
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
- (CVPixelBufferRef) pixelBufferFromCGImage: (CGImageRef) image
|
|
|
|
{
|
2013-08-21 20:44:09 +08:00
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
CGSize frameSize = CGSizeMake(CGImageGetWidth(image), CGImageGetHeight(image));
|
|
|
|
NSDictionary *options = [NSDictionary dictionaryWithObjectsAndKeys:
|
|
|
|
[NSNumber numberWithBool:NO], kCVPixelBufferCGImageCompatibilityKey,
|
|
|
|
[NSNumber numberWithBool:NO], kCVPixelBufferCGBitmapContextCompatibilityKey,
|
|
|
|
nil];
|
|
|
|
CVPixelBufferRef pxbuffer = NULL;
|
|
|
|
CVReturn status = CVPixelBufferCreate(kCFAllocatorDefault, frameSize.width,
|
|
|
|
frameSize.height, kCVPixelFormatType_32ARGB, (CFDictionaryRef) CFBridgingRetain(options),
|
|
|
|
&pxbuffer);
|
|
|
|
NSParameterAssert(status == kCVReturnSuccess && pxbuffer != NULL);
|
2013-08-21 20:44:09 +08:00
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
CVPixelBufferLockBaseAddress(pxbuffer, 0);
|
|
|
|
void *pxdata = CVPixelBufferGetBaseAddress(pxbuffer);
|
2013-08-21 20:44:09 +08:00
|
|
|
|
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
CGColorSpaceRef rgbColorSpace = CGColorSpaceCreateDeviceRGB();
|
|
|
|
CGContextRef context = CGBitmapContextCreate(pxdata, frameSize.width,
|
|
|
|
frameSize.height, 8, 4*frameSize.width, rgbColorSpace,
|
|
|
|
kCGImageAlphaPremultipliedFirst);
|
2013-08-21 20:44:09 +08:00
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
CGContextDrawImage(context, CGRectMake(0, 0, CGImageGetWidth(image),
|
|
|
|
CGImageGetHeight(image)), image);
|
|
|
|
CGColorSpaceRelease(rgbColorSpace);
|
|
|
|
CGContextRelease(context);
|
2013-08-21 20:44:09 +08:00
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
CVPixelBufferUnlockBaseAddress(pxbuffer, 0);
|
2013-08-21 20:44:09 +08:00
|
|
|
|
2013-06-14 19:10:25 +08:00
|
|
|
return pxbuffer;
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
|
|
|
|
#pragma mark - Protocol AVCaptureVideoDataOutputSampleBufferDelegate
|
|
|
|
|
|
|
|
|
|
|
|
- (void)captureOutput:(AVCaptureOutput *)captureOutput didOutputSampleBuffer:(CMSampleBufferRef)sampleBuffer fromConnection:(AVCaptureConnection *)connection
|
|
|
|
{
|
2014-10-17 19:45:13 +08:00
|
|
|
(void)captureOutput;
|
|
|
|
(void)connection;
|
2016-09-14 19:48:41 +08:00
|
|
|
auto strongDelegate = self.delegate;
|
|
|
|
if (strongDelegate) {
|
2012-10-17 07:18:30 +08:00
|
|
|
|
|
|
|
// convert from Core Media to Core Video
|
|
|
|
CVImageBufferRef imageBuffer = CMSampleBufferGetImageBuffer(sampleBuffer);
|
|
|
|
CVPixelBufferLockBaseAddress(imageBuffer, 0);
|
|
|
|
|
|
|
|
void* bufferAddress;
|
|
|
|
size_t width;
|
|
|
|
size_t height;
|
|
|
|
size_t bytesPerRow;
|
|
|
|
|
|
|
|
CGColorSpaceRef colorSpace;
|
|
|
|
CGContextRef context;
|
|
|
|
|
|
|
|
int format_opencv;
|
|
|
|
|
|
|
|
OSType format = CVPixelBufferGetPixelFormatType(imageBuffer);
|
|
|
|
if (format == kCVPixelFormatType_420YpCbCr8BiPlanarFullRange) {
|
|
|
|
|
|
|
|
format_opencv = CV_8UC1;
|
|
|
|
|
|
|
|
bufferAddress = CVPixelBufferGetBaseAddressOfPlane(imageBuffer, 0);
|
|
|
|
width = CVPixelBufferGetWidthOfPlane(imageBuffer, 0);
|
|
|
|
height = CVPixelBufferGetHeightOfPlane(imageBuffer, 0);
|
|
|
|
bytesPerRow = CVPixelBufferGetBytesPerRowOfPlane(imageBuffer, 0);
|
|
|
|
|
|
|
|
} else { // expect kCVPixelFormatType_32BGRA
|
|
|
|
|
|
|
|
format_opencv = CV_8UC4;
|
|
|
|
|
|
|
|
bufferAddress = CVPixelBufferGetBaseAddress(imageBuffer);
|
|
|
|
width = CVPixelBufferGetWidth(imageBuffer);
|
|
|
|
height = CVPixelBufferGetHeight(imageBuffer);
|
|
|
|
bytesPerRow = CVPixelBufferGetBytesPerRow(imageBuffer);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// delegate image processing to the delegate
|
2014-10-17 20:46:47 +08:00
|
|
|
cv::Mat image((int)height, (int)width, format_opencv, bufferAddress, bytesPerRow);
|
2012-10-17 07:18:30 +08:00
|
|
|
|
|
|
|
CGImage* dstImage;
|
|
|
|
|
2016-09-14 19:48:41 +08:00
|
|
|
if ([strongDelegate respondsToSelector:@selector(processImage:)]) {
|
|
|
|
[strongDelegate processImage:image];
|
2012-10-17 07:18:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// check if matrix data pointer or dimensions were changed by the delegate
|
|
|
|
bool iOSimage = false;
|
2014-10-17 20:32:53 +08:00
|
|
|
if (height == (size_t)image.rows && width == (size_t)image.cols && format_opencv == image.type() && bufferAddress == image.data && bytesPerRow == image.step) {
|
2012-10-17 07:18:30 +08:00
|
|
|
iOSimage = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// (create color space, create graphics context, render buffer)
|
|
|
|
CGBitmapInfo bitmapInfo;
|
|
|
|
|
|
|
|
// basically we decide if it's a grayscale, rgb or rgba image
|
|
|
|
if (image.channels() == 1) {
|
|
|
|
colorSpace = CGColorSpaceCreateDeviceGray();
|
|
|
|
bitmapInfo = kCGImageAlphaNone;
|
|
|
|
} else if (image.channels() == 3) {
|
|
|
|
colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
|
|
bitmapInfo = kCGImageAlphaNone;
|
|
|
|
if (iOSimage) {
|
|
|
|
bitmapInfo |= kCGBitmapByteOrder32Little;
|
|
|
|
} else {
|
|
|
|
bitmapInfo |= kCGBitmapByteOrder32Big;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
colorSpace = CGColorSpaceCreateDeviceRGB();
|
|
|
|
bitmapInfo = kCGImageAlphaPremultipliedFirst;
|
|
|
|
if (iOSimage) {
|
|
|
|
bitmapInfo |= kCGBitmapByteOrder32Little;
|
|
|
|
} else {
|
|
|
|
bitmapInfo |= kCGBitmapByteOrder32Big;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (iOSimage) {
|
|
|
|
context = CGBitmapContextCreate(bufferAddress, width, height, 8, bytesPerRow, colorSpace, bitmapInfo);
|
|
|
|
dstImage = CGBitmapContextCreateImage(context);
|
|
|
|
CGContextRelease(context);
|
|
|
|
} else {
|
|
|
|
|
|
|
|
NSData *data = [NSData dataWithBytes:image.data length:image.elemSize()*image.total()];
|
|
|
|
CGDataProviderRef provider = CGDataProviderCreateWithCFData((__bridge CFDataRef)data);
|
|
|
|
|
|
|
|
// Creating CGImage from cv::Mat
|
|
|
|
dstImage = CGImageCreate(image.cols, // width
|
|
|
|
image.rows, // height
|
|
|
|
8, // bits per component
|
|
|
|
8 * image.elemSize(), // bits per pixel
|
|
|
|
image.step, // bytesPerRow
|
|
|
|
colorSpace, // colorspace
|
|
|
|
bitmapInfo, // bitmap info
|
|
|
|
provider, // CGDataProviderRef
|
|
|
|
NULL, // decode
|
|
|
|
false, // should interpolate
|
|
|
|
kCGRenderingIntentDefault // intent
|
|
|
|
);
|
|
|
|
|
|
|
|
CGDataProviderRelease(provider);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// render buffer
|
|
|
|
dispatch_sync(dispatch_get_main_queue(), ^{
|
|
|
|
self.customPreviewLayer.contents = (__bridge id)dstImage;
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2015-09-19 20:05:38 +08:00
|
|
|
recordingCountDown--;
|
|
|
|
if (self.recordVideo == YES && recordingCountDown < 0) {
|
2012-10-17 07:18:30 +08:00
|
|
|
lastSampleTime = CMSampleBufferGetPresentationTimeStamp(sampleBuffer);
|
2012-08-22 17:33:13 +08:00
|
|
|
// CMTimeShow(lastSampleTime);
|
2012-10-17 07:18:30 +08:00
|
|
|
if (self.recordAssetWriter.status != AVAssetWriterStatusWriting) {
|
|
|
|
[self.recordAssetWriter startWriting];
|
|
|
|
[self.recordAssetWriter startSessionAtSourceTime:lastSampleTime];
|
|
|
|
if (self.recordAssetWriter.status != AVAssetWriterStatusWriting) {
|
|
|
|
NSLog(@"[Camera] Recording Error: asset writer status is not writing: %@", self.recordAssetWriter.error);
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
NSLog(@"[Camera] Video recording started");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (self.recordAssetWriterInput.readyForMoreMediaData) {
|
2013-06-14 19:10:25 +08:00
|
|
|
CVImageBufferRef pixelBuffer = [self pixelBufferFromCGImage:dstImage];
|
|
|
|
if (! [self.recordPixelBufferAdaptor appendPixelBuffer:pixelBuffer
|
2012-10-17 07:18:30 +08:00
|
|
|
withPresentationTime:lastSampleTime] ) {
|
|
|
|
NSLog(@"Video Writing Error");
|
|
|
|
}
|
2015-09-19 20:05:38 +08:00
|
|
|
if (pixelBuffer != nullptr)
|
|
|
|
CVPixelBufferRelease(pixelBuffer);
|
2012-10-17 07:18:30 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// cleanup
|
|
|
|
CGImageRelease(dstImage);
|
|
|
|
|
|
|
|
CGColorSpaceRelease(colorSpace);
|
|
|
|
|
|
|
|
CVPixelBufferUnlockBaseAddress(imageBuffer, 0);
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
- (void)updateOrientation;
|
|
|
|
{
|
2013-06-14 19:10:25 +08:00
|
|
|
if (self.rotateVideo == YES)
|
|
|
|
{
|
|
|
|
NSLog(@"rotate..");
|
|
|
|
self.customPreviewLayer.bounds = CGRectMake(0, 0, self.parentView.frame.size.width, self.parentView.frame.size.height);
|
|
|
|
[self layoutPreviewLayer];
|
|
|
|
}
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
- (void)saveVideo;
|
|
|
|
{
|
|
|
|
if (self.recordVideo == NO) {
|
|
|
|
return;
|
|
|
|
}
|
2012-10-17 07:18:30 +08:00
|
|
|
|
2017-10-06 01:22:56 +08:00
|
|
|
UISaveVideoAtPathToSavedPhotosAlbum([self videoFileString], nil, nil, NULL);
|
2012-08-20 23:36:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-08-22 17:33:13 +08:00
|
|
|
- (NSURL *)videoFileURL;
|
2012-08-20 23:36:36 +08:00
|
|
|
{
|
|
|
|
NSString *outputPath = [[NSString alloc] initWithFormat:@"%@%@", NSTemporaryDirectory(), @"output.mov"];
|
|
|
|
NSURL *outputURL = [NSURL fileURLWithPath:outputPath];
|
|
|
|
NSFileManager *fileManager = [NSFileManager defaultManager];
|
|
|
|
if ([fileManager fileExistsAtPath:outputPath]) {
|
|
|
|
NSLog(@"file exists");
|
|
|
|
}
|
|
|
|
return outputURL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2012-08-22 17:33:13 +08:00
|
|
|
- (NSString *)videoFileString;
|
2012-08-20 23:36:36 +08:00
|
|
|
{
|
|
|
|
NSString *outputPath = [[NSString alloc] initWithFormat:@"%@%@", NSTemporaryDirectory(), @"output.mov"];
|
|
|
|
return outputPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
@end
|