AVFoundation框架(五) 媒体捕捉上- 简单的录制

1. 媒体捕捉的类

AVFoundation的照片和视频捕捉功能是它的强项.先看一下其中捕捉相关的类.

找到唯一比较清楚的图.png
  • 捕捉会话 AVCaptureSession:
    AVCaptureSession相当于一个虚拟的插座,连接了输入和输出资源.它管理着从物理设备(摄像头和麦克风)得到的数据流,然后输出到其他地方.
    可以额外配置一个会话预设值,用来控制捕捉数据的格式和质量,默认是AVCaptureSessionPresetHigh
  • 捕捉设备 AVCaptureDevice:
    AVCaptureDevice为物理设备定义一个接口和大量控制方法,例如对焦、曝光、白平衡和闪光等。
  • 捕捉设备输入 AVCaptureDeviceInput:
    在使用AVCaptureDevice进行处理前,需要将它封装到AVCaptureDeviceInput实例中.因为一个捕捉设备不能直接添加到AVCaptureSession中; 如下:
NSError *error;
AVCaptureDeviceInput *input = [AVCaptureDeviceInput deviceInputWithDevice:videoDevice error:&error];
  • 捕捉的输出 AVCaptureOutput:
    AVCaptureOutput有许多扩展类,它本身只是一个抽象基类,用于为从Session得到的数据寻找输出目的地. 看上面结构图就可以看出各扩展类功能方向, 这里单独说一下, AVCaptureAudioDataOutputAVCaptureVideoDataOutput可以直接访问硬件扑捉到的数字样本,可以用于音视频流进行实施处理.
  • 捕捉连接 AVCaptureConnection
    这个类其实就是上图中连接不同组件的连接箭头所表示. 对这些连接的访问可以让开发者对信号流就行底层控制,比如禁用某些特定的连接,或者音频连接中限制单独的音频轨道.
  • 捕捉预览 AVCaptureVideoPreviewLayer
    这个类不在上图中, 它是对捕捉视频数据进行实时预览.在视频角色中类似于AVPlayerLayer

捕捉会话应用流程:

  • 第一步:配置会话
- (BOOL)setupSession:(NSError **)error {
    // 1. 创建捕捉会话
    self.captureSession = [[AVCaptureSession alloc] init];                  
    self.captureSession.sessionPreset = AVCaptureSessionPresetHigh;

     //2. 根据要生成的媒体类型获取捕捉设备,并添加到Session上.
    AVCaptureDevice *videoDevice =                                          
        [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeVideo];
           // 在把此捕捉设备添加到AVCaptureSession之前,先封装成一个input对象.
    AVCaptureDeviceInput *videoInput =                                      
        [AVCaptureDeviceInput deviceInputWithDevice:videoDevice error:error];
    if (videoInput) {
        if ([self.captureSession canAddInput:videoInput]) {  // 测试是否可以add
            [self.captureSession addInput:videoInput];
            self.activeVideoInput = videoInput;
        }
    } else {
        return NO;
    }
/*// 如果要获取音频捕捉部分
    AVCaptureDevice *audioDevice =                                          
        [AVCaptureDevice defaultDeviceWithMediaType:AVMediaTypeAudio];
    AVCaptureDeviceInput *audioInput =                                      
        [AVCaptureDeviceInput deviceInputWithDevice:audioDevice error:error];
    if (audioInput) {
        if ([self.captureSession canAddInput:audioInput]) {                
            [self.captureSession addInput:audioInput];
        }
    } else {
        return NO;
    }
*/

     //3. 设置捕捉会话的输出部分
        // image输出
    self.imageOutput = [[AVCaptureStillImageOutput alloc] init];            
    self.imageOutput.outputSettings = @{AVVideoCodecKey : AVVideoCodecJPEG};

    if ([self.captureSession canAddOutput:self.imageOutput]) {
        [self.captureSession addOutput:self.imageOutput];
    }

      // movie输出
    self.movieOutput = [[AVCaptureMovieFileOutput alloc] init];             

    if ([self.captureSession canAddOutput:self.movieOutput]) {
        [self.captureSession addOutput:self.movieOutput];
    }

    return YES;
}
  • 第二: 启动和停止会话
- (void)startSession {
    if (![self.captureSession isRunning]) {                                 
        dispatch_async([self globalQueue], ^{
            [self.captureSession startRunning];
        });
    }
}

- (void)stopSession {
    if ([self.captureSession isRunning]) {                                  
        dispatch_async([self globalQueue], ^{
            [self.captureSession stopRunning];
        });
    }
}
// 开始和停止会话都是同步耗时操作,所以采用异步执行.
- (dispatch_queue_t)globalQueue {
    return dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
}

iOS 8之后使用硬件设备都需要用户给予权限,使用前都要判断下. 例如:

AVAuthorizationStatus authStatus = [AVCaptureDevice authorizationStatusForMediaType:AVMediaTypeVideo];
    if(authStatus == AVAuthorizationStatusRestricted || authStatus == AVAuthorizationStatusDenied) {
  NSLog(@"相机权限受限!");
}

2. 细节部分.

2.1 捕捉设备的修改与配置

  • 获取摄像头,切换摄像头.
// 当前捕捉会话对应的摄像头.
- (AVCaptureDevice *)activeCamera {                                         
    return self.activeVideoInput.device;
}
// 当前未激活的摄像头 
- (AVCaptureDevice *)inactiveCamera {                                      
    AVCaptureDevice *device = nil;
    if (self.cameraCount > 1) {
        if ([self activeCamera].position == AVCaptureDevicePositionBack) {  
            device = [self cameraWithPosition:AVCaptureDevicePositionFront];
        } else {
            device = [self cameraWithPosition:AVCaptureDevicePositionBack];
        }
    }
    return device;
}
// 切换摄像头
- (BOOL)switchCameras {

    if (![self canSwitchCameras]) {                                         
        return NO;
    }

    // 1. 获取当前未使用的摄像头,并为他创建一个新的Input.
    NSError *error;
    AVCaptureDevice *videoDevice = [self inactiveCamera];                   
    AVCaptureDeviceInput *videoInput =
    [AVCaptureDeviceInput deviceInputWithDevice:videoDevice error:&error];
    // 2. 用新的Input替换掉正在激活的Input
    if (videoInput) {
        [self.captureSession beginConfiguration];    // 原子性配置开始.保证线程安全

        [self.captureSession removeInput:self.activeVideoInput];            

        if ([self.captureSession canAddInput:videoInput]) {                 
            [self.captureSession addInput:videoInput];
            self.activeVideoInput = videoInput;
        } else { // 确保安全,如果新的Input不能添加,继续使用旧的
            [self.captureSession addInput:self.activeVideoInput];
        }

        [self.captureSession commitConfiguration];    // 原子性配置完成

    } else {    // 错误处理
        [self.delegate deviceConfigurationFailedWithError:error];           
        return NO;
    }

    return YES;
}

- (BOOL)canSwitchCameras {                                                  
    return self.cameraCount > 1;
}

- (NSUInteger)cameraCount {                                                 
    return [[AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo] count];
}

// 获取指定位置的AVCaptureDevice
- (AVCaptureDevice *)cameraWithPosition:(AVCaptureDevicePosition)position { 
    NSArray *devices = [AVCaptureDevice devicesWithMediaType:AVMediaTypeVideo];
    for (AVCaptureDevice *device in devices) {  
        if (device.position == position) {
            return device;
        }
    }
    return nil;
}
  • 调整焦距和曝光
    修改AVCaptureDevice配置设备时,一定要先测试修改动作是否被硬件设备支持. 比如前置摄像头不支持对焦操作.尝试一个不被支持的修改动作就会导致程序崩溃. 所以任何修改都要先通过判断一下.
/* 调整对焦 */
- (void)focusAtPoint:(CGPoint)point {                                      
    AVCaptureDevice *device = self.activeVideoInput.device;

    if (device.isFocusPointOfInterestSupported &&                          
        [device isFocusModeSupported:AVCaptureFocusModeAutoFocus]) {     // 判断设备是否支持兴趣点对焦和自动对焦模式.

        NSError *error;
        if ([device lockForConfiguration:&error]) {    // 锁定设备,准备配置修改. 之后修改完释放锁定.
            device.focusPointOfInterest = point;
            device.focusMode = AVCaptureFocusModeAutoFocus;

            [device unlockForConfiguration];
        } else {
            [self.delegate deviceConfigurationFailedWithError:error];
        }
    }
}


/* 
调整曝光 
曝光模式定义了四种方式: 
    AVCaptureExposureModeLocked                            锁定
    AVCaptureExposureModeAutoExpose                    调整一次并锁定
    AVCaptureExposureModeContinuousAutoExposure            一直自动调整
    AVCaptureExposureModeCustom                 
大部分设备是根据环境自动调整, 现在我们做调整曝光度并锁定.  但在iOS中并不支持AVCaptureExposureModeAutoExpose, 所以我们要用其他定义来实现.这个功能
*/
static const NSString *THCameraAdjustingExposureContext;
- (void)exposeAtPoint:(CGPoint)point {

    AVCaptureDevice *device = self.activeVideoInput.device;

    AVCaptureExposureMode exposureMode =
    AVCaptureExposureModeContinuousAutoExposure;

    if (device.isExposurePointOfInterestSupported &&
        [device isExposureModeSupported:exposureMode]) {    // 判断设备是否支持AutoExposure模式

        NSError *error;
        if ([device lockForConfiguration:&error]) {    // 锁定设备,准备配置修改. 之后修改完释放锁定.

            device.exposurePointOfInterest = point;
            device.exposureMode = exposureMode;

            // 判断设备是否支持锁定曝光设置, 如果支持,使用KVO来观察设备"adjustingExposure"属性状态. 然后在曝光调整完成时在该点上锁定曝光.
            if ([device isExposureModeSupported:AVCaptureExposureModeLocked]) {
                [device addObserver:self                                    
                         forKeyPath:@"adjustingExposure"
                            options:NSKeyValueObservingOptionNew
                            context:&THCameraAdjustingExposureContext];
            }

            [device unlockForConfiguration];
        } else {
            [self.delegate deviceConfigurationFailedWithError:error];
        }
    }
}

- (void)observeValueForKeyPath:(NSString *)keyPath
                      ofObject:(id)object
                        change:(NSDictionary *)change
                       context:(void *)context {

    if (context == &THCameraAdjustingExposureContext) {    
    // 这里通过测试Context是否为&THCameraAdjustingExposureContext指针,来判断监听到的回调是否对应我们期望的变更操作.

        AVCaptureDevice *device = (AVCaptureDevice *)object;
        if (!device.isAdjustingExposure &&                                  
            [device isExposureModeSupported:AVCaptureExposureModeLocked]) {

            [object removeObserver:self                                     
                        forKeyPath:@"adjustingExposure"
                           context:&THCameraAdjustingExposureContext];
            // 回到主队列设置exposureMode 
            dispatch_async(dispatch_get_main_queue(), ^{                    
                NSError *error;
                if ([device lockForConfiguration:&error]) {
                    device.exposureMode = AVCaptureExposureModeLocked;
                    [device unlockForConfiguration];
                } else {
                    [self.delegate deviceConfigurationFailedWithError:error];
                }
            });
        }

    } else {
        [super observeValueForKeyPath:keyPath
                             ofObject:object
                               change:change
                              context:context];
    }
}

/* 恢复对焦和曝光设置 */
- (void)resetFocusAndExposureModes {

    AVCaptureDevice *device = self.activeVideoInput.device;

    AVCaptureExposureMode exposureMode =
    AVCaptureExposureModeContinuousAutoExposure;

    AVCaptureFocusMode focusMode = AVCaptureFocusModeContinuousAutoFocus;

    BOOL canResetFocus = [device isFocusPointOfInterestSupported] &&        
    [device isFocusModeSupported:focusMode];

    BOOL canResetExposure = [device isExposurePointOfInterestSupported] &&  
    [device isExposureModeSupported:exposureMode];

    CGPoint centerPoint = CGPointMake(0.5f, 0.5f);    // 创建一个中心扫描点.

    NSError *error;
    if ([device lockForConfiguration:&error]) {

        if (canResetFocus) {    // 焦点可以重置
            device.focusMode = focusMode;
            device.focusPointOfInterest = centerPoint;
        }

        if (canResetExposure) {    // 曝光可以充值
            device.exposureMode = exposureMode;
            device.exposurePointOfInterest = centerPoint;
        }
        
        [device unlockForConfiguration];
        
    } else {
        [self.delegate deviceConfigurationFailedWithError:error];
    }
}
  • 调整闪光灯和手电筒模式(Flash和 Torch 模式)
    AVCaptureDevice可以控制摄像头的LED灯,当拍照时是做闪光灯,当拍视频时做手电筒.
- (void)setFlashMode:(AVCaptureFlashMode)flashMode {

    AVCaptureDevice *device =  self.activeVideoInput.device;

    if (device.flashMode != flashMode &&
        [device isFlashModeSupported:flashMode]) {

        NSError *error;
        if ([device lockForConfiguration:&error]) {
            device.flashMode = flashMode;
            [device unlockForConfiguration];
        } else {
            [self.delegate deviceConfigurationFailedWithError:error];
        }
    }
}


- (void)setTorchMode:(AVCaptureTorchMode)torchMode {

    AVCaptureDevice *device = self.activeVideoInput.device;

    if (device.torchMode != torchMode &&
        [device isTorchModeSupported:torchMode]) {

        NSError *error;
        if ([device lockForConfiguration:&error]) {
            device.torchMode = torchMode;
            [device unlockForConfiguration];
        } else {
            [self.delegate deviceConfigurationFailedWithError:error];
        }
    }
}

3. 基本功能实现-拍照和摄像

  • 拍照
    上面介绍了基本流程,如果要拍照就是希望捕捉会话的输出Output实例是AVCaptureStillImageOutput类型,这需要在第一步配置会话时添加上去. 而关于接下来具体拍摄方法就定义在AVCaptureStillImageOutput中. 代码如下:
- (void)captureStillImage {
    // 获取Output对象当前使用的AVCaptureConnection连接.
    // 原始相机只支持垂直方向,所以当设备旋转时,用户界面保持不变.我们希望横向时相应调整图片方向.
    AVCaptureConnection *connection =                                   
        [self.imageOutput connectionWithMediaType:AVMediaTypeVideo];
        // 这里要先确定Connection 是否支持设置视频方向. 然后设置图片方向.
    if (connection.isVideoOrientationSupported) {    
        connection.videoOrientation = [self currentVideoOrientation];
    }
    

    id handler = ^(CMSampleBufferRef sampleBuffer, NSError *error) {
        if (sampleBuffer != NULL) {    
            NSData *imageData =
                [AVCaptureStillImageOutput
                    jpegStillImageNSDataRepresentation:sampleBuffer];
           // 得到的照片
            UIImage *image = [[UIImage alloc] initWithData:imageData];
        } else {
            NSLog(@"NULL sampleBuffer: %@", [error localizedDescription]);
        }
    };
    // 拍摄静态图片
    [self.imageOutput captureStillImageAsynchronouslyFromConnection:connection
                                                  completionHandler:handler];
}
  • 视频
    同拍照流程一样,设置捕捉会话的输出Output实例是 AVCaptureMovieFileOutput类型. 具体摄像代码如下:
- (void)startRecording {

    if (!self.movieOutput.isRecording) { // 是否在录制中
        // 设置录制视频方向为当前方向
        AVCaptureConnection *videoConnection =                              
            [self.movieOutput connectionWithMediaType:AVMediaTypeVideo];
        if ([videoConnection isVideoOrientationSupported]) {                
            videoConnection.videoOrientation = self.currentVideoOrientation;
        }
        // 设置VideoStabilization.可以提高视频质量.
        if ([videoConnection isVideoStabilizationSupported]) {              
            if ([[[UIDevice currentDevice] systemVersion] floatValue] < 8.0) {
                videoConnection.enablesVideoStabilizationWhenAvailable = YES;
            } else {
                videoConnection.preferredVideoStabilizationMode = AVCaptureVideoStabilizationModeAuto;
            }
        }

        AVCaptureDevice *device = self.activeVideoInput.device;
        // 设置平滑对焦模式,提升移动录制质量
        if (device.isSmoothAutoFocusSupported) {                            
            NSError *error;
            if ([device lockForConfiguration:&error]) {
                device.smoothAutoFocusEnabled = NO;
                [device unlockForConfiguration];
            } else {
                [self.delegate deviceConfigurationFailedWithError:error];
            }
        }

        NSURL *outputURL = [self uniqueURL];   // 定义视频文件唯一输出路径.        
        // 最后,开始录制. 通过AVCaptureFileOutputRecordingDelegate控制录制过程.
        [self.movieOutput startRecordingToOutputFileURL:outputURL      
                                      recordingDelegate:self];

    }
}

AVCaptureMovieFileOutput支持分段捕捉.

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容