【原创】苹果内置录屏SDK-ReplayKit库的使用说明
2015-12-24 15:59
316 查看
1 iOS ReplayKit 录屏SDK 说明 (按照苹果官方的说法是App端加入这些苹果的新特性新SDK更容易被苹果推荐 )
ReplayKit是苹果在iOS9上面提供的一个库组件,可以让玩家在游戏中录制游戏视频,并且可以添加语音评论,然后通过社交网络分享出去。2库的特性说明
要使用ReplayKit需要在工程的Build Phase的Link binary with libraries中加入ReplayKit.framework.目前这个库只支持真机允许,不支持模拟器。
系统版本如果低于iOS9也不支持。
还有这个库支持游戏录屏,但不支持录avplayer播放的视频,这个可能是为了保护视频的版权,避免影视资源被复制拷贝。
视频录制完成之后可以调用ReplayKit的接口显示视频预览页面,对应的接口是返回一个页面的ViewController,至于如何显示这个页面,各个客户端可以自由处理,Demo中只是给了其中一种实现方法。
3库的潜在问题
经过实验,发现ReplayKit有如下情况:录制的启动初始化有时很慢,有见过几十秒才初始化完成的,也碰见过初始化没有成功的。
录制调用了停止接口后系统还会继续录制多几秒的视频。
出现过录制结果为黑屏的情况。
还有这个录屏SDK支付使用麦克风,即是可以一边录制游戏,一边用麦克风讲解。
4Demo说明
附件是Demo的工程,使用Xcode7编译之后可以运行起来(不支持Xcode6,Xcode6没有ReplayKit这个库)连接iPhone或者iPad之后可以编译并运行这个工程,在真机上运行后可以看到如下界面。
参见附件图片
点击 开始按钮 后就会调用开始录屏的接口,但这个时候不是马上进行录屏,ReplayKit需要初始化完成开自动开始录屏,所以Demo加了一个Loading提示“初始化”
初始化完成后 结束 按钮变为可以点击的状态,并提示 “正在录制”
等要结束时点击 结束按钮,会调用ReplayKit的停止接口,停止接口给了回调后可以显示录屏视频的预览页面,至于要不要显示和如何显示,由各个游戏的前端确定,Demo只是给了个参考的例子。
在视频预览页面可以选择保存到系统相册或者分享到社交网络,还可以拷贝到剪切板,这些操作都可以在回调中获取到,游戏前端可以根据这些回调的信息给用户提示(比如“视频成功保存到系统相册”)
>> 系统默认的分享暂时是看到Facebook Youtube这些,并没有看到有微信微博分享。不过这些视频保存到系统相册之后可以上传到优酷,后续生成链接并分享出去的还没有测试过。
Demo中的时间和进度条只是模拟了游戏中的动画,不然只有静止画面,看不出视频的效果。
其他:Demo中的函数都附带了注释说明,可以自由修改并自由分发。
5官网说明
ReplayKit的官网使用说明 https://developer.apple.com/library/ios/documentation/ReplayKit/Reference/ReplayKit_Collection/index.html#//apple_ref/doc/uid/TP40016260;Demo代码:
#import "ViewController.h" #import <ReplayKit/ReplayKit.h> static NSString *StartRecord = @"开始"; static NSString *StopRecord = @"结束"; #if TARGET_IPHONE_SIMULATOR #define SIMULATOR 1 #elif TARGET_OS_IPHONE #define SIMULATOR 0 #endif #define AnimationDuration (0.3) @interface ViewController () <RPPreviewViewControllerDelegate> { } @property (nonatomic, strong)UIButton *btnStart; @property (nonatomic, strong)UIButton *btnStop; @property (nonatomic, strong)NSTimer *progressTimer; @property (nonatomic, strong)UIProgressView *progressView; @property (nonatomic, strong)UIActivityIndicatorView *activity; @property (nonatomic, strong)UIView *tipView; @property (nonatomic, strong)UILabel *lbTip; @property (nonatomic, strong)UILabel *lbTime; @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; // Do any additional setup after loading the view, typically from a nib. } - (void)viewDidAppear:(BOOL)animated { BOOL isVersionOk = [self isSystemVersionOk]; if (!isVersionOk) { NSLog(@"系统版本需要是iOS9.0及以上才支持ReplayKit"); return; } if (SIMULATOR) { [self showSimulatorWarning]; return; } UILabel *lb = nil; CGSize screenSize = [UIScreen mainScreen].bounds.size; //标题 lb = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 320, 140)]; lb.font = [UIFont boldSystemFontOfSize:32]; lb.backgroundColor = [UIColor clearColor]; lb.textColor = [UIColor blackColor]; lb.textAlignment = NSTextAlignmentCenter; lb.numberOfLines = 3; lb.text = @"苹果ReplayKit Demo"; lb.center = CGPointMake(screenSize.width/2, 80); [self.view addSubview:lb]; //创建按钮 UIButton *btn = [self createButtonWithTitle:StartRecord andCenter:CGPointMake(screenSize.width/2 - 100, 200)]; [self.view addSubview:btn]; self.btnStart = btn; btn = [self createButtonWithTitle:StopRecord andCenter:CGPointMake(screenSize.width/2 + 100, 200)]; [self.view addSubview:btn]; self.btnStop = btn; [self setButton:btn enabled:NO]; //loading指示 UIActivityIndicatorView *activity = [[UIActivityIndicatorView alloc] initWithActivityIndicatorStyle:UIActivityIndicatorViewStyleWhiteLarge]; UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 280, 80)]; [self.view addSubview:view]; view.backgroundColor = [UIColor redColor]; view.layer.cornerRadius = 8.0f; view.center = CGPointMake(screenSize.width/2, 300); activity.center = CGPointMake(30, view.frame.size.height/2); [view addSubview:activity]; [activity startAnimating]; self.activity = activity; lb = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 280, 80)]; lb.font = [UIFont boldSystemFontOfSize:20]; lb.backgroundColor = [UIColor clearColor]; lb.textColor = [UIColor blackColor]; lb.layer.cornerRadius = 4.0; lb.textAlignment = NSTextAlignmentCenter; [view addSubview:lb]; self.lbTip = lb; self.tipView = view; [self hideTip]; //显示时间(用于看录制结果时能知道时间) lb = [[UILabel alloc] initWithFrame:CGRectMake(0, 0, 320, 40)]; lb.font = [UIFont boldSystemFontOfSize:20]; lb.backgroundColor = [UIColor redColor]; lb.textColor = [UIColor blackColor]; lb.layer.cornerRadius = 4.0; NSDateFormatter * dateFormat = [[NSDateFormatter alloc] init] ; [dateFormat setDateFormat: @"HH:mm:ss"]; NSString *dateString = [dateFormat stringFromDate:[NSDate date]]; lb.text = dateString; lb.center = CGPointMake(screenSize.width/2, screenSize.height/2 + 100); lb.textAlignment = NSTextAlignmentCenter; [self.view addSubview:lb]; self.lbTime = lb; //进度条 (显示动画,不然看不出画面的变化) UIProgressView *progress = [[UIProgressView alloc] initWithFrame:CGRectMake(0, 0, screenSize.width*0.8, 10)]; progress.center = CGPointMake(screenSize.width/2, screenSize.height/2 + 150); progress.progressViewStyle = UIProgressViewStyleDefault; progress.progress = 0.0; [self.view addSubview:progress]; self.progressView = progress; //计时器 //更新时间 [NSTimer scheduledTimerWithTimeInterval:1.0f target:self selector:@selector(updateTimeString) userInfo:nil repeats:YES]; } #pragma mark - UI控件 //显示 提示信息 - (void)showTipWithText:(NSString *)tip activity:(BOOL)activity{ [self.activity startAnimating]; self.lbTip.text = tip; self.tipView.hidden = NO; if (activity) { self.activity.hidden = NO; [self.activity startAnimating]; } else { [self.activity stopAnimating]; self.activity.hidden = YES; } } //隐藏 提示信息 - (void)hideTip { self.tipView.hidden = YES; [self.activity stopAnimating]; } //创建按钮 - (UIButton *)createButtonWithTitle:(NSString *)title andCenter:(CGPoint)center { CGRect rect = CGRectMake(0, 0, 160, 60); UIButton *btn = [[UIButton alloc] initWithFrame:rect]; btn.layer.cornerRadius = 5.0; btn.layer.borderWidth = 2.0; btn.layer.borderColor = [[UIColor blackColor] CGColor]; btn.backgroundColor = [UIColor lightGrayColor]; btn.center = center; [btn setTitle:title forState:UIControlStateNormal]; [btn setTitleColor:[UIColor blackColor] forState:UIControlStateNormal]; [btn addTarget:self action:@selector(onBtnPressed:) forControlEvents:UIControlEventTouchDown]; return btn; } //设置按钮是否可点击 - (void)setButton:(UIButton *)button enabled:(BOOL)enabled { if (enabled) { button.alpha = 1.0; } else { button.alpha = 0.2; } button.enabled = enabled; } //提示不支持模拟器 - (void)showSimulatorWarning { UIAlertAction *actionOK = [UIAlertAction actionWithTitle:@"好的" style:UIAlertActionStyleDefault handler:^(UIAlertAction *action){ }]; UIAlertAction *actionCancel = [UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction *action){ }]; UIAlertController *alert = [UIAlertController alertControllerWithTitle:@"ReplayKit不支持模拟器" message:@"请使用真机运行这个Demo工程" preferredStyle:UIAlertControllerStyleAlert]; [alert addAction:actionCancel]; [alert addAction:actionOK]; [self presentViewController:alert animated:NO completion:nil]; } //显示弹框提示 - (void)showAlert:(NSString *)title andMessage:(NSString *)message { if (!title) { title = @""; } if (!message) { message = @""; } UIAlertAction *actionCancel = [UIAlertAction actionWithTitle:@"好的" style:UIAlertActionStyleCancel handler:nil]; UIAlertController *alert = [UIAlertController alertControllerWithTitle:title message:message preferredStyle:UIAlertControllerStyleAlert]; [alert addAction:actionCancel]; [self presentViewController:alert animated:NO completion:nil]; } //显示视频预览页面,animation=是否要动画显示 - (void)showVideoPreviewController:(RPPreviewViewController *)previewController withAnimation:(BOOL)animation { __weak ViewController *weakSelf = self; //UI需要放到主线程 dispatch_async(dispatch_get_main_queue(), ^{ CGRect rect = [UIScreen mainScreen].bounds; if (animation) { rect.origin.x += rect.size.width; previewController.view.frame = rect; rect.origin.x -= rect.size.width; [UIView animateWithDuration:AnimationDuration animations:^(){ previewController.view.frame = rect; } completion:^(BOOL finished){ }]; } else { previewController.view.frame = rect; } [weakSelf.view addSubview:previewController.view]; [weakSelf addChildViewController:previewController]; }); } //关闭视频预览页面,animation=是否要动画显示 - (void)hideVideoPreviewController:(RPPreviewViewController *)previewController withAnimation:(BOOL)animation { //UI需要放到主线程 dispatch_async(dispatch_get_main_queue(), ^{ CGRect rect = previewController.view.frame; if (animation) { rect.origin.x += rect.size.width; [UIView animateWithDuration:AnimationDuration animations:^(){ previewController.view.frame = rect; } completion:^(BOOL finished){ //移除页面 [previewController.view removeFromSuperview]; [previewController removeFromParentViewController]; }]; } else { //移除页面 [previewController.view removeFromSuperview]; [previewController removeFromParentViewController]; } }); } #pragma mark - 按钮 回调 //按钮事件 - (void)onBtnPressed:(UIButton *)sender { //点击效果 sender.transform = CGAffineTransformMakeScale(0.8, 0.8); float duration = 0.3; [UIView animateWithDuration:duration animations:^{ sender.transform = CGAffineTransformMakeScale(1.1, 1.1); }completion:^(BOOL finish){ [UIView animateWithDuration:duration animations:^{ sender.transform = CGAffineTransformMakeScale(1.0, 1.0); }completion:^(BOOL finish){ }]; }]; NSString *function = sender.titleLabel.text; if ([function isEqualToString:StartRecord]) { [self startRecord]; } else if ([function isEqualToString:StopRecord]) { [self stopRecord]; } } - (void)startRecord { // [self setButton:self.btnStart enabled:NO]; NSLog(@"ReplayKit只支持真机录屏,支持游戏录屏,不支持录avplayer播放的视频"); NSLog(@"检查机器和版本是否支持ReplayKit录制..."); if ([[RPScreenRecorder sharedRecorder] isAvailable]) { NSLog(@"支持ReplayKit录制"); } else { NSLog(@"!!不支持支持ReplayKit录制!!"); return; } __weak ViewController *weakSelf = self; NSLog(@"%@ 录制", StartRecord); [self showTipWithText:@"录制初始化" activity:YES]; //在此可以设置是否允许麦克风(传YES即是使用麦克风,传NO则不是用麦克风) [[RPScreenRecorder sharedRecorder] startRecordingWithMicrophoneEnabled:NO handler:^(NSError *error){ NSLog(@"录制开始..."); [weakSelf hideTip]; if (error) { NSLog(@"错误信息 %@", error); [weakSelf showTipWithText:error.description activity:NO]; } else { //其他处理 [weakSelf setButton:self.btnStop enabled:YES]; [weakSelf setButton:self.btnStart enabled:NO]; [weakSelf showTipWithText:@"正在录制" activity:NO]; //更新进度条 weakSelf.progressTimer = [NSTimer scheduledTimerWithTimeInterval:0.05f target:self selector:@selector(changeProgressValue) userInfo:nil repeats:YES]; } }]; } - (void)stopRecord { NSLog(@"%@ 录制", StopRecord); [self setButton:self.btnStart enabled:YES]; [self setButton:self.btnStop enabled:NO]; __weak ViewController *weakSelf = self; [[RPScreenRecorder sharedRecorder] stopRecordingWithHandler:^(RPPreviewViewController *previewViewController, NSError * error){ if (error) { NSLog(@"失败消息:%@", error); [weakSelf showTipWithText:error.description activity:NO]; } else { [weakSelf showTipWithText:@"录制完成" activity:NO]; //显示录制到的视频的预览页 NSLog(@"显示预览页面"); previewViewController.previewControllerDelegate = weakSelf; //去除计时器 [weakSelf.progressTimer invalidate]; weakSelf.progressTimer = nil; [self showVideoPreviewController:previewViewController withAnimation:YES]; } }]; } #pragma mark - 视频预览页面 回调 //关闭的回调 - (void)previewControllerDidFinish:(RPPreviewViewController *)previewController { [self hideVideoPreviewController:previewController withAnimation:YES]; } //选择了某些功能的回调(如分享和保存) - (void)previewController:(RPPreviewViewController *)previewController didFinishWithActivityTypes:(NSSet <NSString *> *)activityTypes { __weak ViewController *weakSelf = self; if ([activityTypes containsObject:@"com.apple.UIKit.activity.SaveToCameraRoll"]) { dispatch_async(dispatch_get_main_queue(), ^{ [weakSelf showAlert:@"保存成功" andMessage:@"已经保存到系统相册"]; }); } if ([activityTypes containsObject:@"com.apple.UIKit.activity.CopyToPasteboard"]) { dispatch_async(dispatch_get_main_queue(), ^{ [weakSelf showAlert:@"复制成功" andMessage:@"已经复制到粘贴板"]; }); } } #pragma mark - 计时器 回调 //改变进度条的显示的进度 - (void)changeProgressValue { float progress = self.progressView.progress + 0.01; [self.progressView setProgress:progress animated:NO]; if (progress >= 1.0) { self.progressView.progress = 0.0; } } //更新显示的时间 - (void)updateTimeString { NSDateFormatter * dateFormat = [[NSDateFormatter alloc] init] ; [dateFormat setDateFormat: @"HH:mm:ss"]; NSString *dateString = [dateFormat stringFromDate:[NSDate date]]; self.lbTime.text = dateString; } #pragma mark - 其他 - (void)didReceiveMemoryWarning { [super didReceiveMemoryWarning]; // Dispose of any resources that can be recreated. } //判断对应系统版本是否支持ReplayKit - (BOOL)isSystemVersionOk { if ([[UIDevice currentDevice].systemVersion floatValue] < 9.0) { return NO; } else { return YES; } } @end
相关文章推荐
- CUDA学习之从CPU架构说起
- C#Asp.Net中mongo数据库的基本使用
- Web - JSONP和同源策略实例讲解
- js动态提示输入框剩余字符数
- Android使用动画的注意事项
- 浅析TCP RST
- 给大家来个事务的使用方式
- 蓝屏:Dumping Physical memory to disk
- CheungSSH国产自动化运维工具开源Web界面
- leetcode:88 Merge Sorted Array-每日编程第三十一题
- B样条曲线
- winpython好用到哭了,自带numpy,scipy,matplotlib,scikit-learn
- arm汇编指令的条件码
- 事务的特性和隔离级别
- 通过PHP自带的服务器来查看正则匹配结果的方法
- iOS 常用计算宏
- 寒江独钓前辈的第一个例子的部分分析
- 结构型:设计模式之适配器模式(六)
- 【HTML第二本】HTML.5与CSS.3权威指南 HTML5部分读书笔记
- Android PullToRefresh 完全解析