iOS端使用replaykit录制屏幕的技术细节

栏目: IOS · 发布时间: 6年前

内容简介:这里我们设置代理,通过代理方法的回调我们才能启动录制进程。反馈已完成配置

iOS10:

···

iOS9已经实现了基本的app内容录制、预览、保存、分享,但是其输出的结果其实是一个已经将音频、视频编码并交织到一起成为一个mp4文件,开发者只能处理这个mp4文件,无法对原始音视频数据进行处理。对于有些app可能存在诸如分辨率减小、码率减小、音频编辑等各种需求,都需要对原始的yuv、pcm数据进行处理,或者对编码过程进行定制化干预。

考虑到开发者这个需求,苹果在iOS10的replaykit中开放了这部分api,通过extension形式将录制进程展现给开发者。其实iOS9时录制也是在一个独立于app的进程中进行,只是未开放。iOS10提供了分发相关多个类和api,用户可以通过代理方法获取到屏幕录制的原始数据,做进一步处理。引入时需要通过xcode的file -> new -> target 找到两个相关extension:

iOS端使用replaykit录制屏幕的技术细节

录制

ios10的replaykit的录制已经跟iOS9差异很大,ios10已经支持录制的原始音视频数据的 【实时】获取(iOS9只可以获取到录制停止后编码的mp4),开发者可以自己进行实时分发或者编码后处理。

主要步骤如下:

  1. 启动备选界面:

    iOS10中由于录制作为一个外部的extension,可以供所有系统中app使用,所以不能直接启动这个录制的进程。需要首先启动支持录制的列表sheet,通过下面接口

[RPBroadcastActivityViewController loadBroadcastActivityViewControllerWithHandler:^(RPBroadcastActivityViewController * _Nullable broadcastActivityViewController, NSError * _Nullable error) {

        self.broadcastAVC = broadcastActivityViewController;
        self.broadcastAVC.delegate = self;
        [self presentViewController:self.broadcastAVC animated:YES completion:nil];
    }];

这里我们设置代理,通过代理方法的回调我们才能启动录制进程。

iOS端使用replaykit录制屏幕的技术细节

反馈已完成配置

#import "BroadcastSetupViewController.h"
@implementation BroadcastSetupViewController

- (void)userDidFinishSetup {
    NSURL *broadcastURL = [NSURL URLWithString:@"http://apple.com/broadcast/streamID"];
    NSDictionary *setupInfo = @{ @"broadcastName" : @"example" };
    // Tell ReplayKit that the extension is finished setting up and can begin broadcasting
    [self.extensionContext completeRequestWithBroadcastURL:broadcastURL setupInfo:setupInfo];
}

- (void)userDidCancelSetup {
    [self.extensionContext cancelRequestWithError:[NSError errorWithDomain:@"YourAppDomain" code:-1 userInfo:nil]];
}

- (void)viewDidLoad
{
}
- (void)viewWillAppear:(BOOL)animated
{
    [self userDidFinishSetup];
}

这里的BroadcastSetupViewController就在xxxSetupUI的target中,是这个target建立时自动生成的模板vc,我们可以在这里添加自定义方法来建立一个vc,添加view,用于展示信息,或者用户鉴权,然后根据用户输入情况,决定是否让用户使用录制进程。

如果我们同意用户使用录制进程,这里我们主要需要告知调用的进程我们xxxSetupUI进程已经完成设置,可以开始广播了。其中viewDidLoad、viewWillAppear两个方法是我后填写的,这里主要是需要调用[self userDidFinishSetup]; 方法来完成通知调用方。

注意:

  • 必须调用[self userDidFinishSetup] ,调用进程里面的didFinishWithBroadcastController (下一步启动录制时用到)才能回调

  • 必须在viewWillAppear中才能调用,在viewDidLoad中无法生效(都是坑啊......)

  1. 启动录制:

    上一步,xxxSetupUI进程通过self.extensionContext 将其extension进程中的信息反馈回来,我们的RPBroadcastActivityViewController的代理方法将会回调:

- (void)broadcastActivityViewController:(RPBroadcastActivityViewController *)broadcastActivityViewController didFinishWithBroadcastController:(RPBroadcastController *)broadcastController error:(NSError *)error
{
    dispatch_async(dispatch_get_main_queue(), ^{
        [broadcastActivityViewController dismissViewControllerAnimated:YES completion:nil];
    });

    self.broadcastController = broadcastController;
    [broadcastController startBroadcastWithHandler:^(NSError * _Nullable error) {

    }];
}

回调中我们需要首先将sheet界面dismiss。 然后通过回调回来的broadcastController,调用接口启动录制,这里需要将broadcastController引用下来,用于我们在合适时机使用它结束录制。

  1. 接收原始音视频数据

    上一步启动录制成功后,我们就可以在录制进程中接收到相关回调了,录制进程在target创建时,模板生成了SampleHandler,其中已经复写了相关录制进行的方法:

@implementation SampleHandler
- (void)broadcastStartedWithSetupInfo:(NSDictionary<NSString *,NSObject *> *)setupInfo {
    // User has requested to start the broadcast. Setup info from the UI extension can be supplied but optional. 
}
- (void)broadcastPaused {
    // User has requested to pause the broadcast. Samples will stop being delivered.
}
- (void)broadcastResumed {
    // User has requested to resume the broadcast. Samples delivery will resume.
}
- (void)broadcastFinished {
    // User has requested to finish the broadcast.
}
- (void)processSampleBuffer:(CMSampleBufferRef)sampleBuffer withType:(RPSampleBufferType)sampleBufferType {

    switch (sampleBufferType) {
        case RPSampleBufferTypeVideo:
            // Handle video sample buffer
            break;
        case RPSampleBufferTypeAudioApp:
            // Handle audio sample buffer for app audio
            break;
        case RPSampleBufferTypeAudioMic:
            // Handle audio sample buffer for mic audio
            break;
        default:
            break;
    }
}

首先会回调到broadcastStartedWithSetupInfo方法,这里我们通常进行了一些初始化,例如进程间通知的监听等。下面的几个方法broadcastPaused、broadcastResumed、broadcastFinished表示了录制的进程变化,通常我们会在其中添加进程通知,通过源app这些变化。最后的processSampleBuffer方法就是最终采集到的音频、视频原始数据。其中音频未做混音,包括麦克音频pcm和app音频pcm,而视频输出为yuv数据。

注意:

  • iOS10只支持app内容录制,所以当app切到后台,录制内容将停止;

  • 手机锁屏时,录制进程将停止;

  • 这几个方法中的代码不能阻塞(例如写文件等慢操作),否则导致录制进程停止;

iOS11:

到了iOS11时代,苹果终于开放了对录制内容的升级,从iOS10的app内升级到整个系统级别的录制。但是对于隐私方面的考虑,苹果还是增加了很多用户使用门槛。iOS11中如果只是录制app内的内容,直接使用iOS10的方法即可,但是如果录制系统内容,则变化较多:

  1. 启动录制:

  • 对于录制app内容,iOS11增加了新接口,可以直接启动想要的录制进程,跳过中间列表sheet在点击选择的过程:

+ (void)loadBroadcastActivityViewControllerWithPreferredExtension:(NSString * _Nullable)preferredExtension handler:(nonnull void(^)(RPBroadcastActivityViewController * _Nullable broadcastActivityViewController, NSError * _Nullable error))handler API_AVAILABLE(ios(11.0)) API_UNAVAILABLE(tvos);

对于录制系统内容,iOS11不允许开发直接调用api来启动系统界别的录制,必须是用户通过手动启动。启动方法很复杂:

用户点击进入手机设置页面-> 控制中心-> 自定义 , 找到屏幕录制的功能按钮,将其添加到上方:添加成功后,我们可以在手机上滑唤出控制界面中发现这个启动按钮:

iOS端使用replaykit录制屏幕的技术细节 iOS端使用replaykit录制屏幕的技术细节

注意:

在上方弹出的列表中,需要选择我们创建target对应的app图标,才能使用我们的录制进程进行采集。

  1. 通知启动app:

    由于iOS11录制的启动为手动操作,并且开发者启动录制进程的app也无从知道是否已经启动,所以通常我们会在broadcastStartedWithSetupInfo中发出进程级通知,告知app,录制已经启动。

  2. 结束录制:

    从iOS11的接口设计上,我们推断结束估计也跟启动录制一样,不开放给开发者,所以起初我以为只能通过用户自己再次点击启动录制按钮,选择停止,才能主动停止录制,开发者无法干预这个过程。使用方法同启动录制类似,弹出列表中,直接点击下面的停止。

    但是很明显,这种设计对用户体验影响很大,如果我们的app已经停止了对采集的数据的显示或者分发,但是由于无法干预录制进程,那个进程将持续在工作,最直观体现在手机导航栏上方绿条(与手机通话时同样的机制),直到后来在RPBroadcastSampleHandler的方法里面发现了这个:

/*! @abstract Method that should be called when broadcasting can not proceed due to an error. Calling this method will stop the broadcast and deliver the error back to the broadcasting app through RPBroadcastController's delegate.
    @param error NSError object that will be passed back to the broadcasting app through RPBroadcastControllerDelegate's broadcastController:didFinishWithError: method.
 */
- (void)finishBroadcastWithError:(NSError *)error;

这个方法就藏在上面列出的broadcastStartedWithSetupInfo、broadcastPaused、broadcastResumed、broadcastFinished等方法后面,被我误以为是一个录制状态的回调。那么在启动录制进程的app中怎么使用这个  finishBroadcastWithError 方法来结束录制呢?

由于是手动启动录制进程,在启动录制进程的app中,我们没有相关回调能获取到这个方法的 RPBroadcastSampleHandler实例,所以无法直接启动。只能在录制进程中RPBroadcastSampleHandler实例自己调用,那么我们就可以通过进程通信的方法,前面已经介绍了启动录制时我们先注册进程通知,然后在收到进程通知时,我们调用 [self finishBroadcastWithError: nil]; 即可,这里的error入参,我们可以自定义一个字典,用于将错误信息展示进程结束时弹出的alert窗口中给用户。

iOS12:

iOS11的复杂操作启动屏幕录制,不知道阻塞了多少用户的继续使用。进入到2018年的iOS12,苹果终于想通了,replaykit也迎来了柳暗花明,开发者企盼的api控制启动录制终于来了!

启动录制:

iOS12还是会考虑用户的感知性,要求开发者必须通过replaykit提供的 RPSystemBroadcastPickerView 来展示启动的view,然后通过点击view上面的按钮才能启动:

#ifdef IPHONE_OS_VERSION_iOS12
        _broadPickerView = [[RPSystemBroadcastPickerView alloc] initWithFrame:CGRectMake(20, 5, 20, 20)];
        _broadPickerView.preferredExtension = @"com.cmcc.xiaoximeeting.ScreenRecordUpload";
        [self addSubview:_broadPickerView];
#endif

如上面代码,可以通过属性preferredExtension直接加载我们想要的录制进程。

优化:

虽然我们迎来更多自主控制权,但是悲催的是这里我们还是要等待弹出界面点击启动,才能开始录制。如果我们这个录制只是作为我们本身app的功能点,如何绕过这个点击操作呢? 可以考虑用一些trick方式:

  1. 首先我们将_broadPickerView的frame合理设置,使其隐藏在某个按钮(通常是自定义的启动录制)后面;

  2. 当我们点击到这个按钮时 ,响应链会将点击也传递给这个_broadPickerView,那么这时我们可以再把点击传递给_broadPickerView上面的开始按钮:

- (void)clickedOnStartRecordButton:(UIButton *)sender
{
#ifdef IPHONE_OS_VERSION_iOS12
    if (sender.tag == TAG_SHARESCREEN)
    {
        for (UIView *view in _broadPickerView.subviews)
        {
            if ([view isKindOfClass:[UIButton class]])
            {
                [(UIButton*)view sendActionsForControlEvents:UIControlEventTouchDown];
            }
        }
    }
    else
    {
#endif
   // 其他逻辑代码
#ifdef IPHONE_OS_VERSION_iOS12
}
#endif

注意:

sendActionsForControlEvents:UIControlEventTouchDown传递的参数必须是UIControlEventTouchDown,我之前传的是upinside事件,一直失败,直到尝试了UIControlEventAllTouchEvents,发现可以成功,才发觉事件不对,逐个尝试其他事件后,才定位到是UIControlEventTouchDown。

  1. 当我们点击上层的按钮时,自动点击系统的_broadPickerView上面的开始录制按钮。

总结:

本文主要论述各个iOS系统版本使用replaykit实现屏幕的技术细节,其他需要考虑的点暂不详述,还包括:

  1. 屏幕方向变化,可以考虑使用RPVideoSampleOrientationKey 对采集的yuv数据结构解析出来方向;

  2. 屏幕锁定的通知,虽然进程级通知提供了锁屏的通知,但是appstore不允许使用,可以考虑使用appdelegate的代理方法来判断;

  3. 采集到数据结构中的yuv的缓存空间,不能占用(例如NSData的initWithBytesNoCopy方法虽然可以快速生成NSData,但是将占用这个缓存),否则将导致进程停止;

  4. 系统提供录制进程的内存空间约为50M,我们在内存占用时需要注意超过50M, 进程将被停止;

作者:杭研融合通信iOS

链接:https://www.jianshu.com/p/401b5b632d5b


以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网

查看所有标签

猜你喜欢:

本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们

微机原理与接口技术

微机原理与接口技术

李文英、李勤、刘星、宋蕴新 / 清华大学出版社 / 2001-9 / 26.00元

《高等院校计算机应用技术规划教材•应用型教材系列•微机原理与接口技术》是“高职高专计算机系列教材”之一。全书包括微机原理、汇编语言、接口技术三部分内容。微机原理部分讲述了80x86的内部结构及工作原理、半导体存储器及其系统、微型机总线结构等。汇编语言部分讲述了指令系统、编程技巧。接口技术部分讲述了中断系统、中断控制器、并行接口、串行接口、DMA控制器、定时器,以及A/D、D/A转换器等常用芯片的硬......一起来看看 《微机原理与接口技术》 这本书的介绍吧!

CSS 压缩/解压工具
CSS 压缩/解压工具

在线压缩/解压 CSS 代码

SHA 加密
SHA 加密

SHA 加密工具