iOS视频开发:视频的录制一

一、UIImagePickerController

UIImagePickerControllerUIKit框架里面的一个class,通过这个系统提供的class我们可以简单的是实现拍照、录制视频和音频。

三个步骤:
  1. 当前控制器present一个UIImagePickerController
  2. 在当前界面就可以拍照、录制视频和音频
  3. 实现UIImagePickerControllerdelegate,在delegate可以获取录制的视频和音频,来进行相应的操作.
定制化UIImagePickerController
//    查看摄像头是否可用
if ([UIImagePickerController isSourceTypeAvailable:UIImagePickerControllerSourceTypeCamera] == NO) {   
  return;
}
UIImagePickerController *imagePick = [[UIImagePickerController alloc]init];
imagePick.sourceType = UIImagePickerControllerSourceTypeCamera;
//    我们还可以设置照片和视频拍摄的质量、是否可以开启闪光灯、是否开启手电筒
//    还可以单独设置只支持视频模式
//    imagePick.mediaTypes = [[NSArray alloc] initWithObjects: (NSString *) kUTTypeMovie, nil];
imagePick.mediaTypes = [UIImagePickerController availableMediaTypesForSourceType:UIImagePickerControllerSourceTypeCamera];
//    UINavigationControllerDelegate,UIImagePickerControllerDelegate;
imagePick.delegate = self;
//拍照或者录制结束后是否可以编辑
imagePick.allowsEditing = NO;
[self presentViewController:imagePick animated:YES completion:nil];

界面的自定义

cameraOverlayView属性可以自定义UIImagePickerController界面顶部的控件,但是只在UIImagePickerControllermediaTypesUIImagePickerControllerSourceTypeCamera时可用。

实现UIImagePickerControllerdelegate
- (void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info{
    
    NSString *mediaType = [info objectForKey:UIImagePickerControllerMediaType]; 
    UIImage *originalImage, *editedImage, *imageToSave;
    //    处理图片
    if ([mediaType isEqualToString:(NSString *)kUTTypeImage]) {
        editedImage = info[UIImagePickerControllerOriginalImage];
        originalImage = info[UIImagePickerControllerEditedImage];
        if (editedImage) {
            imageToSave = editedImage;
        }else{
            imageToSave = originalImage;
        }
        UIImageWriteToSavedPhotosAlbum(imageToSave, nil, nil, nil);
    }
    //处理视频
    if ([mediaType isEqualToString:(NSString *)kUTTypeMovie]) {
        NSString *url = [info[UIImagePickerControllerMediaURL] path];
        if (UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(url)) {
            UISaveVideoAtPathToSavedPhotosAlbum(url, nil, nil, nil);
        }
    }
    [picker dismissViewControllerAnimated:YES completion:nil];
    
    
}

二、AVCaptureSession & AVCaptureMovieFileOutput

要获取摄像机捕捉到的视频或者麦克风捕捉到的音频,我们需要对象表示inputs和outputs,并使用AVCaptureSession的实例来协调它们之间的数据流。

  1. AVCaptureDevice 对象,表示声音或者视频采集设备,对应摄像头和麦克风。
  2. AVCaptureInput的子类,配置输入端口。
  3. AVCaptureOutput的子类, 输出采集到的视频或者图像。
  4. AVCaptureSession来协调从输入到输出的数据流。
步骤一: 创建AVCaptureDevice 对象

因为我们需要录制视频和音频所以我们需要视频的AVCaptureDevice和音频的AVCaptureDevice。

//我们同时获取了前摄像头和后摄像头因为等会我们要手动切换
//获取音频device
-(AVCaptureDevice *)audioDevice{
    if (!_audioDevice) {
        _audioDevice = [AVCaptureDevice devicesWithMediaType:AVMediaTypeAudio].firstObject;
    }
    return _audioDevice;
}   
   
//后置摄像头
-(AVCaptureDevice *)backVideoDevice{
    if (!_backVideoDevice) {
        _backVideoDevice = [self getDeviceBy:AVCaptureDevicePositionBack];
        if ([self.currentVideoDevice isTorchAvailable] && [_backVideoDevice isTorchModeSupported:AVCaptureTorchModeOn]) {
            //可以设置是否开启闪光灯,是否开始HDR、视频防抖、白平衡什么的
            //设置device之前需要先 lockForConfiguration
            if ([_backVideoDevice lockForConfiguration:NULL]==YES) {
                self.currentVideoDevice.torchMode = AVCaptureTorchModeOn;
                [self.currentVideoDevice unlockForConfiguration];
            }
        }
        
    }
    }
    return _backVideoDevice;
}
//前置摄像头
-(AVCaptureDevice *)frontVideoDevice{
    if (!_frontVideoDevice) {
        _frontVideoDevice = [self getDeviceBy:AVCaptureDevicePositionFront];
    }
    return _frontVideoDevice;
}
步骤二、配置inputs

每个AVCaptureDevice对应一个input。

//audio input
- (AVCaptureDeviceInput *)audioInput{
    if (!_audioInput) {
        NSError *error = nil;
        _audioInput = [AVCaptureDeviceInput deviceInputWithDevice:self.audioDevice error:&error];
        
    }
    return _audioInput;
}
- (AVCaptureDeviceInput *)videoInput{
    if (!_videoInput) {
        NSError *error  = nil;
        _videoInput = [AVCaptureDeviceInput deviceInputWithDevice:self.currentVideoDevice error:&error];
        
        
    }
    return _videoInput;
}
步骤三: 写入文件

Output有四种:

AVCaptureMovieFileOutput 写入文件
AVCaptureVideoDataOutput 加工视频输出
AVCaptureAudioDataOutput 加工音频输出
AVCaptureStillImageOutput 捕捉输出的图像

写入文件只需要AVCaptureMovieFileOutput就可以了。

// output
- (AVCaptureMovieFileOutput  *)movieFileOutput{
    if (!_movieFileOutput) {
        _movieFileOutput = [[AVCaptureMovieFileOutput  alloc]init];
        //CMTime drution = CMTimeMake(1, 60);
        //设置视频录制时间限制 kCMTimeInvalid(无限制)
        _movieFileOutput.maxRecordedDuration = kCMTimeInvalid;
        // 文件大小限制
        //_movieFileOutput.maxRecordedFileSize = 1024 * 1024;
        AVCaptureConnection *videoConnection = [_movieFileOutput connectionWithMediaType:AVMediaTypeVideo];
        // 是否支持科学防抖
        if ([videoConnection isVideoStabilizationSupported]) {
            videoConnection.preferredVideoStabilizationMode = AVCaptureVideoStabilizationModeAuto;
        }
        videoConnection.videoOrientation = self.previewLayer.connection.videoOrientation;
    }
    return _movieFileOutput;
}
步骤四: 获取AVCaptureSession
- (AVCaptureSession *)session{
    if (!_session) {
        _session = [[AVCaptureSession alloc]init];
        // 设置视频质量
        if ([_session canSetSessionPreset:AVCaptureSessionPresetLow]) {
            [_session setSessionPreset:AVCaptureSessionPresetLow];
        }
        //增加videoinput
        if ([_session canAddInput:self.videoInput]) {
            [_session addInput:self.videoInput];
        }
        //增加videoinput
        if ([_session canAddInput:self.audioInput]) {
            [_session addInput:self.audioInput];
        }
        //增加fileOutput
        if ([_session canAddOutput:self.movieFileOutput]) {
            [_session addOutput:self.movieFileOutput];
        }
    }
    return _session;
}

这个地方需要注意下,每次我们更改AVCaptureSession的属性的时候我们都需要:

[session beginConfiguration];
// Remove an existing capture device.
// Add a new capture device.
// Reset the preset.
[session commitConfiguration];

现在就可以录制视频并写入文件了

为了实时查看我们录制的内容,我们加一个预览层。

-(AVCaptureVideoPreviewLayer *)previewLayer{
    if (!_previewLayer) {
        _previewLayer = [[AVCaptureVideoPreviewLayer alloc] initWithSession:self.session];
        _previewLayer.frame = [UIScreen mainScreen].bounds;
        _previewLayer.videoGravity = AVLayerVideoGravityResizeAspectFill;
        _previewLayer.connection.videoOrientation = AVCaptureVideoOrientationPortrait;
    }
    return _previewLayer;
}

在控制器里面调用session的startRuning方法,这个时候只是采集到了视频显示在了预览层上面,并未开始录制。

- (void)viewDidLoad {
    [super viewDidLoad];
    self.recodingView.delegate = self;
    [self.view.layer insertSublayer:self.previewLayer atIndex:0];
    [self.session startRunning];
   
   
}

点击录制视频,recodingView是我自定义的控件。

-(void)writePath{
    if ([self.movieFileOutput isRecording] ) {
        [self.movieFileOutput stopRecording];
        return;
    }
    NSDateFormatter * dateFormatter = [[NSDateFormatter alloc] init ];
    [dateFormatter setDateFormat:@"yyyyMMddHHmmss"];
    NSString * fileName = [[dateFormatter stringFromDate:[NSDate date]] stringByAppendingString:@".mov"];
    NSString * filePath = [[NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES) lastObject] stringByAppendingPathComponent:fileName];
    NSURL *filePathUrl = [NSURL fileURLWithPath:filePath];
    //写文件到指定的路径,并设置代理
    [self.movieFileOutput startRecordingToOutputFileURL:filePathUrl recordingDelegate:self];
}

设置代理,在视频录制的过程中会发生许多情况,比如说突然来电话,摄像头被其他程序占用,系统会发送相应的通知给我们。

- (void)captureOutput:(AVCaptureFileOutput *)captureOutput
didFinishRecordingToOutputFileAtURL:(NSURL *)outputFileURL
      fromConnections:(NSArray *)connections
                error:(NSError *)error{
    BOOL recordSuccessfully = YES;
    if ([error code] != noErr) {
        id value = [[error userInfo] objectForKey:AVErrorRecordingSuccessfullyFinishedKey];
        if (value) {
            recordSuccessfully = [value boolValue];
        }
    }
    
//   有error的话 有可能也是录制成功了
    /*
    AVErrorMaximumDurationReached  时间限制
    AVErrorMaximumFileSizeReached  文件大小限制
    AVErrorDiskFull                磁盘已满
    AVErrorDeviceWasDisconnected   device连接失败
    AVErrorSessionWasInterrupted   被切断(比如说来电话了)
*/
    
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 220,884评论 6 513
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 94,212评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 167,351评论 0 360
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,412评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,438评论 6 397
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 52,127评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,714评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,636评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 46,173评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,264评论 3 339
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,402评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 36,073评论 5 347
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,763评论 3 332
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,253评论 0 23
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,382评论 1 271
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,749评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,403评论 2 358

推荐阅读更多精彩内容