diff --git a/Example/PLShortVideoKitDemo.xcodeproj/project.pbxproj b/Example/PLShortVideoKitDemo.xcodeproj/project.pbxproj index 221cc012..225e039d 100644 --- a/Example/PLShortVideoKitDemo.xcodeproj/project.pbxproj +++ b/Example/PLShortVideoKitDemo.xcodeproj/project.pbxproj @@ -204,6 +204,7 @@ 76F41C52219D77B40099E668 /* mulitRate.png in Resources */ = {isa = PBXBuildFile; fileRef = 76F41C51219D77B40099E668 /* mulitRate.png */; }; 9343A8C01F66763A007213BD /* Time_Machine_No_Reverser.png in Resources */ = {isa = PBXBuildFile; fileRef = 9343A8BE1F66763A007213BD /* Time_Machine_No_Reverser.png */; }; 9343A8C11F66763A007213BD /* Time_Machine_Reverser.png in Resources */ = {isa = PBXBuildFile; fileRef = 9343A8BF1F66763A007213BD /* Time_Machine_Reverser.png */; }; + BC8DD6BA224E320200E05B26 /* PLScreenRecorderManager.m in Sources */ = {isa = PBXBuildFile; fileRef = BC8DD6B9224E320200E05B26 /* PLScreenRecorderManager.m */; }; BCCF75E121C3884500BF9BA2 /* ImageVideoMixViewController.m in Sources */ = {isa = PBXBuildFile; fileRef = BCCF75DF21C3884500BF9BA2 /* ImageVideoMixViewController.m */; }; C35043641FB3152600F9633D /* draft_video.png in Resources */ = {isa = PBXBuildFile; fileRef = C35043631FB3121E00F9633D /* draft_video.png */; }; C35043971FBED6B900F9633D /* music_no_selected.png in Resources */ = {isa = PBXBuildFile; fileRef = C35043941FBED11C00F9633D /* music_no_selected.png */; }; @@ -488,6 +489,8 @@ 85B60FAE619932ED649E6840 /* Pods-PLShortVideoKitDemo.release.xcconfig */ = {isa = PBXFileReference; includeInIndex = 1; lastKnownFileType = text.xcconfig; name = "Pods-PLShortVideoKitDemo.release.xcconfig"; path = "Pods/Target Support Files/Pods-PLShortVideoKitDemo/Pods-PLShortVideoKitDemo.release.xcconfig"; sourceTree = ""; }; 9343A8BE1F66763A007213BD /* Time_Machine_No_Reverser.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; name = Time_Machine_No_Reverser.png; path = images/Time_Machine_No_Reverser.png; sourceTree = ""; }; 9343A8BF1F66763A007213BD /* Time_Machine_Reverser.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; name = Time_Machine_Reverser.png; path = images/Time_Machine_Reverser.png; sourceTree = ""; }; + BC8DD6B8224E320200E05B26 /* PLScreenRecorderManager.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = PLScreenRecorderManager.h; sourceTree = ""; }; + BC8DD6B9224E320200E05B26 /* PLScreenRecorderManager.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = PLScreenRecorderManager.m; sourceTree = ""; }; BCCF75DF21C3884500BF9BA2 /* ImageVideoMixViewController.m */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.objc; path = ImageVideoMixViewController.m; sourceTree = ""; }; BCCF75E021C3884500BF9BA2 /* ImageVideoMixViewController.h */ = {isa = PBXFileReference; fileEncoding = 4; lastKnownFileType = sourcecode.c.h; path = ImageVideoMixViewController.h; sourceTree = ""; }; C35043631FB3121E00F9633D /* draft_video.png */ = {isa = PBXFileReference; lastKnownFileType = image.png; name = draft_video.png; path = images/draft_video.png; sourceTree = ""; }; @@ -875,6 +878,8 @@ 0A6340301F15E8CC005ADF6C /* PLSSelectionView.m */, 0A534EA8200E27F00032A981 /* PLSViewRecorderManager.h */, 0A534EAA200E27F10032A981 /* PLSViewRecorderManager.m */, + BC8DD6B8224E320200E05B26 /* PLScreenRecorderManager.h */, + BC8DD6B9224E320200E05B26 /* PLScreenRecorderManager.m */, ); path = "UI+Tools"; sourceTree = ""; @@ -1236,6 +1241,7 @@ 0AD98DC4203173F5006551EC /* H265MovieViewController.m in Sources */, 760AE4F820B014FD00DA4EC3 /* VideoSelectViewController.m in Sources */, 760F7706202AF92A0052F513 /* BaseViewController.m in Sources */, + BC8DD6BA224E320200E05B26 /* PLScreenRecorderManager.m in Sources */, 0A398F6C1ED6259000996229 /* PhotoAlbumViewController.m in Sources */, 0AE8AC951F063E3E006166E2 /* PLSClipAudioView.m in Sources */, C39EA3D71F4E753D0007D1C3 /* PLSRateButtonView.m in Sources */, diff --git a/Example/PLShortVideoKitDemo/EditViewController.m b/Example/PLShortVideoKitDemo/EditViewController.m index 841f3904..10a7d49e 100644 --- a/Example/PLShortVideoKitDemo/EditViewController.m +++ b/Example/PLShortVideoKitDemo/EditViewController.m @@ -1153,6 +1153,9 @@ - (void)addMVLayerWithColor:(NSURL *)colorURL alpha:(NSURL *)alphaURL { } [self.shortVideoEditor addMVLayerWithColor:self.colorURL alpha:self.alphaURL timeRange:kCMTimeRangeZero loopEnable:YES]; + if (![self.shortVideoEditor isEditing]) { + [self.shortVideoEditor startEditing]; + } } - (void)addFilter:(NSString *)colorImagePath { @@ -2249,7 +2252,9 @@ - (void)nextButtonClick { } // 添加背景音乐信息 - [self.audioSettingsArray insertObject:self.backgroundAudioSettings atIndex:0]; + if (self.backgroundAudioSettings[PLSURLKey] && ![self.audioSettingsArray containsObject:self.backgroundAudioSettings]) { + [self.audioSettingsArray insertObject:self.backgroundAudioSettings atIndex:0]; + } AVAsset *asset = self.movieSettings[PLSAssetKey]; PLSAVAssetExportSession *exportSession = [[PLSAVAssetExportSession alloc] initWithAsset:asset]; @@ -2271,8 +2276,12 @@ - (void)nextButtonClick { // 旋转视频 exportSession.videoLayerOrientation = self.videoLayerOrientation; - [exportSession addFilter:self.colorImagePath]; - [exportSession addMVLayerWithColor:self.colorURL alpha:self.alphaURL timeRange:kCMTimeRangeZero loopEnable:YES]; + if (self.colorImagePath) { + [exportSession addFilter:self.colorImagePath]; + } + if (self.colorURL && self.alphaURL) { + [exportSession addMVLayerWithColor:self.colorURL alpha:self.alphaURL timeRange:kCMTimeRangeZero loopEnable:YES]; + } __weak typeof(self) weakSelf = self; [exportSession setCompletionBlock:^(NSURL *url) { @@ -2295,7 +2304,9 @@ - (void)nextButtonClick { [exportSession setProcessingBlock:^(float progress) { // 更新进度 UI NSLog(@"Asset Export Progress: %f", progress); - weakSelf.progressLabel.text = [NSString stringWithFormat:@"%d%%", (int)(progress * 100)]; + dispatch_async(dispatch_get_main_queue(), ^{ + weakSelf.progressLabel.text = [NSString stringWithFormat:@"%d%%", (int)(progress * 100)]; + }); }]; [exportSession exportAsynchronously]; diff --git a/Example/PLShortVideoKitDemo/Info.plist b/Example/PLShortVideoKitDemo/Info.plist index 15ab1b6c..f91c572b 100644 --- a/Example/PLShortVideoKitDemo/Info.plist +++ b/Example/PLShortVideoKitDemo/Info.plist @@ -15,9 +15,9 @@ CFBundlePackageType APPL CFBundleShortVersionString - 1.16.1 + 2.0.0 CFBundleVersion - 1.16.1.git-2019-03-29-38e723e + 2.0.0.git-2019-03-29-1d37a62 Fabric APIKey diff --git a/Example/PLShortVideoKitDemo/Librarys/libMuseProcessor.a b/Example/PLShortVideoKitDemo/Librarys/libMuseProcessor.a index bc296285..ed5e2127 100644 Binary files a/Example/PLShortVideoKitDemo/Librarys/libMuseProcessor.a and b/Example/PLShortVideoKitDemo/Librarys/libMuseProcessor.a differ diff --git a/Example/PLShortVideoKitDemo/RecordViewController.m b/Example/PLShortVideoKitDemo/RecordViewController.m index 9c3e0681..9b2ecbca 100644 --- a/Example/PLShortVideoKitDemo/RecordViewController.m +++ b/Example/PLShortVideoKitDemo/RecordViewController.m @@ -17,6 +17,7 @@ #import "PLSFilterGroup.h" #import "PLSViewRecorderManager.h" #import "PLSRateButtonView.h" +#import "PLScreenRecorderManager.h" #define AlertViewShow(msg) [[[UIAlertView alloc] initWithTitle:@"warning" message:[NSString stringWithFormat:@"%@", msg] delegate:nil cancelButtonTitle:@"ok" otherButtonTitles:nil] show] @@ -37,13 +38,15 @@ @interface RecordViewController () UICollectionViewDataSource, UICollectionViewDelegateFlowLayout, PLSViewRecorderManagerDelegate, -PLSRateButtonViewDelegate +PLSRateButtonViewDelegate, +PLScreenRecorderManagerDelegate > @property (strong, nonatomic) PLSVideoConfiguration *videoConfiguration; @property (strong, nonatomic) PLSAudioConfiguration *audioConfiguration; @property (strong, nonatomic) PLShortVideoRecorder *shortVideoRecorder; @property (strong, nonatomic) PLSViewRecorderManager *viewRecorderManager; +@property (strong, nonatomic) PLScreenRecorderManager *screenRecorderManager; @property (strong, nonatomic) PLSProgressBar *progressBar; @property (strong, nonatomic) UIButton *recordButton; @property (strong, nonatomic) UIButton *viewRecordButton; @@ -88,6 +91,8 @@ @interface RecordViewController () @property (strong, nonatomic) UIButton *monitorButton; // 实时截图按钮 @property (strong, nonatomic) UIButton *snapshotButton; +// 帧率切换按钮 +@property (strong, nonatomic) UIButton *frameRateButton; // 录制前是否开启自动检测设备方向调整视频拍摄的角度(竖屏、横屏) @property (assign, nonatomic) BOOL isUseAutoCheckDeviceOrientationBeforeRecording; @@ -166,10 +171,11 @@ - (void)setupShortVideoRecorder { self.videoConfiguration = [PLSVideoConfiguration defaultConfiguration]; self.videoConfiguration.position = AVCaptureDevicePositionFront; - self.videoConfiguration.videoFrameRate = 25; - self.videoConfiguration.averageVideoBitRate = 1024*1000; - self.videoConfiguration.videoSize = CGSizeMake(544, 960); + self.videoConfiguration.videoFrameRate = 30; + self.videoConfiguration.averageVideoBitRate = 1000*2500; + self.videoConfiguration.videoSize = CGSizeMake(720, 1280); self.videoConfiguration.videoOrientation = AVCaptureVideoOrientationPortrait; + self.videoConfiguration.sessionPreset = AVCaptureSessionPreset1280x720; self.audioConfiguration = [PLSAudioConfiguration defaultConfiguration]; @@ -393,6 +399,17 @@ - (void)setupRightButtonView { [self.musicButton addTarget:self action:@selector(musicButtonOnClick:) forControlEvents:UIControlEventTouchUpInside]; [self.rightScrollView addSubview:self.musicButton]; + index ++; + // 30FPS/60FPS + self.frameRateButton = [[UIButton alloc] initWithFrame:CGRectMake(0, index * 60 + 10, 46, 46)]; + self.frameRateButton.layer.cornerRadius = 23; + self.frameRateButton.backgroundColor = backgroundColor; + [self.frameRateButton setTitle:@"30帧" forState:(UIControlStateNormal)]; + self.frameRateButton.titleLabel.font = [UIFont systemFontOfSize:14]; + [self.frameRateButton setTitleColor:[UIColor whiteColor] forState:UIControlStateNormal]; + [self.frameRateButton addTarget:self action:@selector(frameRateButtonOnClick:) forControlEvents:UIControlEventTouchUpInside]; + [self.rightScrollView addSubview:self.frameRateButton]; + index ++; //是否开启 SDK 退到后台监听 self.monitorButton = [UIButton buttonWithType:UIButtonTypeCustom]; @@ -544,6 +561,7 @@ - (void)getFirstMovieFromPhotoAlbum { - (void)backButtonEvent:(id)sender { if (self.viewRecordButton.isSelected) { [self.viewRecorderManager cancelRecording]; + [self.screenRecorderManager cancelRecording]; } if ([self.shortVideoRecorder getFilesCount] > 0) { self.alertView = [[UIAlertView alloc] initWithTitle:@"提醒" message:[NSString stringWithFormat:@"放弃这个视频(共%ld个视频段)?", (long)[self.shortVideoRecorder getFilesCount]] delegate:self cancelButtonTitle:@"取消" otherButtonTitles:@"确定", nil]; @@ -559,7 +577,7 @@ - (void)squareRecordButtonEvent:(id)sender { UIButton *button = (UIButton *)sender; button.selected = !button.selected; if (button.selected) { - self.videoConfiguration.videoSize = CGSizeMake(480, 480); + self.videoConfiguration.videoSize = CGSizeMake(720, 720); [self.shortVideoRecorder reloadvideoConfiguration:self.videoConfiguration]; self.shortVideoRecorder.maxDuration = 10.0f; @@ -571,7 +589,7 @@ - (void)squareRecordButtonEvent:(id)sender { }); } else { - self.videoConfiguration.videoSize = CGSizeMake(544, 960); + self.videoConfiguration.videoSize = CGSizeMake(720, 1280); [self.shortVideoRecorder reloadvideoConfiguration:self.videoConfiguration]; self.shortVideoRecorder.maxDuration = 10.0f; @@ -585,26 +603,39 @@ - (void)squareRecordButtonEvent:(id)sender { //录制 self.view - (void)viewRecorderButtonClick:(id)sender { - if (!self.viewRecorderManager) { - self.viewRecorderManager = [[PLSViewRecorderManager alloc] initWithRecordedView:self.view]; - self.viewRecorderManager.delegate = self; - } - - if (self.viewRecordButton.isSelected) { - self.viewRecordButton.selected = NO; - [self.viewRecorderManager stopRecording]; - - [[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationWillResignActiveNotification object:nil]; - } - else { - self.viewRecordButton.selected = YES; - [self.viewRecorderManager startRecording]; + if (@available(iOS 11.0, *)) { + if (!self.screenRecorderManager) { + self.screenRecorderManager = [[PLScreenRecorderManager alloc] init]; + self.screenRecorderManager.delegate = self; + } + if (self.viewRecordButton.isSelected) { + self.viewRecordButton.selected = NO; + [self.screenRecorderManager stopRecording]; + } else { + self.viewRecordButton.selected = YES; + [self.screenRecorderManager startRecording]; + } + } else { + if (!self.viewRecorderManager) { + self.viewRecorderManager = [[PLSViewRecorderManager alloc] initWithRecordedView:self.shortVideoRecorder.previewView]; + self.viewRecorderManager.delegate = self; + } - [[NSNotificationCenter defaultCenter] addObserver:self - selector:@selector(applicationWillResignActive:) - name:UIApplicationWillResignActiveNotification - object:nil]; - } + if (self.viewRecordButton.isSelected) { + self.viewRecordButton.selected = NO; + [self.viewRecorderManager stopRecording]; + + [[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationWillResignActiveNotification object:nil]; + } + else { + self.viewRecordButton.selected = YES; + [self.viewRecorderManager startRecording]; + + [[NSNotificationCenter defaultCenter] addObserver:self + selector:@selector(applicationWillResignActive:) + name:UIApplicationWillResignActiveNotification + object:nil]; + }} } // 打开/关闭闪光灯 @@ -626,8 +657,22 @@ - (void)beautyFaceButtonEvent:(id)sender { } // 切换前后置摄像头 -- (void)toggleCameraButtonEvent:(id)sender { - [self.shortVideoRecorder toggleCamera]; +- (void)toggleCameraButtonEvent:(UIButton *)sender { + // 采集帧率不大于 30 帧的时候,使用 [self.shortVideoRecorder toggleCamera] 和 [self.shortVideoRecorder toggleCamera:block] 都可以。当采集大于 30 帧的时候,为确保切换成功,需要先停止采集,再切换相机,切换完成再启动采集。如果不先停止采集,部分机型上采集 60 帧的时候,切换摄像头可能会耗时几秒钟 + if (self.videoConfiguration.videoFrameRate > 30) { + sender.enabled = NO; + __weak typeof(self) weakself = self; + [self.shortVideoRecorder stopCaptureSession]; + [self.shortVideoRecorder toggleCamera:^(BOOL isFinish) { + [weakself checkActiveFormat];// 默认的 active 可能最大只支持采集 30 帧,这里手动设置一下 + [weakself.shortVideoRecorder startCaptureSession]; + dispatch_async(dispatch_get_main_queue(), ^{ + sender.enabled = YES; + }); + }]; + } else { + [self.shortVideoRecorder toggleCamera]; + } } // 七牛滤镜 @@ -671,6 +716,23 @@ - (void)musicButtonOnClick:(id)sender { } } +- (void)frameRateButtonOnClick:(UIButton *)button { + if (60 == self.videoConfiguration.videoFrameRate) { + self.videoConfiguration.videoFrameRate = 30; + self.videoConfiguration.averageVideoBitRate = 1000 * 2500; + self.videoConfiguration.sessionPreset = AVCaptureSessionPreset1280x720; + [button setTitle:@"30帧" forState:(UIControlStateNormal)]; + [self.shortVideoRecorder reloadvideoConfiguration:self.videoConfiguration]; + } else { + self.videoConfiguration.videoFrameRate = 60; + self.videoConfiguration.averageVideoBitRate = 1000 * 3500; + self.videoConfiguration.sessionPreset = AVCaptureSessionPresetInputPriority; + [button setTitle:@"60帧" forState:(UIControlStateNormal)]; + [self.shortVideoRecorder reloadvideoConfiguration:self.videoConfiguration]; + [self checkActiveFormat]; + } +} + // 拍照 -(void)snapshotButtonOnClick:(UIButton *)sender { sender.enabled = NO; @@ -767,6 +829,7 @@ - (void)endButtonEvent:(id)sender { AVAsset *asset = self.shortVideoRecorder.assetRepresentingAllFiles; [self playEvent:asset]; [self.viewRecorderManager cancelRecording]; + [self.screenRecorderManager cancelRecording]; self.viewRecordButton.selected = NO; } @@ -857,6 +920,31 @@ - (void)viewRecorderManager:(PLSViewRecorderManager *)manager didFinishRecording [self presentViewController:videoEditViewController animated:YES completion:nil]; } +#pragma mark - PLScreenRecorderManagerDelegate +- (void)screenRecorderManager:(PLScreenRecorderManager *)manager didFinishRecordingToAsset:(AVAsset *)asset totalDuration:(CGFloat)totalDuration { + self.viewRecordButton.selected = NO; + // 设置音视频、水印等编辑信息 + NSMutableDictionary *outputSettings = [[NSMutableDictionary alloc] init]; + // 待编辑的原始视频素材 + NSMutableDictionary *plsMovieSettings = [[NSMutableDictionary alloc] init]; + plsMovieSettings[PLSAssetKey] = asset; + plsMovieSettings[PLSStartTimeKey] = [NSNumber numberWithFloat:0.f]; + plsMovieSettings[PLSDurationKey] = [NSNumber numberWithFloat:totalDuration]; + plsMovieSettings[PLSVolumeKey] = [NSNumber numberWithFloat:1.0f]; + outputSettings[PLSMovieSettingsKey] = plsMovieSettings; + + EditViewController *videoEditViewController = [[EditViewController alloc] init]; + videoEditViewController.settings = outputSettings; + [self presentViewController:videoEditViewController animated:YES completion:nil]; +} + +- (void)screenRecorderManager:(PLScreenRecorderManager *)manager errorOccur:(NSError *)error { + NSString *message = [NSString stringWithFormat:@"%@", error]; + UIAlertView *alert = [[UIAlertView alloc] initWithTitle:@"错误" message:message delegate:nil cancelButtonTitle:@"确定" otherButtonTitles: nil]; + [alert show]; + self.viewRecordButton.selected = NO; +} + #pragma mark -- PLShortVideoRecorderDelegate 摄像头/麦克风鉴权的回调 - (void)shortVideoRecorder:(PLShortVideoRecorder *__nonnull)recorder didGetCameraAuthorizationStatus:(PLSAuthorizationStatus)status { if (status == PLSAuthorizationStatusAuthorized) { @@ -916,6 +1004,7 @@ - (void)shortVideoRecorder:(PLShortVideoRecorder *)recorder didRecordingToOutput self.importMovieView.hidden = YES; self.musicButton.hidden = YES; self.filePathButton.hidden = YES; + self.frameRateButton.hidden = YES; self.durationLabel.text = [NSString stringWithFormat:@"%.2fs", totalDuration]; } @@ -933,6 +1022,7 @@ - (void)shortVideoRecorder:(PLShortVideoRecorder *)recorder didDeleteFileAtURL:( self.importMovieView.hidden = NO; self.musicButton.hidden = NO; self.filePathButton.hidden = NO; + self.frameRateButton.hidden = NO; } AVAsset *asset = [AVAsset assetWithURL:_URL]; @@ -1211,5 +1301,39 @@ - (void)applicationDidBecomeActive:(id)sender { NSLog(@"%s, %d, applicationDidBecomeActive:", __func__, __LINE__); } +- (void)checkActiveFormat { + + CGSize needCaptureSize = self.videoConfiguration.videoSize; + + if (AVCaptureVideoOrientationPortrait == self.videoConfiguration.videoOrientation || + AVCaptureVideoOrientationPortraitUpsideDown == self.videoConfiguration.videoOrientation) { + needCaptureSize = CGSizeMake(self.videoConfiguration.videoSize.height, self.videoConfiguration.videoSize.width); + } + + AVCaptureDeviceFormat *activeFormat = self.shortVideoRecorder.videoActiveFormat; + AVFrameRateRange *frameRateRange = [activeFormat.videoSupportedFrameRateRanges firstObject]; + + CMVideoDimensions captureSize = CMVideoFormatDescriptionGetDimensions(activeFormat.formatDescription); + if (frameRateRange.maxFrameRate < self.videoConfiguration.videoFrameRate || + frameRateRange.minFrameRate > self.videoConfiguration.videoFrameRate || + needCaptureSize.width > captureSize.width || + needCaptureSize.height > captureSize.height) { + + NSArray *videoFormats = self.shortVideoRecorder.videoFormats; + for (AVCaptureDeviceFormat *format in videoFormats) { + frameRateRange = [format.videoSupportedFrameRateRanges firstObject]; + captureSize = CMVideoFormatDescriptionGetDimensions(format.formatDescription); + + if (frameRateRange.maxFrameRate >= self.videoConfiguration.videoFrameRate && + frameRateRange.minFrameRate <= self.videoConfiguration.videoFrameRate && + captureSize.width >= needCaptureSize.width && + captureSize.height >= needCaptureSize.height) { + NSLog(@"size = {%d x %d}, fps = %f ~ %f", captureSize.width, captureSize.height, frameRateRange.minFrameRate, frameRateRange.maxFrameRate); + self.shortVideoRecorder.videoActiveFormat = format; + break; + } + } + } +} @end diff --git a/Example/PLShortVideoKitDemo/SegmentEffects/MulitClipViewController.m b/Example/PLShortVideoKitDemo/SegmentEffects/MulitClipViewController.m index c1da9c87..9f9de124 100644 --- a/Example/PLShortVideoKitDemo/SegmentEffects/MulitClipViewController.m +++ b/Example/PLShortVideoKitDemo/SegmentEffects/MulitClipViewController.m @@ -369,7 +369,7 @@ - (void)clipView:(PLSClipMulitMediaView *)clipView finishClip:(NSArray +#import + +NS_ASSUME_NONNULL_BEGIN + +@class PLScreenRecorderManager; + +@protocol PLScreenRecorderManagerDelegate + +@optional +- (void)screenRecorderManager:(PLScreenRecorderManager *)manager didFinishRecordingToAsset:(AVAsset *)asset totalDuration:(CGFloat)totalDuration; +- (void)screenRecorderManager:(PLScreenRecorderManager *)manager errorOccur:(NSError *)error; +@end + +@interface PLScreenRecorderManager : NSObject + +@property (weak, nonatomic) id delegate; + +- (void)startRecording; + +- (void)stopRecording; + +- (void)cancelRecording; + +@end + +NS_ASSUME_NONNULL_END diff --git a/Example/PLShortVideoKitDemo/UI+Tools/PLScreenRecorderManager.m b/Example/PLShortVideoKitDemo/UI+Tools/PLScreenRecorderManager.m new file mode 100644 index 00000000..5b4c5bbf --- /dev/null +++ b/Example/PLShortVideoKitDemo/UI+Tools/PLScreenRecorderManager.m @@ -0,0 +1,152 @@ +// +// PLScreenRecorderManager.m +// PLShortVideoKitDemo +// +// Created by hxiongan on 2019/3/28. +// Copyright © 2019年 Pili Engineering, Qiniu Inc. All rights reserved. +// + +#import "PLScreenRecorderManager.h" +#import +#include +#import + +@interface PLScreenRecorderManager () + +@property (nonatomic, strong) PLShortVideoRecorder *shortVideoRecorder; +@property (nonatomic, weak) RPScreenRecorder *screenRecorder; + +@end + + +@implementation PLScreenRecorderManager + +- (instancetype)init { + self = [super init]; + if (self) { + [self setupShortRecorder]; + [self setupScreenRecorder]; + } + return self; +} + +- (void)setupShortRecorder { + PLSVideoConfiguration *videoConfiguration = [PLSVideoConfiguration defaultConfiguration]; + videoConfiguration.videoFrameRate = 60; + videoConfiguration.videoSize = CGSizeMake(720, 1280); + videoConfiguration.averageVideoBitRate = 3500 * 1000; + PLSAudioConfiguration *audioConfiguration = [PLSAudioConfiguration defaultConfiguration]; + self.shortVideoRecorder = [[PLShortVideoRecorder alloc] initWithVideoConfiguration:videoConfiguration audioConfiguration:audioConfiguration captureEnabled:NO]; + self.shortVideoRecorder.maxDuration = 120; + self.shortVideoRecorder.delegate = self; + self.shortVideoRecorder.backgroundMonitorEnable = NO; +} + +- (void)setupScreenRecorder { + self.screenRecorder = [RPScreenRecorder sharedRecorder]; + self.screenRecorder.microphoneEnabled = YES; +} + +- (void)startRecording { + __weak typeof(self) weakSelf = self; + if (@available(iOS 11.0, *)) { + + [weakSelf.screenRecorder startCaptureWithHandler:^(CMSampleBufferRef _Nonnull sampleBuffer, RPSampleBufferType bufferType, NSError * _Nullable error) { + if (RPSampleBufferTypeVideo == bufferType) { + [weakSelf.shortVideoRecorder writePixelBuffer:CMSampleBufferGetImageBuffer(sampleBuffer) timeStamp:CMSampleBufferGetPresentationTimeStamp(sampleBuffer)]; + } else if (RPSampleBufferTypeAudioMic == bufferType){ + [weakSelf.shortVideoRecorder writeSampleBuffer:sampleBuffer]; + } + } completionHandler:^(NSError * _Nullable error) { + if (error) { + dispatch_async(dispatch_get_main_queue(), ^{ + if ([weakSelf.delegate respondsToSelector:@selector(screenRecorderManager:errorOccur:)]) { + [weakSelf.delegate screenRecorderManager:weakSelf errorOccur:error]; + } + [weakSelf cancelRecording]; + }); + } else { + dispatch_sync(dispatch_get_main_queue(), ^{ + // 注意: [shortVideoRecorder startRecording] 需要再主线程中执行,不然拍的时长会不正确 + [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationDidEnterBackground:) name:UIApplicationDidEnterBackgroundNotification object:nil]; + [weakSelf.shortVideoRecorder deleteAllFiles]; + [weakSelf.shortVideoRecorder startRecording]; + }); + } + }]; + } else { + // Fallback on earlier versions + } +} + +- (void)stopRecording { + if (@available(iOS 11.0, *)) { + [self.screenRecorder stopCaptureWithHandler:^(NSError * _Nullable error) { + if (error) { + NSLog(@"screenRecorder stop error: %@", error); + } + }]; + } else { + // Fallback on earlier versions + } + [self.shortVideoRecorder stopRecording]; + [[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationDidEnterBackgroundNotification object:nil]; +} + +- (void)cancelRecording { + if (@available(iOS 11.0, *)) { + [self.screenRecorder stopCaptureWithHandler:^(NSError * _Nullable error) { + if (error) { + NSLog(@"screenRecorder stop error: %@", error); + } + }]; + } else { + // Fallback on earlier versions + } + NSLog(@"cancel"); + [self.shortVideoRecorder cancelRecording]; + [[NSNotificationCenter defaultCenter] removeObserver:self name:UIApplicationDidEnterBackgroundNotification object:nil]; +} + +- (void)applicationDidEnterBackground:(id)sender { + [self stopRecording]; +} + +- (void)dealloc { + [self cancelRecording]; + NSLog(@"dealloc: %@", self.description); +} + +#pragma mark - Recorder Delegate + +// 开始录制一段视频时 +- (void)shortVideoRecorder:(PLShortVideoRecorder *)recorder didStartRecordingToOutputFileAtURL:(NSURL *)fileURL { + NSLog(@"start recording fileURL: %@", fileURL); +} + +// 完成一段视频的录制时 +- (void)shortVideoRecorder:(PLShortVideoRecorder *)recorder didFinishRecordingToOutputFileAtURL:(NSURL *)fileURL fileDuration:(CGFloat)fileDuration totalDuration:(CGFloat)totalDuration { + NSLog(@"finish recording fileURL: %@, fileDuration: %f, totalDuration: %f", fileURL, fileDuration, totalDuration); + + AVAsset *asset = self.shortVideoRecorder.assetRepresentingAllFiles; + if (self.delegate && [self.delegate respondsToSelector:@selector(screenRecorderManager:didFinishRecordingToAsset:totalDuration:)]) { + [self.delegate screenRecorderManager:self didFinishRecordingToAsset:asset totalDuration:totalDuration]; + } +} + +// 在达到指定的视频录制时间 maxDuration 后,如果再调用 [PLShortVideoRecorder startRecording],直接执行该回调 +- (void)shortVideoRecorder:(PLShortVideoRecorder *)recorder didFinishRecordingMaxDuration:(CGFloat)maxDuration { + NSLog(@"finish recording maxDuration: %f", maxDuration); + if (@available(iOS 11.0, *)) { + [self.screenRecorder stopCaptureWithHandler:^(NSError * _Nullable error) {}]; + } else { + // Fallback on earlier versions + } + AVAsset *asset = self.shortVideoRecorder.assetRepresentingAllFiles; + if (self.delegate && [self.delegate respondsToSelector:@selector(screenRecorderManager:didFinishRecordingToAsset:totalDuration:)]) { + [self.delegate screenRecorderManager:self didFinishRecordingToAsset:asset totalDuration:maxDuration]; + } +} + +@end +