Если BroadcastSampleHandler используется для записи экрана, файл записи экрана не может быть воспроизведен после завершения вызова.

Когда я использовал Replaykit для разработки видео, я провел функциональный тест. Сначала я нажал кнопку «Начать трансляцию» и включил микрофон. Когда поступит голосовой вызов Wechat, повесьте трубку и продолжите запись. Через некоторое время запись останавливается, и обнаруживается, что записанный файл невозможно воспроизвести.

Это мой код:

      
#import "ZGBroadcastManager.h"
#import "SharePath.h"


static ZGBroadcastManager *_sharedManager = nil;

@interface ZGBroadcastManager ()

@property (nonatomic, weak) RPBroadcastSampleHandler *sampleHandler;

//使用以下保存mp4
@property (strong, nonatomic) AVAssetWriter *assetWriter;
@property (strong, nonatomic) AVAssetWriterInput *videoInput;
@property (strong, nonatomic) AVAssetWriterInput *audioAppInput;
@property (strong, nonatomic) AVAssetWriterInput *audioMicInput;
@end


@implementation ZGBroadcastManager

+ (instancetype)sharedManager {
    if (!_sharedManager) {
        @synchronized (self) {
            if (!_sharedManager) {
                _sharedManager = [[self alloc] init];
            }
        }
    }
    return _sharedManager;
}

- (void)setupAssetWriter {
    if ([self.assetWriter canAddInput:self.videoInput]) {
        [self.assetWriter addInput:self.videoInput];
    } else {
        NSAssert(false, @"添加视频写入失败");
    }
    if ([self.assetWriter canAddInput:self.audioAppInput]) {
        [self.assetWriter addInput:self.audioAppInput];
    } else {
        NSAssert(false, @"添加App音频写入失败");
    }
    if ([self.assetWriter canAddInput:self.audioMicInput]) {
        [self.assetWriter addInput:self.audioMicInput];
    } else {
        NSAssert(false, @"添加Mic音频写入失败");
    }
}


- (NSString *)timestamp {
    long long timeinterval = (long long)([NSDate timeIntervalSinceReferenceDate] * 1000);
    return [NSString stringWithFormat:@"%lld", timeinterval];
}

- (AVAssetWriter *)assetWriter {
    if (!_assetWriter) {
        NSError *error = nil;
        NSUserDefaults *sharedDefaults = [[NSUserDefaults alloc] initWithSuiteName:GroupIDKey];
        NSString *fileName = [NSString stringWithFormat:@"%@",[self timestamp]];
        [sharedDefaults setObject:fileName forKey:@"FileKey"];
        [sharedDefaults synchronize];
        NSURL *filePathURL = [SharePath filePathUrlWithFileName:fileName];
        NSLog(@"filePathURL------%@",filePathURL.path);//保存在共享文件夹中
            self.filePath = [NSString stringWithFormat:@"%@",filePathURL.path];
            NSLog(@"sampleHandler---self.filePath ----- %@",self.filePath);
            NSLog(@"sampleHandler------filePathURL.path:%@",filePathURL.path);
            _assetWriter = [[AVAssetWriter alloc] initWithURL:filePathURL fileType:(AVFileTypeMPEG4) error:&error];
            NSLog(@"_assetWriter---文件写入成功!");
            NSAssert(!error, @"_assetWriter初始化失败");
       
    }
    return _assetWriter;
}

- (AVAssetWriterInput *)videoInput {
    if (!_videoInput) {
        CGSize size = [UIScreen mainScreen].bounds.size;
        //写入视频大小
        NSInteger numPixels = size.width  * size.height;
        //每像素比特
        CGFloat bitsPerPixel = 10;
        NSInteger bitsPerSecond = numPixels * bitsPerPixel;
        // 码率和帧率设置
        NSDictionary *compressionProperties = @{
            AVVideoAverageBitRateKey : @(bitsPerSecond),//码率(平均每秒的比特率)
            AVVideoExpectedSourceFrameRateKey : @(15),//帧率(如果使用了AVVideoProfileLevelKey则该值应该被设置,否则可能会丢弃帧以满足比特流的要求)
            AVVideoMaxKeyFrameIntervalKey : @(15),//关键帧最大间隔
            AVVideoProfileLevelKey : AVVideoProfileLevelH264HighAutoLevel,
            AVVideoH264EntropyModeKey:AVVideoH264EntropyModeCABAC,
        };
        
        NSDictionary *videoOutputSettings = @{
            AVVideoCodecKey : AVVideoCodecTypeH264,
            AVVideoScalingModeKey : AVVideoScalingModeResizeAspect,
            AVVideoWidthKey : @(size.width * 2),
            AVVideoHeightKey : @(size.height * 2),
            AVVideoCompressionPropertiesKey : compressionProperties
        };
        _videoInput = [[AVAssetWriterInput alloc] initWithMediaType:AVMediaTypeVideo outputSettings:videoOutputSettings];
        _videoInput.expectsMediaDataInRealTime = YES;//实时录制
    }
    return _videoInput;
}

- (AVAssetWriterInput *)audioAppInput {
    if (!_audioAppInput) {
        NSDictionary *audioCompressionSettings = @{ AVEncoderBitRatePerChannelKey : @(28000),
                                                    AVFormatIDKey : @(kAudioFormatMPEG4AAC),
                                                    AVNumberOfChannelsKey : @(2),
                                                    AVSampleRateKey : @(22050) };

        _audioAppInput = [AVAssetWriterInput assetWriterInputWithMediaType:AVMediaTypeAudio outputSettings:audioCompressionSettings];
        _audioAppInput.expectsMediaDataInRealTime = YES;//实时录制
    }
    return _audioAppInput;
}
- (AVAssetWriterInput *)audioMicInput {
    if (!_audioMicInput) {
        NSDictionary *audioCompressionSettings = @{ AVEncoderBitRatePerChannelKey : @(28000),
                                                    AVFormatIDKey : @(kAudioFormatMPEG4AAC),
                                                    AVNumberOfChannelsKey : @(2),
                                                    AVSampleRateKey : @(22050) };

        _audioMicInput = [AVAssetWriterInput assetWriterInputWithMediaType:AVMediaTypeAudio outputSettings:audioCompressionSettings];
        _audioMicInput.expectsMediaDataInRealTime = YES;//实时录制
    }
    return _audioMicInput;
}
- (void)startBroadcast:(RPBroadcastSampleHandler *)sampleHandler {
    
    self.sampleHandler = sampleHandler;
    NSUserDefaults *share = [[NSUserDefaults alloc] initWithSuiteName:GroupIDKey];
    [self setupAssetWriter];//
    
    // Add an observer for stop broadcast notificatio
    NSLog(@"点击开始直播了");
    [share setInteger:1 forKey:@"isStart"];
    CFNotificationCenterAddObserver(CFNotificationCenterGetDarwinNotifyCenter(),
                                    (__bridge const void *)(self),
                                    onBroadcastFinish,
                                    (CFStringRef)@"ZGFinishBroadcastUploadExtensionProcessNotification",
                                    NULL,
                                    CFNotificationSuspensionBehaviorDeliverImmediately);

    CFNotificationCenterAddObserver(CFNotificationCenterGetDarwinNotifyCenter(), NULL, updateEnabled, NotificationOff, NULL, CFNotificationSuspensionBehaviorDeliverImmediately);
    // Do some business logic when starting screen capture here.

}

- (void)stopBroadcast {
    [self stopWriting];
    // Remove observer for stop broadcast notification
    CFNotificationCenterRemoveObserver(CFNotificationCenterGetDarwinNotifyCenter(),
                                       (__bridge const void *)(self),
                                       (CFStringRef)@"ZGFinishBroadcastUploadExtensionProcessNotification",
                                       NULL);
    
    CFNotificationCenterRemoveObserver(CFNotificationCenterGetDarwinNotifyCenter(),
                                       (__bridge const void *)(self),
                                       (CFStringRef)NotificationOff,
                                       NULL);
    
    
    NSUserDefaults *sharedDefaults = [[NSUserDefaults alloc] initWithSuiteName:GroupIDKey];
    NSString *fileName = [sharedDefaults objectForKey:@"FileKey"];
    if(fileName!=nil && fileName.length >0){
        CFNotificationCenterPostNotification(CFNotificationCenterGetDarwinNotifyCenter(), (CFStringRef)@"ScreenDidFinishNotif", NULL, nil, YES);
    }
    NSLog(@"停止录屏!");

    // Do some business logic when finishing screen capture here.
}

- (void)handleSampleBuffer:(CMSampleBufferRef)sampleBuffer withType:(RPSampleBufferType)sampleBufferType {
    
    AVAssetWriterStatus status = self.assetWriter.status;
    if ( status == AVAssetWriterStatusFailed || status == AVAssetWriterStatusCompleted || status == AVAssetWriterStatusCancelled) {
        NSAssert(false,@"屏幕录制AVAssetWriterStatusFailed error :%@", self.assetWriter.error);
        return;
    }
    if (status == AVAssetWriterStatusUnknown) {
        [self.assetWriter startWriting];
        CMTime time = CMSampleBufferGetPresentationTimeStamp(sampleBuffer);
        [self.assetWriter startSessionAtSourceTime:time];
    }
    
    switch (sampleBufferType) {
        case RPSampleBufferTypeVideo:
            // Handle video sample buffer
            @autoreleasepool {
                if (status == AVAssetWriterStatusWriting) {
                    if (self.videoInput.isReadyForMoreMediaData) {
                       BOOL success = [self.videoInput appendSampleBuffer:sampleBuffer];
                        if (!success) {
                            [self stopWriting];
                        }
                    }
                }
            }
            break;
        case RPSampleBufferTypeAudioApp:
            NSLog(@"RPSampleBufferTypeAudioApp------");
            @autoreleasepool {
                if (status == AVAssetWriterStatusWriting) {
                    if (self.audioAppInput.isReadyForMoreMediaData) {
                        BOOL success = [self.audioAppInput appendSampleBuffer:sampleBuffer];
                        NSLog(@"执行了audioAppInput");
                        if (!success) {
                            [self stopWriting];
                        }
                    }
                }
            }
            // Handle audio sample buffer for app audio
            break;
         case RPSampleBufferTypeAudioMic:
           @autoreleasepool {
                if (status == AVAssetWriterStatusWriting) {
                        if (self.audioMicInput.isReadyForMoreMediaData) {
                            BOOL success = [self.audioMicInput appendSampleBuffer:sampleBuffer];
                            NSLog(@"执行了audioMicInput");
                            if (!success) {
                                [self stopWriting];
                            }
                       }

               }
}

            // Handle audio sample buffer for mic audio
            break;
        default:
            break;
    }
}


#pragma mark - Finish broadcast function

// Handle stop broadcast notification from main app process
void onBroadcastFinish(CFNotificationCenterRef center, void *observer, CFStringRef name, const void *object, CFDictionaryRef userInfo) {

    // Stop broadcast
    [[ZGBroadcastManager sharedManager] stopBroadcast];

    RPBroadcastSampleHandler *handler = [ZGBroadcastManager sharedManager].sampleHandler;
    if (handler) {
        // Finish broadcast extension process with no error
        #pragma clang diagnostic push
        #pragma clang diagnostic ignored "-Wnonnull"
        [handler finishBroadcastWithError:nil];
        #pragma clang diagnostic pop
    } else {
        NSLog(@"⚠️ RPBroadcastSampleHandler is null, can not stop broadcast upload extension process");
    }
}
- (void)stopWriting{
    
    if(self.assetWriter.status == AVAssetWriterStatusWriting){
        [self.videoInput markAsFinished];
        [self.audioAppInput markAsFinished];
        [self.audioMicInput markAsFinished];
        [self.assetWriter finishWriting];
        self.videoInput = nil;
        self.audioAppInput = nil;
        self.audioMicInput = nil;
        self.assetWriter = nil;
    }
}

@end

Как мне справиться с тем, что я могу добиться, повесив трубку, что записанное видео завершено

0 ответов

Другие вопросы по тегам