iOS: в экспортированном видео отсутствует звук

Я пытаюсь экспортировать записанное видео. И преуспеть в этом. Но звук отсутствует в окончательном экспортированном видео. Поэтому я искал его и добавил ниже код для аудио.

if ([[videoAsset tracksWithMediaType:AVMediaTypeAudio] count] > 0)
{
    [videoTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, videoAsset.duration)

                        ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0]

                         atTime:kCMTimeZero error:nil];
}

Но я не могу сохранить видео после добавления вышеуказанного кода. Я получаю сообщение об ошибке:

"session.status 4 error Domain = AVFoundationErrorDomain Code = -11841 "Операция остановлена" UserInfo = 0x17027e140 {NSLocalizedDescription = Операция остановлена, NSLocalizedFailureReason = Не удалось создать видео.}"

- (void)exportDidFinish:(AVAssetExportSession*)session {

NSLog(@"session.status %ld error %@",session.status,session.error);}

Вот код, который я использовал для экспорта видео. Итак, у вас есть какие-либо идеи, как я могу достичь своей цели по экспорту видео со звуком? Спасибо!!

- (void)getVideoOutput{    
exportInProgress=YES;
NSLog(@"videoOutputFileUrl %@",videoOutputFileUrl);
AVAsset *videoAsset = [AVAsset assetWithURL:videoOutputFileUrl];
NSLog(@"videoAsset %@",videoAsset);
// 1 - Early exit if there's no video file selected

NSLog(@"video asset %@",videoAsset);

if (!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, videoAsset.duration)

                    ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0]

                     atTime:kCMTimeZero error:nil];

/* getting an error AVAssetExportSessionStatusFailed
if ([[videoAsset tracksWithMediaType:AVMediaTypeAudio] count] > 0)
{
    [videoTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, videoAsset.duration)

                        ofTrack:[[videoAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0]

                         atTime:kCMTimeZero error:nil];
}*/


// 3.1 - Create AVMutableVideoCompositionInstruction

AVMutableVideoCompositionInstruction *mainInstruction = [AVMutableVideoCompositionInstruction videoCompositionInstruction];

mainInstruction.timeRange = CMTimeRangeMake(kCMTimeZero, videoAsset.duration);



// 3.2 - Create an AVMutableVideoCompositionLayerInstruction for the video track and fix the orientation.

AVMutableVideoCompositionLayerInstruction *videolayerInstruction = [AVMutableVideoCompositionLayerInstruction videoCompositionLayerInstructionWithAssetTrack:videoTrack];

AVAssetTrack *videoAssetTrack = [[videoAsset tracksWithMediaType:AVMediaTypeVideo] objectAtIndex:0];

UIImageOrientation videoAssetOrientation_  = UIImageOrientationUp;

BOOL isVideoAssetPortrait_  = NO;

CGAffineTransform videoTransform = videoAssetTrack.preferredTransform;

if (videoTransform.a == 0 && videoTransform.b == 1.0 && videoTransform.c == -1.0 && videoTransform.d == 0) {

    videoAssetOrientation_ = UIImageOrientationRight;

    isVideoAssetPortrait_ = YES;

}

if (videoTransform.a == 0 && videoTransform.b == -1.0 && videoTransform.c == 1.0 && videoTransform.d == 0) {

    videoAssetOrientation_ =  UIImageOrientationLeft;

    isVideoAssetPortrait_ = YES;

}

if (videoTransform.a == 1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == 1.0) {

    videoAssetOrientation_ =  UIImageOrientationUp;

}

if (videoTransform.a == -1.0 && videoTransform.b == 0 && videoTransform.c == 0 && videoTransform.d == -1.0) {

    videoAssetOrientation_ = UIImageOrientationDown;

}

[videolayerInstruction setTransform:videoAssetTrack.preferredTransform atTime:kCMTimeZero];

[videolayerInstruction setOpacity:0.0 atTime: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;

}



float renderWidth, renderHeight;

renderWidth = naturalSize.width;

renderHeight = naturalSize.height;

mainCompositionInst.renderSize = CGSizeMake(renderWidth, renderHeight);

mainCompositionInst.instructions = [NSArray arrayWithObject:mainInstruction];

mainCompositionInst.frameDuration = CMTimeMake(1, 30);


int totalSeconds= (int) CMTimeGetSeconds(videoAsset.duration);

[self applyVideoEffectsToComposition:mainCompositionInst size:naturalSize videoDuration:totalSeconds];



// 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(), ^{

    dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{


        [self exportDidFinish:exporter];


    });

}];

}


person iRoid Solutions    schedule 24.11.2014    source источник


Ответы (1)


Я не уверен, поможет ли это, но вот как я сделал это в проекте:

  1. Подготовьте финальную композицию.

    AVMutableComposition *composition = [[AVMutableComposition alloc] init];
    
  2. Подготовьте видеодорожку

    AVMutableCompositionTrack *videoTrack = [composition addMutableTrackWithMediaType:AVMediaTypeVideo preferredTrackID:kCMPersistentTrackID_Invalid];
    
  3. Подготовьте звуковую дорожку

    AVMutableCompositionTrack *audioTrack = [composition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];
    
  4. Вставьте видеоданные из актива в видеодорожку

    AVAssetTrack *video = [[asset tracksWithMediaType:AVMediaTypeVideo] firstObject];
    [videoTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, asset.duration) ofTrack:video atTime:kCMTimeZero error:&error];
    
  5. Вставьте аудиоданные из актива в аудиодорожку

    AVAssetTrack *audio = [[asset tracksWithMediaType:AVMediaTypeAudio] firstObject];
    [audioTrack insertTimeRange:CMTimeRangeMake(kCMTimeZero, asset.duration) ofTrack:audio atTime:kCMTimeZero error:&error];
    
  6. Затем вы можете добавить некоторые инструкции для обработки видео и/или аудио данных.

  7. Наконец, вы должны иметь возможность экспортировать, используя:

    AVAssetExportSession *exporter = [[AVAssetExportSession alloc] initWithAsset:composition presetName:AVAssetExportPresetMediumQuality];
    [exporter exportAsynchronouslyWithCompletionHandler:^{ /* code when the export is complete */ }];
    

Кроме того, проверьте, правильно ли вы записали звук.
При первом срабатывании камеры iOS должна спросить, хотите ли вы разрешить использование микрофона. Проверьте в настройках вашего устройства, разрешено ли это.

Другой вариант: вы можете получить свой необработанный ресурс, используя окно «Окно»> «Устройство» в Xcode.
Выберите свое устройство и экспортируйте данные на свой компьютер. Затем найдите записанный актив и откройте его, например, с помощью VLC. Проверьте потоки с помощью Cmd + I, чтобы увидеть, есть ли аудио- и видеодорожка.

person Neovov    schedule 24.11.2014
comment
Спасибо за вашу помощь. Я понял. Я добавлял звук в видеодорожку. Он работал нормально после создания для него отдельной звуковой дорожки. - person iRoid Solutions; 24.11.2014