前言

用代码在简单视频编辑中,主要就是加美颜、水印(贴图)、视频截取、视频拼接、音视频的处理,在美颜中,使用GPUImage即可实现多种滤镜、磨皮美颜的功能,并且可以脸部识别实时美颜等功能,这个有很多成熟的处理方案,所以现在主要说后面的水印(贴图)、视频截取、视频拼接、音视频的处理,在文章结尾会给出一个完整的测试demo,该demo可以操作视频之后保存到系统相册,文章主要说明下注意的点。

上篇讲了视频编辑功能详解上篇-添加水印,本篇就说下视频裁剪、视频拼接、音视频的处理。

原理

正如上篇提到的,因为GPUImage只是对视频进行滤镜处理,并没有涉及到视频轨和音轨的处理,所以在视频的处理裁剪等编辑上面主要还是使用的AVFoundation对视频轨和音轨进行处理。

本篇略长,如果只是使用,可以直接去文章最后下载demo源码,复制使用即可。不过建议看懂源码,这样在视频的移动编辑上面就可以自己随便改了。

一、视频裁剪

完整源码:

//使用gpuimage重新录制一次
-(void)saveVedioPath:(NSURL*)vedioPath WithFileName:(NSString*)fileName andCallBack:(JLXCommonToolVedioCompletionHandler)competion
{self.completionHandler = competion;// 滤镜filter = [[GPUImageAlphaBlendFilter alloc] init];//    //mix即为叠加后的透明度,这里就直接写1.0了[(GPUImageDissolveBlendFilter *)filter setMix:1.0f];// 播放NSURL *sampleURL  = vedioPath;AVAsset *asset = [AVAsset assetWithURL:sampleURL];movieFile = [[GPUImageMovie alloc] initWithAsset:asset];movieFile.runBenchmark = YES;movieFile.playAtActualSpeed = NO;AVAssetTrack *videoAssetTrack = [[asset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];UIImageOrientation videoAssetOrientation_  = UIImageOrientationUp;//拍摄的时候视频是否是竖屏拍的BOOL isVideoAssetvertical  = NO;CGAffineTransform videoTransform = videoAssetTrack.preferredTransform;if (videoTransform.a == 0 && videoTransform.b == 1.0 && videoTransform.c == -1.0 && videoTransform.d == 0) {isVideoAssetvertical = YES;videoAssetOrientation_ =  UIImageOrientationUp;//正着拍}if (videoTransform.a == 0 && videoTransform.b == -1.0 && videoTransform.c == 1.0 && videoTransform.d == 0) {//        videoAssetOrientation_ =  UIImageOrientationLeft;isVideoAssetvertical = YES;videoAssetOrientation_ = UIImageOrientationDown;//倒着拍}if (videoTransform.a == 1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == 1.0) {isVideoAssetvertical = NO;videoAssetOrientation_ =  UIImageOrientationLeft;//左边拍的}if (videoTransform.a == -1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == -1.0) {isVideoAssetvertical = NO;videoAssetOrientation_ = UIImageOrientationRight;//右边拍}GPUImageView *filterView = [[GPUImageView alloc] initWithFrame:CGRectMake(0, 0, asset.naturalSize.width, asset.naturalSize.height)];[filterView setTransform:CGAffineTransformMakeRotation(M_PI_2)];UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0, 0, asset.naturalSize.width, asset.naturalSize.height)];[view setBackgroundColor:[UIColor clearColor]];GPUImageUIElement *uielement = [[GPUImageUIElement alloc] initWithView:view];NSString *pathToMovie = [NSHomeDirectory() stringByAppendingPathComponent:[NSString stringWithFormat:@"Documents/%@.mp4",fileName]];unlink([pathToMovie UTF8String]);NSURL *movieURL = [NSURL fileURLWithPath:pathToMovie];movieWriter = [[GPUImageMovieWriter alloc] initWithMovieURL:movieURL size:CGSizeMake(asset.naturalSize.width, asset.naturalSize.height)];GPUImageFilter* progressFilter = [[GPUImageFilter alloc] init];[movieFile addTarget:progressFilter];[progressFilter addTarget:filter];[uielement addTarget:filter];movieWriter.shouldPassthroughAudio = YES;if ([[asset tracksWithMediaType:AVMediaTypeAudio] count] > 0){movieFile.audioEncodingTarget = movieWriter;} else {//no audiomovieFile.audioEncodingTarget = nil;}[movieFile enableSynchronizedEncodingUsingMovieWriter:movieWriter];// 显示到界面[filter addTarget:filterView];[filter addTarget:movieWriter];[movieWriter startRecording];[movieFile startProcessing];__weak typeof(self) weakSelf = self;[progressFilter setFrameProcessingCompletionBlock:^(GPUImageOutput *output, CMTime time) {[uielement update];}];[movieWriter setCompletionBlock:^{__strong typeof(self) strongSelf = weakSelf;[strongSelf->filter removeTarget:strongSelf->movieWriter];[strongSelf->movieWriter finishRecording];if (strongSelf.completionHandler) {strongSelf.completionHandler(movieURL,nil,isVideoAssetvertical);}}];
}/**裁剪视频@param videoPath 视频的路径@param startTime 截取视频开始时间@param endTime 截取视频结束时间,如果为0则为整个视频@param videoSize 视频截取的大小,如果为0则不裁剪视频大小@param videoDealPoint Point(x,y):传zero则为裁剪从0,0开始@param fileName 文件名字@param shouldScale 是否拉伸,false的话不拉伸,裁剪黑背景*/
- (void)saveVideoPath:(NSURL*)videoPath withStartTime:(float)startTime withEndTime:(float)endTime withSize:(CGSize)videoSize withVideoDealPoint:(CGPoint)videoDealPoint WithFileName:(NSString*)fileName shouldScale:(BOOL)shouldScale
{if (!videoPath) {[SVProgressHUD dismiss];return;}[SVProgressHUD showWithStatus:@"裁剪视频到系统相册"];//1 创建AVAsset实例 AVAsset包含了video的所有信息 self.videoUrl输入视频的路径//封面图片NSDictionary *opts = [NSDictionary dictionaryWithObject:@(YES) forKey:AVURLAssetPreferPreciseDurationAndTimingKey];videoAsset = [AVURLAsset URLAssetWithURL:videoPath options:opts];     //初始化视频媒体文件bool isWXVideo = false;for (int i=0; i<videoAsset.metadata.count; i++) {AVMetadataItem * item = [videoAsset.metadata objectAtIndex:i];NSLog(@"======metadata:%@,%@,%@,%@",item.identifier,item.extraAttributes,item.value,item.dataType);NSDictionary *dic = [self StrToArrayOrNSDictionary:[NSString stringWithFormat:@"%@",item.value]];if ([[dic.allKeys objectAtIndex:0] isEqualToString: @"WXVer"]) {isWXVideo = true;[self saveVedioPath:videoPath WithFileName:@"wxVideo" andCallBack:^(NSURL *assetURL, NSError *error,BOOL isVideoAssetvertical) {dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{NSString *newVideoPath = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents/wxVideo.mp4"];[self goSaveVideoPath:[NSURL fileURLWithPath:newVideoPath] withStartTime:startTime withEndTime:endTime withSize:videoSize withVideoDealPoint:videoDealPoint WithFileName:fileName shouldScale:shouldScale isWxVideoAssetvertical:isVideoAssetvertical];});}];break;}}if ([[videoAsset tracksWithMediaType:AVMediaTypeAudio] count] == 0){[self saveVedioPath:videoPath WithFileName:@"wxVideo" andCallBack:^(NSURL *assetURL, NSError *error,BOOL isVideoAssetvertical) {dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{NSString *newVideoPath = [NSHomeDirectory() stringByAppendingPathComponent:@"Documents/wxVideo.mp4"];[self goSaveVideoPath:[NSURL fileURLWithPath:newVideoPath] withStartTime:startTime withEndTime:endTime withSize:videoSize withVideoDealPoint:videoDealPoint WithFileName:fileName shouldScale:shouldScale isWxVideoAssetvertical:isVideoAssetvertical];});}];return;}if (!isWXVideo) {[self goSaveVideoPath:videoPath withStartTime:startTime withEndTime:endTime withSize:videoSize withVideoDealPoint:videoDealPoint WithFileName:fileName shouldScale:shouldScale isWxVideoAssetvertical:NO];}
}//Assetvertical gpuimage会把微信的竖屏渲染成横屏,横屏还是横屏
- (void)goSaveVideoPath:(NSURL*)videoPath withStartTime:(float)startTime withEndTime:(float)endTime withSize:(CGSize)videoSize withVideoDealPoint:(CGPoint)videoDealPoint WithFileName:(NSString*)fileName shouldScale:(BOOL)shouldScale isWxVideoAssetvertical:(BOOL)Assetvertical{if (!videoPath) {[SVProgressHUD dismiss];return;}//1 创建AVAsset实例 AVAsset包含了video的所有信息 self.videoUrl输入视频的路径//封面图片NSDictionary *opts = [NSDictionary dictionaryWithObject:@(YES) forKey:AVURLAssetPreferPreciseDurationAndTimingKey];//    NSDictionary *opts = [NSDictionary dictionaryWithObjectsAndKeys:@(YES),AVURLAssetPreferPreciseDurationAndTimingKey,AVAssetReferenceRestrictionForbidNone,AVURLAssetReferenceRestrictionsKey, nil];videoAsset = [AVURLAsset URLAssetWithURL:videoPath options:opts];     //初始化视频媒体文件//开始时间CMTime startCropTime = CMTimeMakeWithSeconds(startTime, 600);//结束时间CMTime endCropTime = CMTimeMakeWithSeconds(endTime, 600);if (endTime == 0) {endCropTime = CMTimeMakeWithSeconds(videoAsset.duration.value/videoAsset.duration.timescale-startTime, videoAsset.duration.timescale);}//2 创建AVMutableComposition实例. apple developer 里边的解释 【AVMutableComposition is a mutable subclass of AVComposition you use when you want to create a new composition from existing assets. You can add and remove tracks, and you can add, remove, and scale time ranges.】AVMutableComposition *mixComposition = [AVMutableComposition composition];//有声音if ([[videoAsset tracksWithMediaType:AVMediaTypeAudio] count] > 0){//声音采集AVURLAsset * audioAsset = [[AVURLAsset alloc] initWithURL:videoPath options:opts];//音频通道AVMutableCompositionTrack * audioTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];//音频采集通道AVAssetTrack * audioAssetTrack = [[audioAsset tracksWithMediaType:AVMediaTypeAudio] lastObject];[audioTrack insertTimeRange:CMTimeRangeMake(startCropTime, endCropTime) ofTrack:audioAssetTrack atTime:kCMTimeZero error:nil];}//3 视频通道  工程文件中的轨道,有音频轨、视频轨等,里面可以插入各种对应的素材AVMutableCompositionTrack *videoTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideopreferredTrackID:kCMPersistentTrackID_Invalid];NSError *error;//把视频轨道数据加入到可变轨道中 这部分可以做视频裁剪TimeRange[videoTrack insertTimeRange:CMTimeRangeMake(startCropTime, endCropTime)ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeVideo] lastObject]atTime:kCMTimeZero error:&error];//3.1 AVMutableVideoCompositionInstruction 视频轨道中的一个视频,可以缩放、旋转等AVMutableVideoCompositionInstruction *mainInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];mainInstruction.timeRange = CMTimeRangeMake(kCMTimeZero, videoTrack.timeRange.duration);// 3.2 AVMutableVideoCompositionLayerInstruction 一个视频轨道,包含了这个轨道上的所有视频素材AVMutableVideoCompositionLayerInstruction *videolayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:videoTrack];AVAssetTrack *videoAssetTrack = [[videoAsset tracksWithMediaType:AVMediaTypeVideo] lastObject];UIImageOrientation videoAssetOrientation_  = UIImageOrientationUp;//拍摄的时候视频是否是竖屏拍的BOOL isVideoAssetvertical  = NO;CGAffineTransform videoTransform = videoAssetTrack.preferredTransform;if (videoTransform.a == 0 && videoTransform.b == 1.0 && videoTransform.c == -1.0 && videoTransform.d == 0) {isVideoAssetvertical = YES;videoAssetOrientation_ =  UIImageOrientationUp;//正着拍}if (videoTransform.a == 0 && videoTransform.b == -1.0 && videoTransform.c == 1.0 && videoTransform.d == 0) {//        videoAssetOrientation_ =  UIImageOrientationLeft;isVideoAssetvertical = YES;videoAssetOrientation_ = UIImageOrientationDown;//倒着拍}if (videoTransform.a == 1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == 1.0) {isVideoAssetvertical = NO;videoAssetOrientation_ =  UIImageOrientationLeft;//左边拍的}if (videoTransform.a == -1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == -1.0) {isVideoAssetvertical = NO;videoAssetOrientation_ = UIImageOrientationRight;//右边拍}float scaleX = 1.0,scaleY = 1.0,scale = 1.0;CGSize originVideoSize;if (isVideoAssetvertical || Assetvertical) {originVideoSize = CGSizeMake([[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].height, [[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].width);}else{originVideoSize = CGSizeMake([[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].width, [[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].height);}float x = videoDealPoint.x;float y = videoDealPoint.y;if (shouldScale) {scaleX = videoSize.width/originVideoSize.width;scaleY = videoSize.height/originVideoSize.height;scale  = MAX(scaleX, scaleY);if (scaleX>scaleY) {NSLog(@"竖屏");}else{NSLog(@"横屏");}}else{scaleX = 1.0;scaleY = 1.0;scale = 1.0;}if (Assetvertical) {CGAffineTransform trans = CGAffineTransformMake(videoAssetTrack.preferredTransform.a*scale, videoAssetTrack.preferredTransform.b*scale, videoAssetTrack.preferredTransform.c*scale, videoAssetTrack.preferredTransform.d*scale, videoAssetTrack.preferredTransform.tx*scale-x+720, videoAssetTrack.preferredTransform.ty*scale-y);//    [videolayerInstruction setTransform:trans atTime:kCMTimeZero];CGAffineTransform trans2 = CGAffineTransformRotate(trans, M_PI_2);[videolayerInstruction setTransform:trans2 atTime:kCMTimeZero];}else{CGAffineTransform trans = CGAffineTransformMake(videoAssetTrack.preferredTransform.a*scale, videoAssetTrack.preferredTransform.b*scale, videoAssetTrack.preferredTransform.c*scale, videoAssetTrack.preferredTransform.d*scale, videoAssetTrack.preferredTransform.tx*scale-x, videoAssetTrack.preferredTransform.ty*scale-y);[videolayerInstruction setTransform:trans atTime:kCMTimeZero];}//裁剪区域//    [videolayerInstruction setCropRectangle:CGRectMake(0, 0, 720, 720) atTime:kCMTimeZero];// 3.3 - Add instructionsmainInstruction.layerInstructions = [NSArray arrayWithObjects:videolayerInstruction,nil];//AVMutableVideoComposition:管理所有视频轨道,可以决定最终视频的尺寸,裁剪需要在这里进行AVMutableVideoComposition *mainCompositionInst = [AVMutableVideoComposition videoComposition];CGSize naturalSize;//    if(isVideoAssetvertical){//        naturalSize = CGSizeMake(videoAssetTrack.naturalSize.height, videoAssetTrack.naturalSize.width);//    } else {//        naturalSize = videoAssetTrack.naturalSize;//    }naturalSize = originVideoSize;int64_t renderWidth = 0, renderHeight = 0;if (videoSize.height ==0.0 || videoSize.width == 0.0) {renderWidth = naturalSize.width;renderHeight = naturalSize.height;}else{renderWidth = ceil(videoSize.width);renderHeight = ceil(videoSize.height);}mainCompositionInst.renderSize = CGSizeMake(renderWidth, renderHeight);mainCompositionInst.instructions = [NSArray arrayWithObject:mainInstruction];mainCompositionInst.frameDuration = CMTimeMake(1, 30);// 4 - 输出路径NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);NSString *documentsDirectory = [paths objectAtIndex:0];NSString *myPathDocs =  [documentsDirectory stringByAppendingPathComponent:[NSString stringWithFormat:@"%@.mp4",fileName]];unlink([myPathDocs UTF8String]);NSURL* videoUrl = [NSURL fileURLWithPath:myPathDocs];//    dlink = [CADisplayLink displayLinkWithTarget:self selector:@selector(updateProgress)];//    [dlink setFrameInterval:15];//    [dlink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSDefaultRunLoopMode];//    [dlink setPaused:NO];// 5 - 视频文件输出exporter = [[AVAssetExportSession alloc] initWithAsset:mixCompositionpresetName:AVAssetExportPresetHighestQuality];exporter.outputURL=videoUrl;exporter.outputFileType = AVFileTypeMPEG4;exporter.shouldOptimizeForNetworkUse = YES;exporter.videoComposition = mainCompositionInst;[exporter exportAsynchronouslyWithCompletionHandler:^{dispatch_async(dispatch_get_main_queue(), ^{//这里是输出视频之后的操作,做你想做的[self cropExportDidFinish:exporter];});}];
}- (void)cropExportDidFinish:(AVAssetExportSession*)session {if (session.status == AVAssetExportSessionStatusCompleted) {NSURL *outputURL = session.outputURL;dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{[SVProgressHUD dismiss];__block PHObjectPlaceholder *placeholder;if (UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(outputURL.path)) {NSError *error;[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{PHAssetChangeRequest* createAssetRequest = [PHAssetChangeRequest creationRequestForAssetFromVideoAtFileURL:outputURL];placeholder = [createAssetRequest placeholderForCreatedAsset];} error:&error];if (error) {[SVProgressHUD showErrorWithStatus:[NSString stringWithFormat:@"%@",error]];}else{[SVProgressHUD showSuccessWithStatus:@"视频已经保存到相册"];}}else {[SVProgressHUD showErrorWithStatus:NSLocalizedString(@"视频保存相册失败,请设置软件读取相册权限", nil)];}});}else{NSLog(@"%@",session.error);[SVProgressHUD showErrorWithStatus:NSLocalizedString(@"裁剪失败", nil)];}
}// 将JSONDATA转化为字典或者数组
- (id)DataToArrayOrNSDictionary:(NSData *)jsonData{NSError *error = nil;id jsonObject = [NSJSONSerialization JSONObjectWithData:jsonData options:NSJSONReadingAllowFragments error:&error];if (jsonObject != nil && error == nil){return jsonObject;}else{// 解析错误return nil;}
}// 将JSON串转化为字典或者数组
- (id)StrToArrayOrNSDictionary:(NSString *)jsonStr {NSData *jsonData = [jsonStr dataUsingEncoding:NSUTF8StringEncoding];return [self DataToArrayOrNSDictionary:jsonData];
}

调用的时候,直接使用

-(void)cropImage{NSURL *videoPath = [NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"selfS" ofType:@"MOV"]];[self saveVideoPath:videoPath withStartTime:0.1 withEndTime:0 withSize:CGSizeMake(300, 300) withVideoDealPoint:CGPointMake(50, 50) WithFileName:@"cropVideo" shouldScale:YES];
}

其实对于一般的视频裁剪,只需要使用下面这个函数即可

- (void)goSaveVideoPath:(NSURL*)videoPath withStartTime:(float)startTime withEndTime:(float)endTime withSize:(CGSize)videoSize withVideoDealPoint:(CGPoint)videoDealPoint WithFileName:(NSString*)fileName shouldScale:(BOOL)shouldScale isWxVideoAssetvertical:(BOOL)Assetvertical

1.1、微信的处理

但是在实际使用过程中发现,单独使用这个函数中的AVFoundation处理裁剪视频的时候,对微信的支持并不好,如果使用微信自带的相机拍摄那个十秒视频虽然可以裁剪成功,但是是蓝屏的,只有声音没有画面,在打印的metadata信息中对比发现

WX拍摄2017-05-11 19:35:58.529751+0800 JianLiXiu[9592:2774766] ======metadata:uiso/dscp,{dataType = 2;dataTypeNamespace = "com.apple.quicktime.udta";
},{"WXVer":369428256},com.apple.metadata.datatype.UTF-8
2017-05-11 19:35:58.531034+0800 JianLiXiu[9592:2774766] ======commonMetadata:uiso/dscp,{dataType = 2;dataTypeNamespace = "com.apple.quicktime.udta";
},{"WXVer":369428256},com.apple.metadata.datatype.UTF-8WX下载======metadata:uiso/dscp,{dataType = 2;dataTypeNamespace = "com.apple.quicktime.udta";
},{"WXVer":369428256},com.apple.metadata.datatype.UTF-8======commonMetadata:uiso/dscp,{dataType = 2;dataTypeNamespace = "com.apple.quicktime.udta";
},{"WXVer":369428256},com.apple.metadata.datatype.UTF-8自带拍摄2017-05-11 19:35:09.708010+0800 JianLiXiu[9592:2774253] ======metadata:uiso/loci,{dataType = 2;dataTypeNamespace = "com.apple.quicktime.udta";
},+31.1711+121.3836+045.636/,com.apple.metadata.datatype.UTF-8
2017-05-11 19:35:09.710245+0800 JianLiXiu[9592:2774253] ======metadata:uiso/date,{dataType = 0;dataTypeNamespace = "com.apple.quicktime.udta";
},2017-05-11T17:40:41+0800,com.apple.metadata.datatype.raw-data
2017-05-11 19:35:09.712193+0800 JianLiXiu[9592:2774253] ======commonMetadata:uiso/date,{dataType = 0;dataTypeNamespace = "com.apple.quicktime.udta";
},2017-05-11T17:40:41+0800,com.apple.metadata.datatype.raw-data
2017-05-11 19:35:09.712746+0800 JianLiXiu[9592:2774253] ======commonMetadata:uiso/loci,{dataType = 2;dataTypeNamespace = "com.apple.quicktime.udta";
},+31.1711+121.3836+045.636/,com.apple.metadata.datatype.UTF-8

所以通过视频的metadata信息来判断是不是微信的视频,通过metadata是否含有{"WXVer":369428256}这个来判断是不是微信处理过的视频,如果是微信的视频先使用GPUImage重新渲染编码一次然后再处理。

代码中的

CGAffineTransform videoTransform = videoAssetTrack.preferredTransform;

是代表的视频的拍摄方向,在正常拍摄的时候打印如下

自带竖 0,1,-1,0
自带横 1,0,0,1qq竖 1,0,0,1
qq横 0,-1,1,0微信竖 0,1,-1,0
微信横 1,0,0,1

但微信的视频在GPUImage重新录制之后,拍摄方向变成了

微信竖 1,0,0,1
微信横 1,0,0,1

也就是说不管横竖,全部变成了一样的,这样就导致横屏拍摄的时候裁剪正常,竖屏拍摄的时候,裁剪之后出来的视频时错误的,所以要先判断原视频是什么方向,之后在裁剪处理的时候去处理。

1.2、裁剪位置的选择

int64_t renderWidth = 0, renderHeight = 0;

是要渲染输出的视频大小是多少,所以如果想设置指定大小,那就在这个设置,输出的大小就可以了,但是这个裁剪是默认从(0,0)开始的,如果是从中间开始的就需要设置videolayerInstruction这个的移动了。

videolayerInstruction是这个视频轨的动画等移动效果,视频的移动,翻转,缩小等都在这个上面进行操作。通过设置CGAffineTransform来达到裁剪位置

x会跟着c的值进行拉伸(View的宽度是跟着改变),y会跟着b的值进行拉伸(View的高度跟着改变),要注意到的是c和b的值改变不会影响到View的point(center中心点)的改变。这是个很有意思的两个参数。

x会跟着t.x进行x做表平移,y会跟着t.y进行平移。这里的point(center)是跟着变换的。

下面是Apple整合的transform

平移 :

①根据本身的transform进行平移 CGAffineTransformMakeTranslation(CGFloat tx,CGFloat ty)

②根据本身的transform后者另外的transform进行平移CGAffineTransformTranslate(CGAffineTransform t,CGFloat tx,CGFloat ty)

缩放 :

①根据本身的transform进行缩放

CGAffineTransformMakeScale(CGFloat sx,CGFloat sy)

②根据本身的transform后者另外的transform进行缩放

CGAffineTransformScale(CGAffineTransform t,CGFloat sx,CGFloat sy)

旋转 :

① 根据本身的transform进行旋转

CGAffineTransformMakeRotation(CGFloat angle) (angle 旋转的角度)

②根据本身的transform后者另外的transform进行旋转

CGAffineTransformRotate(CGAffineTransform t,CGFloat angle)

具体可以参考《iOS 仿射变换CGAffineTransform详解》

所以结合渲染的大小和要移动的效果,就可以知道是从哪个点开始裁剪,裁剪多大的视频了。

1.3、裁剪时间的选择

裁剪时间主要是在视频轨和音轨编辑的时候,设置插入的时长,从而控制裁剪时间。

//把视频轨道数据加入到可变轨道中 这部分可以做视频裁剪TimeRange[videoTrack insertTimeRange:CMTimeRangeMake(startCropTime, endCropTime)ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeVideo] lastObject]atTime:kCMTimeZero error:&error];

1.4、对没有声音视频的处理,比如延时视频

上篇说了如果没有声音的视频在处理的时候会错误,所以在裁剪的时候也要注意是否含有声音,如果没有声音就不要添加音轨了

 //有声音if ([[videoAsset tracksWithMediaType:AVMediaTypeAudio] count] > 0){//声音采集AVURLAsset * audioAsset = [[AVURLAsset alloc] initWithURL:videoPath options:opts];//音频通道AVMutableCompositionTrack * audioTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];//音频采集通道AVAssetTrack * audioAssetTrack = [[audioAsset tracksWithMediaType:AVMediaTypeAudio] lastObject];[audioTrack insertTimeRange:CMTimeRangeMake(startCropTime, endCropTime) ofTrack:audioAssetTrack atTime:kCMTimeZero error:nil];}

1.5、原视频大小的处理

如果不裁剪视频的大小,只是默认大小裁剪时间的时候,发现获取的naturalSize在竖屏的时候是错误的,比如一个视频720*1280,结果获得的naturalSize是1280*720,横竖刚好相反,所以需要对竖屏的大小进行处理一下,这样就是正确的了

 if (isVideoAssetvertical || Assetvertical) {originVideoSize = CGSizeMake([[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].height, [[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].width);}else{originVideoSize = CGSizeMake([[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].width, [[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] naturalSize].height);}

二、视频拼接、音视频的处理

视频的拼接和音视频的拼接是一样的,都是处理视频轨和音轨的开始时间,如果第二个视频的开始时间是第一个视频的结束时间,那么就是两段视频的拼接,如果开始时间相同,那么就是两个视频在混合了。

下面给出一个两个视频拼接,再加上一个背景音乐的例子。

-(void)addFirstVideo:(NSURL*)firstVideoPath andSecondVideo:(NSURL*)secondVideo withMusic:(NSURL*)musicPath{[SVProgressHUD showWithStatus:@"正在合成到系统相册中"];AVAsset *firstAsset = [AVAsset assetWithURL:firstVideoPath];AVAsset *secondAsset = [AVAsset assetWithURL:secondVideo];AVAsset *musciAsset = [AVAsset assetWithURL:musicPath];// 1 - Create AVMutableComposition object. This object will hold your AVMutableCompositionTrack instances.AVMutableComposition *mixComposition = [[AVMutableComposition alloc] init];// 2 - Video trackAVMutableCompositionTrack *firstTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideopreferredTrackID:kCMPersistentTrackID_Invalid];[firstTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, firstAsset.duration)ofTrack:[[firstAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] atTime:kCMTimeZero error:nil];[firstTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, secondAsset.duration)ofTrack:[[secondAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0] atTime:firstAsset.duration error:nil];if (musciAsset!=nil){AVMutableCompositionTrack *AudioTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeAudiopreferredTrackID:kCMPersistentTrackID_Invalid];[AudioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, CMTimeAdd(firstAsset.duration, secondAsset.duration))ofTrack:[[musciAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0] atTime:kCMTimeZero error:nil];}// 4 - Get pathNSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);NSString *documentsDirectory = [paths objectAtIndex:0];NSString *myPathDocs =  [documentsDirectory stringByAppendingPathComponent:[NSString stringWithFormat:@"mergeVideo-%d.mov",arc4random() % 1000]];NSURL *url = [NSURL fileURLWithPath:myPathDocs];// 5 - Create exporterexporter = [[AVAssetExportSession alloc] initWithAsset:mixCompositionpresetName:AVAssetExportPresetHighestQuality];exporter.outputURL=url;exporter.outputFileType = AVFileTypeQuickTimeMovie;exporter.shouldOptimizeForNetworkUse = YES;[exporter exportAsynchronouslyWithCompletionHandler:^{dispatch_async(dispatch_get_main_queue(), ^{[self exportDidFinish:exporter];});}];
}- (void)exportDidFinish:(AVAssetExportSession*)session {if (session.status == AVAssetExportSessionStatusCompleted) {NSURL *outputURL = session.outputURL;dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.2 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{[SVProgressHUD dismiss];__block PHObjectPlaceholder *placeholder;if (UIVideoAtPathIsCompatibleWithSavedPhotosAlbum(outputURL.path)) {NSError *error;[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{PHAssetChangeRequest* createAssetRequest = [PHAssetChangeRequest creationRequestForAssetFromVideoAtFileURL:outputURL];placeholder = [createAssetRequest placeholderForCreatedAsset];} error:&error];if (error) {[SVProgressHUD showErrorWithStatus:[NSString stringWithFormat:@"%@",error]];}else{[SVProgressHUD showSuccessWithStatus:@"视频已经保存到相册"];}}else {[SVProgressHUD showErrorWithStatus:NSLocalizedString(@"视频保存相册失败,请设置软件读取相册权限", nil)];}});}
}

调用的时候

-(void)addMusic{NSURL *videoPath1 = [NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"selfS" ofType:@"MOV"]];NSURL *videoPath2 = [NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"selfH" ofType:@"MOV"]];NSURL *music = [NSURL fileURLWithPath:[[NSBundle mainBundle] pathForResource:@"music" ofType:@"mp3"]];[self addFirstVideo:videoPath1 andSecondVideo:videoPath2 withMusic:music];
}

这样就是videopath1和videopath2拼接,然后加上一个背景音乐music,之后生成一个视频保存到相册。

(2017-08-14)2.1、音频音量的调节

最近需要把背景音乐的音量调小,当然背景音乐文件音量调小即可,但是毕竟每个文件处理都不太方便,所以还是用代码把指定音轨的音量减小,这里只减小背景音乐的音量,不减小其他的音量,所以用到了AVMutableAudioMix这个。

在原工程增加以下代码即可

//修改背景音乐的音量startAVMutableAudioMix *videoAudioMixTools = [AVMutableAudioMix audioMix];if (musciAsset) {//调节音量//获取音频轨道AVMutableAudioMixInputParameters *firstAudioParam = [AVMutableAudioMixInputParameters audioMixInputParametersWithTrack:AudioTrack];//设置音轨音量,可以设置渐变,设置为1.0就是全音量[firstAudioParam setVolumeRampFromStartVolume:1.0 toEndVolume:1.0 timeRange:CMTimeRangeMake(kCMTimeZero, CMTimeAdd(firstAsset.duration, secondAsset.duration))];[firstAudioParam setTrackID:AudioTrack.trackID];videoAudioMixTools.inputParameters = [NSArray arrayWithObject:firstAudioParam];}//end
exporter.audioMix = videoAudioMixTools;

三、demo下载

github下载:https://github.com/DamonHu/VideoEditDemo

gitosc下载:http://git.oschina.net/DamonHoo/VideoEditDemo

四、demo演示

五、总结

本篇已完结,在视频的处理上面主要就是对视频轨、音频轨的活用,当然以后还有字幕轨等,但是原理都是一样的,参看里面的源码即可看到所有轨道的编辑。GPUImage是一个很好的补充,在渲染方面很好用,可以结合一下采用最适合项目的方案。

在项目中碰到了很多问题,在demo中尽量避免了这类问题,如果你也碰到了,可以看看谷歌,我下面的参考文章也是碰到问题时看到的有价值的文章,希望对你有用。

六、参考文章

讲解篇

BUG篇

转载自http://www.hudongdong.com/ios/550.htm

IOS视频编辑,视频裁剪,视频拼接,音频处理,视频处理相关推荐

  1. 【SeeMusic】视频编辑 ( 顶部裁剪 | 底部裁剪 | 左侧裁剪 | 右侧裁剪 | 明亮度 | 对比度 | 色调 | 饱和度 )

    SeeMusic 系列文章目录 [SeeMusic]下载安装并注册 SeeMusic 软件 [SeeMusic]创建 SeeMusic 工程并编辑相关内容 ( 创建工程 | 导入 MIDI 文件 | ...

  2. Android使用MediaCodec和OpenGL对多段视频画面进行裁剪和拼接

    太久没写博客了,由于工作,过年还有孩子出生搞得自己焦头烂额,现在有些时间了就搞点东西.发现浏览量突破10万了,也是挺高兴的,虽然很多东西写的不好,可也看到了自己的进步,也是前年到现在的累积.刚开始我只 ...

  3. android开源视频编辑手机,手机也能剪大片 - 视频编辑专题 - Android 应用 - 专题 - 【最美应用】...

    Adobe Premiere Clip Adobe Premiere Clip 是阿逗比家在移动端上一款非常给力的视频剪辑应用,这款Premiere Clip 和你在 PC 上用过 Premiere ...

  4. android视频编辑好,安卓比较好用的视频剪辑软件 手机视频剪辑软件推荐

    安卓比较好用的视频剪辑软件,随着智能手机的越来越普及,视频也是变成大家平常娱乐的消遣方式,很多的用户是喜欢看短视频,自己也想展示一下自己,对于剪辑软件也不是太熟悉,对此也是想要知道安卓比较好用的视频剪 ...

  5. 美摄iOS端短视频SDK视频编辑的流程及方法

    美摄短视频SDK提供视频编辑功能,支持视频图片素材混合导入.滤镜.配音.时间特效.画中画等丰富的编辑效果.本文介绍iOS端短视频SDK视频编辑的流程及方法. 短视频SDK主要包含"视频录制& ...

  6. 用计算机对视频进行剪裁和编辑,视频编辑王怎么剪辑视频 分割裁剪视频的方法介绍...

    视频编辑王作为一款专业的视频编辑处理工具,软件内置的功能也是深受广大视频处理人士的喜爱,小编了解到很多用户不知道怎么剪辑视频,那么小编我今天就来为大家讲讲吧! 1. 首先需要先进入到视频编辑王的界面之 ...

  7. QT利用opengl 进行视频裁剪、拼接,4宫格,9宫格

    一.概述 1.1 前言 在上一篇文章我们讲了Y420P视频数据如何裁剪.拼接.旋转等,但是缺点也很明显,一是工作量大,代码量较大.二是容错率低,因为涉及到大量的浮点型计算,导致在数据拷贝的时候存在误差 ...

  8. iOS 如何对音频、视频合成,配音、卡拉OK技术

    iOS 如何对音频.视频合成,配音.卡拉OK技术 iOS原生的开发框架集成了丰富的视频/音频处理功能,所以用不着去找第三方框架.只要: #import <AVFoundation/AVFound ...

  9. 六款好用的视频编辑软件推荐

    文章来源:https://www.reneelab.com.cn/free-video-editing-software-no-watermark.html 目录 1.都叫兽™视频编辑软件 2.iMo ...

  10. android音视频工程师,音视频学习 (十三) Android 中通过 FFmpeg 命令对音视频编辑处理(已开源)...

    ## 音视频学习 (十三) Android 中通过 FFmpeg 命令对音视频编辑处理(已开源) ## 视音频编辑器 ## 前言 有时候我们想对音视频进行加工处理,比如视频编辑.添加字幕.裁剪等功能处 ...

最新文章

  1. 用IE重起计算机或者关机
  2. Bzoj3060 [Poi2012]Tour de Byteotia
  3. ZooKeeper的配置文件优化性能(转)
  4. 【剑指offer】不使用新变量,交换两个变量的值,C++实现
  5. 盘点VS2015 预览版的5个新特性
  6. 平安、招行、工行、微众、网商们怎么都在布局卫星物联网?
  7. 【机器学习基础】算法工程师必备的机器学习--EM
  8. 数据结构_二叉树遍历
  9. dcloud会员激活mui
  10. 无法启动windows audio服务,错误提示126.
  11. VS2008(C#)制作网页Tab标签切换方法(四)
  12. Linux下的Scala安装
  13. python 输入整数_python输入整数
  14. 阿里云如何设置content-type,微软的在线预览无法使用问题
  15. android hidl简单实例1
  16. TSC打印机使用教程终极版(转)
  17. workerman wss 配置备忘录
  18. 3D建模软件测试自学,收藏:5个自学3DMAX教程以及3D模型资源的网站
  19. 如何度过人生艰难:魔都28岁硬核知识型美少女自救指南
  20. 腾讯云 mysql 数据库名_腾讯云数据库MySQL如何选择配置

热门文章

  1. JVM运行和类加载全过程
  2. 三、Solr管理控制台(二)
  3. 最好用的发短信(验证码、语音短信)接口
  4. 程序员常用资源工具集合
  5. Teambition使用教程
  6. Hdu2184汉诺塔VIII
  7. 声纹识别(一)——简介
  8. QStackedWidget切换动画实现以及尝试过程中花点、花屏、背景无法透明解决处理
  9. 关于enq: TX - allocate ITL entry的问题分析
  10. I Want To Spend My Lifetime Loving You