FUVideoRenderViewModel.m 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296
  1. //
  2. // FUVideoRenderViewModel.m
  3. // FULiveDemo
  4. //
  5. // Created by 项林平 on 2022/8/9.
  6. //
  7. #import "FUVideoRenderViewModel.h"
  8. #import <FURenderKit/FUVideoReader.h>
  9. #import <FURenderKit/FUVideoProcessor.h>
  10. @interface FUVideoRenderViewModel ()<FUVideoReaderDelegate>
  11. @property (nonatomic, assign) FUVideoOrientation videoOrientation;
  12. @property (nonatomic, strong) NSURL *videoURL;
  13. /// 保存时需要Processor边读边写
  14. @property (nonatomic, strong) FUVideoProcessor *videoProcessor;
  15. /// 播放时只需要Reader
  16. @property (nonatomic, strong) FUVideoReader *videoReader;
  17. @property (nonatomic, strong) AVPlayer *audioReader;
  18. @property (nonatomic, assign) BOOL isDestroyed;
  19. @property (nonatomic, strong) CADisplayLink *displayLink;
  20. @property (nonatomic, strong) NSOperationQueue *previewQueue;
  21. @end
  22. @implementation FUVideoRenderViewModel {
  23. // 缓存的预览帧
  24. CVPixelBufferRef previewBuffer;
  25. }
  26. #pragma mark - Initializer
  27. - (instancetype)initWithVideoURL:(NSURL *)videoURL {
  28. self = [super init];
  29. if (self) {
  30. self.rendering = YES;
  31. self.detectingParts = FUDetectingPartsFace;
  32. if (self.necessaryAIModelTypes & FUAIModelTypeFace) {
  33. [FURenderKitManager setFaceProcessorDetectMode:FUFaceProcessorDetectModeVideo];
  34. }
  35. if (self.necessaryAIModelTypes & FUAIModelTypeHuman) {
  36. [FURenderKitManager setHumanProcessorDetectMode:FUHumanProcessorDetectModeVideo];
  37. }
  38. self.videoURL = videoURL;
  39. self.previewQueue = [[NSOperationQueue alloc] init];
  40. self.previewQueue.maxConcurrentOperationCount = 1;
  41. }
  42. return self;
  43. }
  44. - (void)dealloc {
  45. if (previewBuffer != NULL) {
  46. CVPixelBufferRelease(previewBuffer);
  47. previewBuffer = NULL;
  48. }
  49. }
  50. #pragma mark - Instance methods
  51. - (void)startPreviewing {
  52. if (previewBuffer == NULL) {
  53. UIImage *previewImage = [FUUtility previewImageFromVideoURL:self.videoURL preferredTrackTransform:NO];
  54. previewBuffer = [FUImageHelper pixelBufferFromImage:previewImage];
  55. }
  56. if (!_displayLink) {
  57. self.displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(displayLinkAction)];
  58. [self.displayLink addToRunLoop:[NSRunLoop currentRunLoop] forMode:NSRunLoopCommonModes];
  59. self.displayLink.frameInterval = 4;
  60. }
  61. self.displayLink.paused = NO;
  62. }
  63. - (void)stopPreviewing {
  64. self.displayLink.paused = YES;
  65. [self.displayLink invalidate];
  66. self.displayLink = nil;
  67. [self.previewQueue cancelAllOperations];
  68. if (previewBuffer != NULL) {
  69. CVPixelBufferRelease(previewBuffer);
  70. previewBuffer = NULL;
  71. }
  72. }
  73. - (void)startReading {
  74. [self stopPreviewing];
  75. // 视频解码
  76. [self.videoReader start];
  77. // 播放音频
  78. [self.audioReader play];
  79. }
  80. - (void)stopReading {
  81. if (_videoReader) {
  82. [self.videoReader stop];
  83. }
  84. if (_audioReader) {
  85. [self.audioReader pause];
  86. _audioReader = nil;
  87. }
  88. }
  89. - (void)startProcessing {
  90. [self stopPreviewing];
  91. self.videoProcessor = [[FUVideoProcessor alloc] initWithReadingURL:self.videoURL writingURL:[NSURL fileURLWithPath:kFUFinalPath]];
  92. @FUWeakify(self)
  93. self.videoProcessor.processingVideoBufferHandler = ^CVPixelBufferRef _Nonnull(CVPixelBufferRef _Nonnull videoPixelBuffer, CGFloat time) {
  94. @FUStrongify(self)
  95. videoPixelBuffer = [self processVideoPixelBuffer:videoPixelBuffer];
  96. if (self.delegate && [self.delegate respondsToSelector:@selector(videoRenderProcessingProgress:)]) {
  97. if (time >= 0 && self.videoProcessor.reader.duration > 0) {
  98. // 计算进度
  99. CGFloat progress = time / self.videoProcessor.reader.duration;
  100. [self.delegate videoRenderProcessingProgress:progress];
  101. }
  102. }
  103. return videoPixelBuffer;
  104. };
  105. self.videoProcessor.processingFinishedHandler = ^{
  106. @FUStrongify(self)
  107. if (self.delegate && [self.delegate respondsToSelector:@selector(videoRenderDidFinishProcessing)]) {
  108. [self.delegate videoRenderDidFinishProcessing];
  109. }
  110. };
  111. [self.videoProcessor startProcessing];
  112. }
  113. - (void)stopProcessing {
  114. if (self.videoProcessor) {
  115. [self.videoProcessor cancelProcessing];
  116. self.videoProcessor = nil;
  117. }
  118. }
  119. - (void)destroy {
  120. [self stopReading];
  121. [self stopPreviewing];
  122. [self stopProcessing];
  123. self.isDestroyed = YES;
  124. }
  125. #pragma mark - Event response
  126. - (void)displayLinkAction {
  127. [self.previewQueue addOperationWithBlock:^{
  128. CVPixelBufferRetain(self->previewBuffer);
  129. [self renderVideoPixelBuffer:self->previewBuffer];
  130. CVPixelBufferRelease(self->previewBuffer);
  131. }];
  132. }
  133. #pragma mark - Private methods
  134. - (void)renderVideoPixelBuffer:(CVPixelBufferRef)videoPixelBuffer {
  135. [FURenderKitManager updateBeautyBlurEffect];
  136. @autoreleasepool {
  137. if (self.isRendering) {
  138. FURenderInput *input = [[FURenderInput alloc] init];
  139. input.pixelBuffer = videoPixelBuffer;
  140. input.renderConfig.imageOrientation = FUImageOrientationUP;
  141. switch (self.videoReader.videoOrientation) {
  142. case FUVideoOrientationPortrait:
  143. input.renderConfig.imageOrientation = FUImageOrientationUP;
  144. break;
  145. case FUVideoOrientationLandscapeRight:
  146. input.renderConfig.imageOrientation = FUImageOrientationLeft;
  147. break;
  148. case FUVideoOrientationUpsideDown:
  149. input.renderConfig.imageOrientation = FUImageOrientationDown;
  150. break;
  151. case FUVideoOrientationLandscapeLeft:
  152. input.renderConfig.imageOrientation = FUImageOrientationRight;
  153. break;
  154. }
  155. FURenderOutput *output = [[FURenderKit shareRenderKit] renderWithInput:input];
  156. videoPixelBuffer = output.pixelBuffer;
  157. }
  158. if (self.delegate && [self.delegate respondsToSelector:@selector(videoRenderDidOutputPixelBuffer:)]) {
  159. [self.delegate videoRenderDidOutputPixelBuffer:videoPixelBuffer];
  160. }
  161. if (self.detectingParts != FUDetectingPartsNone) {
  162. // 需要检查人脸/人体/手势检测状态
  163. if (self.delegate && [self.delegate respondsToSelector:@selector(videoRenderShouldCheckDetectingStatus:)]) {
  164. [self.delegate videoRenderShouldCheckDetectingStatus:self.detectingParts];
  165. }
  166. }
  167. }
  168. }
  169. - (CVPixelBufferRef)processVideoPixelBuffer:(CVPixelBufferRef)videoPixelBuffer {
  170. @autoreleasepool {
  171. FURenderInput *input = [[FURenderInput alloc] init];
  172. input.pixelBuffer = videoPixelBuffer;
  173. input.renderConfig.imageOrientation = FUImageOrientationUP;
  174. switch (self.videoProcessor.reader.videoOrientation) {
  175. case FUVideoOrientationPortrait:
  176. input.renderConfig.imageOrientation = FUImageOrientationUP;
  177. break;
  178. case FUVideoOrientationLandscapeRight:
  179. input.renderConfig.imageOrientation = FUImageOrientationLeft;
  180. break;
  181. case FUVideoOrientationUpsideDown:
  182. input.renderConfig.imageOrientation = FUImageOrientationDown;
  183. break;
  184. case FUVideoOrientationLandscapeLeft:
  185. input.renderConfig.imageOrientation = FUImageOrientationRight;
  186. break;
  187. }
  188. FURenderOutput *output = [[FURenderKit shareRenderKit] renderWithInput:input];
  189. videoPixelBuffer = output.pixelBuffer;
  190. }
  191. return videoPixelBuffer;
  192. }
  193. #pragma mark - FUVideoReaderDelegate
  194. - (void)videoReaderDidOutputVideoSampleBuffer:(CMSampleBufferRef)videoSampleBuffer {
  195. CVPixelBufferRef videoBuffer = CMSampleBufferGetImageBuffer(videoSampleBuffer);
  196. [self renderVideoPixelBuffer:videoBuffer];
  197. CMSampleBufferInvalidate(videoSampleBuffer);
  198. CFRelease(videoSampleBuffer);
  199. }
  200. - (void)videoReaderDidFinishReading {
  201. [self stopReading];
  202. if (self.delegate && [self.delegate respondsToSelector:@selector(videoRenderDidFinishReading)]) {
  203. [self.delegate videoRenderDidFinishReading];
  204. }
  205. // 获取最后视频帧,需要循环render预览
  206. UIImage *lastImage = [FUUtility lastFrameImageFromVideoURL:self.videoURL preferredTrackTransform:NO];
  207. previewBuffer = [FUImageHelper pixelBufferFromImage:lastImage];
  208. if (!self.isDestroyed) {
  209. dispatch_async(dispatch_get_main_queue(), ^{
  210. [self startPreviewing];
  211. });
  212. }
  213. }
  214. #pragma mark - Getters
  215. - (FUVideoReader *)videoReader {
  216. if (!_videoReader) {
  217. FUVideoReaderSettings *settings = [[FUVideoReaderSettings alloc] init];
  218. settings.readingAutomatically = YES;
  219. settings.videoOutputFormat = kCVPixelFormatType_32BGRA;
  220. _videoReader = [[FUVideoReader alloc] initWithURL:self.videoURL settings:settings];
  221. _videoReader.delegate = self;
  222. }
  223. return _videoReader;
  224. }
  225. - (AVPlayer *)audioReader {
  226. if (!_audioReader) {
  227. _audioReader = [AVPlayer playerWithURL:self.videoURL];
  228. _audioReader.actionAtItemEnd = AVPlayerActionAtItemEndNone;
  229. }
  230. return _audioReader;
  231. }
  232. - (BOOL)isReading {
  233. if (_videoReader) {
  234. return self.videoReader.isReading;
  235. }
  236. return NO;
  237. }
  238. - (BOOL)isProcessing {
  239. if (_videoProcessor) {
  240. return self.videoProcessor.reader.isReading || self.videoProcessor.writer.isWriting;
  241. }
  242. return NO;
  243. }
  244. - (FUVideoOrientation)videoOrientation {
  245. return self.videoReader.videoOrientation;
  246. }
  247. #pragma mark - Overriding
  248. - (FUAIModelType)necessaryAIModelTypes {
  249. return FUAIModelTypeFace;
  250. }
  251. - (CGFloat)downloadButtonBottomConstant {
  252. return FUHeightIncludeBottomSafeArea(10);
  253. }
  254. @end