3

アプリでオーディオを再生するために AudioQueue Service を使用しています。複数のオーディオ ファイルを一緒に再生する必要があります。私が今やっていることは、ファイルを再生するのに必要なだけの Audio Queue を作成することです。つまり、2 つのオーディオ ファイルに対して 2 つのオーディオ キューを作成し、それらを同時に開始して、オーディオ ミキシング効果を持たせます。

したがって、基本的には、これが「エレガントな」方法であることを知りたいと思います。

私は Audio Unit サービスとその MixerHost の例を認識していることに注意してください。そのオプションを提案しないでください。Audio Queue サービスのみを使用してサウンド ミキシングを行う必要があります。

4

1 に答える 1

1
- (void) setUpAndAddAudioAtPath:(NSURL*)assetURL toComposition:(AVMutableComposition *)composition {

    AVURLAsset *songAsset = [AVURLAsset URLAssetWithURL:assetURL options:nil];

    AVMutableCompositionTrack *track = [composition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];
    AVAssetTrack *sourceAudioTrack = [[songAsset tracksWithMediaType:AVMediaTypeAudio] objectAtIndex:0];

    NSError *error = nil;
    BOOL ok = NO;

    CMTime startTime = CMTimeMakeWithSeconds(0, 1);
    CMTime trackDuration = songAsset.duration;
    CMTime longestTime = CMTimeMake(848896, 44100); //(19.24 seconds)
    CMTimeRange tRange = CMTimeRangeMake(startTime, trackDuration);

    //Set Volume
    AVMutableAudioMixInputParameters *trackMix = [AVMutableAudioMixInputParameters audioMixInputParametersWithTrack:track];
    [trackMix setVolume:0.8f atTime:startTime];
    [audioMixParams addObject:trackMix];

    //Insert audio into track
    ok = [track insertTimeRange:tRange ofTrack:sourceAudioTrack atTime:CMTimeMake(0, 44100) error:&error];
}


- (BOOL) exportAudio {

    if (defaultSoundPath == nil || recordingSoundPath == nil) {

        [actvityIdicatiorView stopAnimating];
        [actvityIdicatiorView setHidden:YES];


        UIAlertView *alertView=[[UIAlertView alloc]initWithTitle:@"Select Sound" message:@"Both Sound is selected" delegate:self cancelButtonTitle:@"Ok" otherButtonTitles:nil];
        [alertView show];
        return NO;



    }

    AVMutableComposition *composition = [AVMutableComposition composition];
    if (audioMixParams) {
        [audioMixParams release];
        audioMixParams=nil;
    }
    audioMixParams = [[NSMutableArray alloc] initWithObjects:nil];

    //Add Audio Tracks to Composition

    NSString *sourceA= [[NSBundle mainBundle] pathForResource:@"Beach Soundscape" ofType:@"mp3"];

    //NSString *URLPath1 = pathToYourAudioFile1;

    NSURL *assetURL1 = [NSURL fileURLWithPath:sourceA];
    [self setUpAndAddAudioAtPath:assetURL1 toComposition:composition];

    NSString *sourceB = [[NSBundle mainBundle] pathForResource:@"DrumsMonoSTP" ofType:@"aif"];
    // NSString *URLPath2 = pathToYourAudioFile2;
    NSURL *assetURL2 = [NSURL fileURLWithPath:sourceB];
    [self setUpAndAddAudioAtPath:assetURL2 toComposition:composition];

    AVMutableAudioMix *audioMix = [AVMutableAudioMix audioMix];
    audioMix.inputParameters = [NSArray arrayWithArray:audioMixParams];

    //If you need to query what formats you can export to, here's a way to find out
    NSLog (@"compatible presets for songAsset: %@",
           [AVAssetExportSession exportPresetsCompatibleWithAsset:composition]);

    AVAssetExportSession *exporter = [[AVAssetExportSession alloc]
                                      initWithAsset: composition
                                      presetName: AVAssetExportPresetAppleM4A];


    exporter.audioMix = audioMix;
    exporter.outputFileType = @"com.apple.m4a-audio";

    //    NSArray *paths = NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES);
    //
    //    NSString *fileName = @"someFilename";

    //NSString *exportFile = [[paths objectAtIndex:0] stringByAppendingFormat: @"/%@.m4a", fileName];

    mixingSoundPath= [[self mixingSoundFolder] stringByAppendingFormat: @"/Mixing%@.m4a", [self dateString]];
    [mixingSoundPath retain];

    // set up export
    //myDeleteFile(exportFile);
    NSURL *exportURL = [NSURL fileURLWithPath:mixingSoundPath];
    exporter.outputURL = exportURL;

    static BOOL isComplete;

    // do the export
    [exporter exportAsynchronouslyWithCompletionHandler:^{

        int exportStatus = exporter.status;

        NSLog(@"exporter.......%i",exportStatus);


        switch (exportStatus) {

            case AVAssetExportSessionStatusFailed:
                //    NSError *exportError =exporter.error;
                isComplete=NO;
                NSLog (@"AVAssetExportSessionStatusFailed");


                NSLog (@"Error == %@", exporter.error);
                break;

            case AVAssetExportSessionStatusCompleted:
                [self mixingDidFinshing];

                isComplete=YES;
                break;
            case AVAssetExportSessionStatusUnknown:
                NSLog (@"AVAssetExportSessionStatusUnknown");
                isComplete=NO;

                break;
            case AVAssetExportSessionStatusExporting:
                isComplete=NO;

                NSLog (@"AVAssetExportSessionStatusExporting");

                break;
            case AVAssetExportSessionStatusCancelled:
                isComplete=NO;

                NSLog (@"AVAssetExportSessionStatusCancelled");

                break;
            case AVAssetExportSessionStatusWaiting:
                isComplete=NO;

                NSLog (@"AVAssetExportSessionStatusWaiting"); 

                break;
            default:  
                NSLog (@"didn't get export status"); 
                isComplete=NO;
                break;
        }


    }];

    return isComplete;

}
于 2012-08-18T11:12:15.167 に答える