使用AVAssetExportPresetHighestQuality时无法合成视频

时间:2018-08-30 09:24:18

标签: ios objective-c

我想创建一个将许多视频拼接在一起的应用程序。问题似乎是当我将指令与AVAssetExportPresetHighestQuality结合使用时,我收到一条错误消息,指出

  

导出失败->原因:视频无法合成。,用户信息:   {       NSLocalizedDescription =“操作已停止”;       NSLocalizedFailureReason =“无法合成视频。”;       NSUnderlyingError =“错误域= NSOSStatusErrorDomain代码= -17390 \”(空)\“”; }

如果我将其更改为AVAssetExportPresetPassthrough,则可以正常运行,但忽略了说明。有谁知道使用以下代码可能是什么问题。我快到了,但是这个问题使我受阻。

AVMutableComposition *mixComposition = [[AVMutableComposition alloc] init];

    AVMutableCompositionTrack *track = [mixComposition addMutableTrackWithMediaType:AVMediaTypeVideo
                                                                   preferredTrackID:kCMPersistentTrackID_Invalid];

    AVMutableCompositionTrack *audioTrack = [mixComposition addMutableTrackWithMediaType:AVMediaTypeAudio
                                                                        preferredTrackID:kCMPersistentTrackID_Invalid];

    CMTime insertTime = kCMTimeZero;

    NSMutableArray *arrayInstructions = [[NSMutableArray alloc] init];

    int i = 0;

    for (NSMutableDictionary * dict in self.arraySelectedAssets) {

            AVAsset *asset = [dict objectForKey:@"avasset"];

            //[self orientationForTrack:asset];

            AVAssetTrack* videoAssetTrack = [[asset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];
            AVAssetTrack* audioAssetTrack = [[asset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0];

            [track insertTimeRange:CMTimeRangeMake(kCMTimeZero, asset.duration) ofTrack:videoAssetTrack atTime:insertTime error:nil];

            [audioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, asset.duration) ofTrack:audioAssetTrack atTime:insertTime error:nil];

            AVMutableVideoCompositionInstruction *firstVideoCompositionInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];
            // Set the time range of the first instruction to span the duration of the first video track.
            firstVideoCompositionInstruction.timeRange = CMTimeRangeMake(insertTime, videoAssetTrack.timeRange.duration);


            AVMutableVideoCompositionLayerInstruction* firstVideoLayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:[[asset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0]];
            CGAffineTransform translateToCenter = CGAffineTransformMakeTranslation( 0,-1334);
            CGAffineTransform rotateBy90Degrees = CGAffineTransformMakeRotation( M_PI_2);
            CGAffineTransform shrinkWidth = CGAffineTransformMakeScale(0.1, 0.1); // needed because Apple does a "stretch" by default - really, we should find and undo apple's stretch - I suspect it'll be a CALayer defaultTransform, or UIView property causing this
            CGAffineTransform finalTransform = CGAffineTransformConcat( shrinkWidth, CGAffineTransformConcat(translateToCenter, rotateBy90Degrees) );
            [firstVideoLayerInstruction setTransform:finalTransform atTime:kCMTimeZero];

            firstVideoCompositionInstruction.layerInstructions = @[firstVideoLayerInstruction];

            [arrayInstructions addObject:firstVideoCompositionInstruction];

        insertTime = CMTimeAdd(insertTime, videoAssetTrack.timeRange.duration);

        i = i + 1;

    }

    AVMutableVideoComposition *mutableVideoComposition = [AVMutableVideoComposition videoComposition];
    mutableVideoComposition.instructions = arrayInstructions;
    mutableVideoComposition.renderSize = CGSizeMake(1334, 750);
    mutableVideoComposition.frameDuration = CMTimeMake(1, 30);


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

    // 5 - Create exporter

    self.timerExporter = [NSTimer scheduledTimerWithTimeInterval:0.01f
                                             target:self
                                           selector:@selector(exporterProgress)
                                           userInfo:nil
                                            repeats:YES];

    // 5 - Create exporter
    self.exporter = [[AVAssetExportSession alloc] initWithAsset:mixComposition
                                                                      presetName:AVAssetExportPresetHighestQuality];
    self.exporter .outputURL=self.combinedVideoURL;
    self.exporter .outputFileType = AVFileTypeQuickTimeMovie;
    self.exporter .shouldOptimizeForNetworkUse = YES;
    self.exporter.videoComposition = mutableVideoComposition;
    [self.exporter  exportAsynchronouslyWithCompletionHandler:^{

        [self.timerExporter invalidate];

        switch (self.exporter.status) {
            case AVAssetExportSessionStatusFailed:
                NSLog(@"Export failed -> Reason: %@, User Info: %@",
                      self.exporter.error.localizedFailureReason,
                      self.exporter.error.userInfo.description);
                [self showError:self.exporter.error.localizedFailureReason];
                break;
            case AVAssetExportSessionStatusCancelled:
                NSLog(@"Export cancelled");
                break;

            case AVAssetExportSessionStatusCompleted:
                NSLog(@"Export finished");

                dispatch_async(dispatch_get_main_queue(), ^{
                    self.labelProgressText.text = [NSString stringWithFormat:@"%@ (100%%)", NSLocalizedString(@"Combining The Videos", nil)];
                    [self applyTheFilter];
                });

                break;

        }

    }];

1 个答案:

答案 0 :(得分:0)

恐怕这不是您要的答案。我在转换和导出单个视频时遇到了同样的问题-AVAssetExportPresetHighestQuality适用于某些资产,而不适用于其他资产。

我当时的猜测是无法使用的资产的大小/帧率/质量不够高,无法使用AVAssetExportPresetHighestQuality进行渲染。

和您一样,我最终使用了AVAssetExportPresetPassthrough。对于您来说,最终结果可能是您拼接在一起的所有资产都将以其原始格式呈现。