想要实现这个功能,可以先了解一下有关录制视频的相关知识:素材库、素材的轨道、合成视频的工程文件等等。

相关术语:

AVAsset:素材库里的素材; 
AVAssetTrack:素材的轨道; 
AVMutableComposition:一个用来合成视频的工程文件; 
AVMutableCompositionTrack:工程文件中的轨道,有音频轨、视频轨等,里面可以插入各种对应的素材; 
AVMutableVideoCompositionLayerInstruction:视频轨道中的一个视频,可以缩放、旋转等; 
AVMutableVideoCompositionInstruction:一个视频轨道,包含了这个轨道上的所有视频素材; 
AVMutableVideoComposition:管理所有视频轨道,可以决定最终视频的尺寸,裁剪需要在这里进行; 
AVAssetExportSession:配置渲染参数并渲染。复制代码




问题记录:

下面带横线的是错误思路,一开始走进了误区,这种方式并不能解决点击view触发toucherBegan的问题

长按拍摄,利用的是touchesbegan开始录制,touchesEnded结束录制;存在一个小问题就是点击这个view的时候,会触发touchesBegan,然而不会触发touchesEnded。这里使用了一个延时机制去触发touchesBegan事件的触发,代码如下

- (void)touchesBegan:(NSSet<UITouch *> *)touches withEvent:(UIEvent *)event {
    NSLog(@"开始触摸");
    if ([[touches anyObject] view] == self.progressView) {
        if (!self.capture) {    // 表示是否正在录制
            double delayInSeconds = 0.5;    // 长按0.5s触发录制事件
            dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
            dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
                [self startCapture];    // 开始录制
            });
        }else {
            [self startCapture];
        }
        
    }
}复制代码

解决方案就是用长按手势来触发

UILongPressGestureRecognizer *longPress = [[UILongPressGestureRecognizer alloc]initWithTarget:self action:@selector(LongPressProgressView:)];
        longPress.minimumPressDuration = 0.5;
        [progressView addGestureRecognizer:longPress];复制代码
- (void)LongPressProgressView:(UILongPressGestureRecognizer *)gestureRecognizer {
    if (gestureRecognizer.state == UIGestureRecognizerStateBegan) {
        [self startCapture];
    }else if (gestureRecognizer.state == UIGestureRecognizerStateEnded) {
        if ([self.mediaCaptureDelegate respondsToSelector:@selector(stopCapture)]) {
            self.capture = NO;
            self.bgView.hidden = YES;
            [self.mediaCaptureDelegate stopCapture];
            [self.progressView clearProgress];
        }
    }
    
}复制代码

其次有个可以拖动、缩放、旋转的文本框,点击可以重新编辑,我是封装了一个类FWTextView,具体代码可以从下面的demo里找到,如图



最关键的一点就是水印的添加,主要功能就是这个么。先说下思路:

首先我们要知道我们能看到的视频实际上是由一个叫做videoLayer负责显示的,和他同级的有个layer叫做animationLayer,我们能够控制的其实就是这个东西,他可以由我们自己创建,他们有一个共同的父类叫做parentLayer。

添加图片水印的代码:

CALayer *imgLayer = [CALayer layer];
    imgLayer.contents = (id)img.CGImage;
    imgLayer.frame = CGRectMake(0, 0, size.width, size.height);复制代码

创建好了图片layer后,就需要创建videoLayer

//把文字和图标都添加到layer
    CALayer *overlayLayer = [CALayer layer];
    [overlayLayer addSublayer:imgLayer];
    overlayLayer.frame = CGRectMake(0, 0, size.width, size.height);
    [overlayLayer setMasksToBounds:YES];
    [overlayLayer addSublayer:imgLayer];
    CALayer *parentLayer = [CALayer layer];
    CALayer *videoLayer = [CALayer layer];
    parentLayer.frame = CGRectMake(0, 0, size.width, size.height);
    parentLayer.backgroundColor = [UIColor redColor].CGColor;
    videoLayer.frame = CGRectMake(0, 0, size.width, size.height);
    [parentLayer addSublayer:videoLayer];
    [parentLayer addSublayer:overlayLayer];
    
    composition.animationTool = [AVVideoCompositionCoreAnimationTool
                                 videoCompositionCoreAnimationToolWithPostProcessingAsVideoLayer:videoLayer inLayer:parentLayer];复制代码

这里主要是告诉系统我们的layer层时parentLayer,在parentLayer里负责video显示的是我们的videoLayer。

最后就是录制完的视频保存本地,这里使用的不是系统的,ios10开始ALAssetsLibrary被标志为弃用(DEPRECATED),并建议使用Photos framework的PHPhotoLibrary,使用需先引用#import <Photos/Photos.h>

方法1:同步存到系统相册(iOS10系统执行该方法,无法保存成功)

__block NSString *createdAssetID =nil;//唯一标识,可以用于图片资源获取
    NSError *error =nil;
    [[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
        createdAssetID = [PHAssetChangeRequest creationRequestForAssetFromImage:image].placeholderForCreatedAsset.localIdentifier;
    } error:&error];复制代码

补充:

上面的方法1在系统是10的真机上出现保存视频失败,更新一下新的方法:

NSString *path = [SNSImageEngine getRealHouseVideoPathWithVideoPath:videoPath];
            if (UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(path)) {
                UISaveVideoAtPathToSavedPhotosAlbum(path , self, @selector(video:didFinishSavingWithError:contextInfo:), nil);
            }else {
                [MBProgressHUD showMessage:@"保存失败"];
            }


// 视频保存回调
- (void)video:(NSString *)videoPath didFinishSavingWithError:(NSError *)error contextInfo:(void *)contextInfo {
    if (error == nil) {
        [MBProgressHUD showMessage:@"已保存至手机"];
    }else {
        [MBProgressHUD showMessage:@"保存失败"];
    }
}
复制代码

上面代码里的path是视频在本地的路径,还有一个回调方法提供。

方法2:存到某个自定义相册

[[PHPhotoLibrary sharedPhotoLibrary]performChanges:^{
        PHAssetChangeRequest *changeAssetRequest = [PHAssetChangeRequest creationRequestForAssetFromImage:image];
        PHAssetCollection *targetCollection = [[PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeAlbum subtype:PHAssetCollectionSubtypeSmartAlbumUserLibrary options:nil]lastObject];
        PHAssetCollectionChangeRequest *changeCollectionRequest = [PHAssetCollectionChangeRequest changeRequestForAssetCollection:targetCollection];
        PHObjectPlaceholder *assetPlaceholder = [changeAssetRequest placeholderForCreatedAsset];
        [changeCollectionRequest addAssets:@[assetPlaceholder]];
    } completionHandler:^(BOOL success,NSError * _Nullable error) {
        NSLog(@"finished adding");
    }];复制代码

到此基本结束了,后期有更新的话,会继续更新。


需求更改,要求录制时间加长,大于10s,会出现新的bug。视频超过10s后,声音会丢失。

  • 问题所在:AVCaptureMovieFileOutput他有默认的时间限制,默认值是 10 秒
  • 更改办法:设置 AVCaptureMovieFileOutput 的 movieFragmentInterval 属性为 kCMTimeInvalid,视频录制就不会受到限制
-(AVCaptureMovieFileOutput *)movieFileOutput{
    
    if (!_movieFileOutput) {
        _movieFileOutput = [[AVCaptureMovieFileOutput alloc] init];
        /*  默认的录制视频时间是10秒,如果视频大于10秒必须禁用他,否则录制的视频将会没有声音*/
        _movieFileOutput.movieFragmentInterval = kCMTimeInvalid;
    }
    return _movieFileOutput;
}

复制代码