ios - 使用 AVMutableVideoComposition 导出时出现视频方向问题

标签 ios iphone video avfoundation

下面是我用来导出视频的函数:

- (void) videoOutput
{
//1 - Early exit if there's no video file selected
if (!self.videoAsset) {
    UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"Error" message:@"Please Load a Video Asset First"
                                                   delegate:nil cancelButtonTitle:@"OK" otherButtonTitles:nil];
    [alert show];
    return;
}

// 2 - Create AVMutableComposition object. This object will hold your AVMutableCompositionTrack instances.
AVMutableComposition *mixComposition = [[AVMutableComposition alloc] init];

// 3 - Video track
AVMutableCompositionTrack *videoTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideo
                                                                    preferredTrackID:kCMPersistentTrackID_Invalid];
[videoTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, self.videoAsset.duration)
                    ofTrack:[[self.videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0]
                     atTime:kCMTimeZero error:nil];

// 3.1 - Create AVMutableVideoCompositionInstruction
AVMutableVideoCompositionInstruction *mainInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];
mainInstruction.timeRange = CMTimeRangeMake(kCMTimeZero, self.videoAsset.duration);

// 3.2 - Create an AVMutableVideoCompositionLayerInstruction for the video track and fix the orientation.
AVMutableVideoCompositionLayerInstruction *videolayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:videoTrack];
AVAssetTrack *videoAssetTrack = [[self.videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];

BOOL isVideoAssetPortrait_  = NO;
CGAffineTransform videoTransform = videoAssetTrack.preferredTransform;
if (videoTransform.a == 0 && videoTransform.b == 1.0 && videoTransform.c == -1.0 && videoTransform.d == 0) {
    isVideoAssetPortrait_ = YES;
}
if (videoTransform.a == 0 && videoTransform.b == -1.0 && videoTransform.c == 1.0 && videoTransform.d == 0) {

    isVideoAssetPortrait_ = YES;
}
if (videoTransform.a == 1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == 1.0) {
    isVideoAssetPortrait_  = NO;
}
if (videoTransform.a == -1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == -1.0) {
    isVideoAssetPortrait_  = NO;
}
[videolayerInstruction setTransform:videoAssetTrack.preferredTransform atTime:kCMTimeZero];
[videolayerInstruction setOpacity:0.0 atTime:self.videoAsset.duration];

// 3.3 - Add instructions
mainInstruction.layerInstructions = [NSArray arrayWithObjects:videolayerInstruction,nil];

AVMutableVideoComposition *mainCompositionInst = [AVMutableVideoComposition videoComposition];

CGSize naturalSize;
if(isVideoAssetPortrait_){
    naturalSize = CGSizeMake(videoAssetTrack.naturalSize.height, videoAssetTrack.naturalSize.width);
} else {
    naturalSize = videoAssetTrack.naturalSize;
}

mainCompositionInst.renderSize = naturalSize;
mainCompositionInst.instructions = [NSArray arrayWithObject:mainInstruction];
mainCompositionInst.frameDuration = CMTimeMake(1, 30);

// 4 - Get path
NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
NSString *documentsDirectory = [paths objectAtIndex:0];
NSString *myPathDocs =  [documentsDirectory stringByAppendingPathComponent:
                         [NSString stringWithFormat:@"FinalVideo-%d.mov",arc4random() % 1000]];
NSURL *url = [NSURL fileURLWithPath:myPathDocs];

// 5 - Create exporter
AVAssetExportSession *exporter = [[AVAssetExportSession alloc] initWithAsset:mixComposition
                                                                  presetName:AVAssetExportPresetHighestQuality];
exporter.outputURL=url;
exporter.outputFileType = AVFileTypeQuickTimeMovie;
exporter.shouldOptimizeForNetworkUse = YES;
exporter.videoComposition = mainCompositionInst;
[exporter exportAsynchronouslyWithCompletionHandler:^{
    dispatch_async(dispatch_get_main_queue(), ^{
        [self exportDidFinish:exporter];
    });
}];
}

问题是我第一次使用这个函数导出竖屏视频时,变量videoTransform(videoAssetTrack.preferredTransform)是:

videoTransform.a == 0 && videoTransform.b == 1.0 && videoTransform.c == -1.0 && videoTransform.d == 0

变量isVideoAssetPortrait_ 等于YES。一切都是对的。但是,在导出完成并保存到相机胶卷后,我使用此功能重新加载结果视频。这一次,videoTransform 发生了变化:

videoTransform.a == 1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == 1.0

isVideoAssetPortrait_ 等于NO。这意味着在一次导出后,videoTransform 改变了它的值(方向从纵向 -> 横向)

我google了很多关于AV Foundation的视频定位的问题,但是还没有找到解决方案。

感谢您阅读我的长篇解释。如果您有任何问题,请告诉我。

最佳答案

关于视频轨道,没有纵向/横向这样的概念。它只有它的尺寸和应用的转换以正确呈现它。默认情况下,“人像”视频按照相机生成的方式进行编码(比如说风景),并使用 90 度旋转来正确呈现它。

当你导出它时,它的方向没有改变,它只是重新编码物理旋转,所以不需要旋转来正确呈现它。 这就是你第二次获得单位矩阵的原因(这并不意味着它从纵向变为横向),但这次它的自然大小也被交换,并且根据你的代码一切都应该没问题。

请说明后一种情况有什么问题。

关于ios - 使用 AVMutableVideoComposition 导出时出现视频方向问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19660250/

相关文章:

node.js - 使用 NodeJS 在本地保存部分视频文件

wpf - WPF 中的绿屏

ios - 当我向 TestFlight 添加新的测试人员时,我是否必须为他们提供一个新的版本 "notified"?

ios - 应用程序右侧的 ionic 空白

iphone - iOS - 确定选择了哪个 MKPointAnnotation

iphone - iPhone 上的状态更改网络可达性通知是强制性的吗?

ios - 如何避免三种数据库操作中的厄运金字塔?

ios - Swift 在 TableView 中展开 Optional 值时意外发现 nil

javascript - 启动画面后 iPhone ChildBrowser 重定向

javascript - vimeo 视频结束时执行操作 WordPress