| // Copyright 2013 The Flutter Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| @import AVFoundation; |
| @import video_player_avfoundation; |
| @import XCTest; |
| |
| #import <OCMock/OCMock.h> |
| #import <video_player_avfoundation/AVAssetTrackUtils.h> |
| #import <video_player_avfoundation/FVPVideoPlayerPlugin_Test.h> |
| |
| // TODO(stuartmorgan): Convert to using mock registrars instead. |
| NSObject<FlutterPluginRegistry> *GetPluginRegistry(void) { |
| #if TARGET_OS_IOS |
| return (NSObject<FlutterPluginRegistry> *)[[UIApplication sharedApplication] delegate]; |
| #else |
| return (FlutterViewController *)NSApplication.sharedApplication.windows[0].contentViewController; |
| #endif |
| } |
| |
| #if TARGET_OS_IOS |
| @interface FakeAVAssetTrack : AVAssetTrack |
| @property(readonly, nonatomic) CGAffineTransform preferredTransform; |
| @property(readonly, nonatomic) CGSize naturalSize; |
| @property(readonly, nonatomic) UIImageOrientation orientation; |
| - (instancetype)initWithOrientation:(UIImageOrientation)orientation; |
| @end |
| |
| @implementation FakeAVAssetTrack |
| |
| - (instancetype)initWithOrientation:(UIImageOrientation)orientation { |
| _orientation = orientation; |
| _naturalSize = CGSizeMake(800, 600); |
| return self; |
| } |
| |
| - (CGAffineTransform)preferredTransform { |
| switch (_orientation) { |
| case UIImageOrientationUp: |
| return CGAffineTransformMake(1, 0, 0, 1, 0, 0); |
| case UIImageOrientationDown: |
| return CGAffineTransformMake(-1, 0, 0, -1, 0, 0); |
| case UIImageOrientationLeft: |
| return CGAffineTransformMake(0, -1, 1, 0, 0, 0); |
| case UIImageOrientationRight: |
| return CGAffineTransformMake(0, 1, -1, 0, 0, 0); |
| case UIImageOrientationUpMirrored: |
| return CGAffineTransformMake(-1, 0, 0, 1, 0, 0); |
| case UIImageOrientationDownMirrored: |
| return CGAffineTransformMake(1, 0, 0, -1, 0, 0); |
| case UIImageOrientationLeftMirrored: |
| return CGAffineTransformMake(0, -1, -1, 0, 0, 0); |
| case UIImageOrientationRightMirrored: |
| return CGAffineTransformMake(0, 1, 1, 0, 0, 0); |
| } |
| } |
| |
| @end |
| #endif |
| |
| @interface VideoPlayerTests : XCTestCase |
| @end |
| |
| @interface StubAVPlayer : AVPlayer |
| @property(readonly, nonatomic) NSNumber *beforeTolerance; |
| @property(readonly, nonatomic) NSNumber *afterTolerance; |
| @property(readonly, assign) CMTime lastSeekTime; |
| @end |
| |
| @implementation StubAVPlayer |
| |
| - (void)seekToTime:(CMTime)time |
| toleranceBefore:(CMTime)toleranceBefore |
| toleranceAfter:(CMTime)toleranceAfter |
| completionHandler:(void (^)(BOOL finished))completionHandler { |
| _beforeTolerance = [NSNumber numberWithLong:toleranceBefore.value]; |
| _afterTolerance = [NSNumber numberWithLong:toleranceAfter.value]; |
| _lastSeekTime = time; |
| [super seekToTime:time |
| toleranceBefore:toleranceBefore |
| toleranceAfter:toleranceAfter |
| completionHandler:completionHandler]; |
| } |
| |
| @end |
| |
| @interface StubFVPAVFactory : NSObject <FVPAVFactory> |
| |
| @property(nonatomic, strong) StubAVPlayer *stubAVPlayer; |
| @property(nonatomic, strong) AVPlayerItemVideoOutput *output; |
| |
| - (instancetype)initWithPlayer:(StubAVPlayer *)stubAVPlayer |
| output:(AVPlayerItemVideoOutput *)output; |
| |
| @end |
| |
| @implementation StubFVPAVFactory |
| |
| // Creates a factory that returns the given items. Any items that are nil will instead return |
| // a real object just as the non-test implementation would. |
| - (instancetype)initWithPlayer:(StubAVPlayer *)stubAVPlayer |
| output:(AVPlayerItemVideoOutput *)output { |
| self = [super init]; |
| _stubAVPlayer = stubAVPlayer; |
| _output = output; |
| return self; |
| } |
| |
| - (AVPlayer *)playerWithPlayerItem:(AVPlayerItem *)playerItem { |
| return _stubAVPlayer ?: [AVPlayer playerWithPlayerItem:playerItem]; |
| } |
| |
| - (AVPlayerItemVideoOutput *)videoOutputWithPixelBufferAttributes: |
| (NSDictionary<NSString *, id> *)attributes { |
| return _output ?: [[AVPlayerItemVideoOutput alloc] initWithPixelBufferAttributes:attributes]; |
| } |
| |
| @end |
| |
| #pragma mark - |
| |
| /** Test implementation of FVPDisplayLinkFactory that returns a provided display link nstance. */ |
| @interface StubFVPDisplayLinkFactory : NSObject <FVPDisplayLinkFactory> |
| |
| /** This display link to return. */ |
| @property(nonatomic, strong) FVPDisplayLink *displayLink; |
| |
| - (instancetype)initWithDisplayLink:(FVPDisplayLink *)displayLink; |
| |
| @end |
| |
| @implementation StubFVPDisplayLinkFactory |
| - (instancetype)initWithDisplayLink:(FVPDisplayLink *)displayLink { |
| self = [super init]; |
| _displayLink = displayLink; |
| return self; |
| } |
| - (FVPDisplayLink *)displayLinkWithRegistrar:(id<FlutterPluginRegistrar>)registrar |
| callback:(void (^)(void))callback { |
| return self.displayLink; |
| } |
| |
| @end |
| |
| /** Non-test implementation of the diplay link factory. */ |
| @interface FVPDefaultDisplayLinkFactory : NSObject <FVPDisplayLinkFactory> |
| @end |
| |
| @implementation FVPDefaultDisplayLinkFactory |
| - (FVPDisplayLink *)displayLinkWithRegistrar:(id<FlutterPluginRegistrar>)registrar |
| callback:(void (^)(void))callback { |
| return [[FVPDisplayLink alloc] initWithRegistrar:registrar callback:callback]; |
| } |
| |
| @end |
| |
| #pragma mark - |
| |
| @implementation VideoPlayerTests |
| |
| - (void)testBlankVideoBugWithEncryptedVideoStreamAndInvertedAspectRatioBugForSomeVideoStream { |
| // This is to fix 2 bugs: 1. blank video for encrypted video streams on iOS 16 |
| // (https://github.com/flutter/flutter/issues/111457) and 2. swapped width and height for some |
| // video streams (not just iOS 16). (https://github.com/flutter/flutter/issues/109116). An |
| // invisible AVPlayerLayer is used to overwrite the protection of pixel buffers in those streams |
| // for issue #1, and restore the correct width and height for issue #2. |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"testPlayerLayerWorkaround"]; |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| [[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| FlutterError *error; |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&error]; |
| XCTAssertNil(error); |
| XCTAssertNotNil(textureMessage); |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureMessage.textureId)]; |
| XCTAssertNotNil(player); |
| |
| XCTAssertNotNil(player.playerLayer, @"AVPlayerLayer should be present."); |
| XCTAssertNotNil(player.playerLayer.superlayer, @"AVPlayerLayer should be added on screen."); |
| } |
| |
| - (void)testSeekToWhilePausedStartsDisplayLinkTemporarily { |
| NSObject<FlutterTextureRegistry> *mockTextureRegistry = |
| OCMProtocolMock(@protocol(FlutterTextureRegistry)); |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"SeekToWhilePausedStartsDisplayLinkTemporarily"]; |
| NSObject<FlutterPluginRegistrar> *partialRegistrar = OCMPartialMock(registrar); |
| OCMStub([partialRegistrar textures]).andReturn(mockTextureRegistry); |
| FVPDisplayLink *mockDisplayLink = |
| OCMPartialMock([[FVPDisplayLink alloc] initWithRegistrar:registrar |
| callback:^(){ |
| }]); |
| StubFVPDisplayLinkFactory *stubDisplayLinkFactory = |
| [[StubFVPDisplayLinkFactory alloc] initWithDisplayLink:mockDisplayLink]; |
| AVPlayerItemVideoOutput *mockVideoOutput = OCMPartialMock([[AVPlayerItemVideoOutput alloc] init]); |
| FVPVideoPlayerPlugin *videoPlayerPlugin = [[FVPVideoPlayerPlugin alloc] |
| initWithAVFactory:[[StubFVPAVFactory alloc] initWithPlayer:nil output:mockVideoOutput] |
| displayLinkFactory:stubDisplayLinkFactory |
| registrar:partialRegistrar]; |
| |
| FlutterError *initalizationError; |
| [videoPlayerPlugin initialize:&initalizationError]; |
| XCTAssertNil(initalizationError); |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/hls/bee.m3u8" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FlutterError *createError; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&createError]; |
| NSInteger textureId = textureMessage.textureId; |
| |
| // Ensure that the video playback is paused before seeking. |
| FlutterError *pauseError; |
| [videoPlayerPlugin pause:textureMessage error:&pauseError]; |
| |
| XCTestExpectation *initializedExpectation = [self expectationWithDescription:@"seekTo completes"]; |
| FVPPositionMessage *message = [FVPPositionMessage makeWithTextureId:textureId position:1234]; |
| [videoPlayerPlugin seekTo:message |
| completion:^(FlutterError *_Nullable error) { |
| [initializedExpectation fulfill]; |
| }]; |
| [self waitForExpectationsWithTimeout:30.0 handler:nil]; |
| |
| // Seeking to a new position should start the display link temporarily. |
| OCMVerify([mockDisplayLink setRunning:YES]); |
| |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureId)]; |
| XCTAssertEqual([player position], 1234); |
| |
| // Simulate a buffer being available. |
| OCMStub([mockVideoOutput hasNewPixelBufferForItemTime:kCMTimeZero]) |
| .ignoringNonObjectArgs() |
| .andReturn(YES); |
| // Any non-zero value is fine here since it won't actually be used, just NULL-checked. |
| CVPixelBufferRef fakeBufferRef = (CVPixelBufferRef)1; |
| OCMStub([mockVideoOutput copyPixelBufferForItemTime:kCMTimeZero itemTimeForDisplay:NULL]) |
| .ignoringNonObjectArgs() |
| .andReturn(fakeBufferRef); |
| // Simulate a callback from the engine to request a new frame. |
| [player copyPixelBuffer]; |
| // Since a frame was found, and the video is paused, the display link should be paused again. |
| OCMVerify([mockDisplayLink setRunning:NO]); |
| } |
| |
| - (void)testSeekToWhilePlayingDoesNotStopDisplayLink { |
| NSObject<FlutterTextureRegistry> *mockTextureRegistry = |
| OCMProtocolMock(@protocol(FlutterTextureRegistry)); |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"SeekToWhilePlayingDoesNotStopDisplayLink"]; |
| NSObject<FlutterPluginRegistrar> *partialRegistrar = OCMPartialMock(registrar); |
| OCMStub([partialRegistrar textures]).andReturn(mockTextureRegistry); |
| FVPDisplayLink *mockDisplayLink = |
| OCMPartialMock([[FVPDisplayLink alloc] initWithRegistrar:registrar |
| callback:^(){ |
| }]); |
| StubFVPDisplayLinkFactory *stubDisplayLinkFactory = |
| [[StubFVPDisplayLinkFactory alloc] initWithDisplayLink:mockDisplayLink]; |
| AVPlayerItemVideoOutput *mockVideoOutput = OCMPartialMock([[AVPlayerItemVideoOutput alloc] init]); |
| FVPVideoPlayerPlugin *videoPlayerPlugin = [[FVPVideoPlayerPlugin alloc] |
| initWithAVFactory:[[StubFVPAVFactory alloc] initWithPlayer:nil output:mockVideoOutput] |
| displayLinkFactory:stubDisplayLinkFactory |
| registrar:partialRegistrar]; |
| |
| FlutterError *initalizationError; |
| [videoPlayerPlugin initialize:&initalizationError]; |
| XCTAssertNil(initalizationError); |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/hls/bee.m3u8" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FlutterError *createError; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&createError]; |
| NSInteger textureId = textureMessage.textureId; |
| |
| // Ensure that the video is playing before seeking. |
| FlutterError *pauseError; |
| [videoPlayerPlugin play:textureMessage error:&pauseError]; |
| |
| XCTestExpectation *initializedExpectation = [self expectationWithDescription:@"seekTo completes"]; |
| FVPPositionMessage *message = [FVPPositionMessage makeWithTextureId:textureId position:1234]; |
| [videoPlayerPlugin seekTo:message |
| completion:^(FlutterError *_Nullable error) { |
| [initializedExpectation fulfill]; |
| }]; |
| [self waitForExpectationsWithTimeout:30.0 handler:nil]; |
| OCMVerify([mockDisplayLink setRunning:YES]); |
| |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureId)]; |
| XCTAssertEqual([player position], 1234); |
| |
| // Simulate a buffer being available. |
| OCMStub([mockVideoOutput hasNewPixelBufferForItemTime:kCMTimeZero]) |
| .ignoringNonObjectArgs() |
| .andReturn(YES); |
| // Any non-zero value is fine here since it won't actually be used, just NULL-checked. |
| CVPixelBufferRef fakeBufferRef = (CVPixelBufferRef)1; |
| OCMStub([mockVideoOutput copyPixelBufferForItemTime:kCMTimeZero itemTimeForDisplay:NULL]) |
| .ignoringNonObjectArgs() |
| .andReturn(fakeBufferRef); |
| // Simulate a callback from the engine to request a new frame. |
| [player copyPixelBuffer]; |
| // Since the video was playing, the display link should not be paused after getting a buffer. |
| OCMVerify(never(), [mockDisplayLink setRunning:NO]); |
| } |
| |
| - (void)testDeregistersFromPlayer { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"testDeregistersFromPlayer"]; |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| FlutterError *error; |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&error]; |
| XCTAssertNil(error); |
| XCTAssertNotNil(textureMessage); |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureMessage.textureId)]; |
| XCTAssertNotNil(player); |
| AVPlayer *avPlayer = player.player; |
| |
| [videoPlayerPlugin dispose:textureMessage error:&error]; |
| XCTAssertEqual(videoPlayerPlugin.playersByTextureId.count, 0); |
| XCTAssertNil(error); |
| |
| [self keyValueObservingExpectationForObject:avPlayer keyPath:@"currentItem" expectedValue:nil]; |
| [self waitForExpectationsWithTimeout:30.0 handler:nil]; |
| } |
| |
| - (void)testBufferingStateFromPlayer { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"testLiveStreamBufferEndFromPlayer"]; |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| FlutterError *error; |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&error]; |
| XCTAssertNil(error); |
| XCTAssertNotNil(textureMessage); |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureMessage.textureId)]; |
| XCTAssertNotNil(player); |
| AVPlayer *avPlayer = player.player; |
| [avPlayer play]; |
| |
| [player onListenWithArguments:nil |
| eventSink:^(NSDictionary<NSString *, id> *event) { |
| if ([event[@"event"] isEqualToString:@"bufferingEnd"]) { |
| XCTAssertTrue(avPlayer.currentItem.isPlaybackLikelyToKeepUp); |
| } |
| |
| if ([event[@"event"] isEqualToString:@"bufferingStart"]) { |
| XCTAssertFalse(avPlayer.currentItem.isPlaybackLikelyToKeepUp); |
| } |
| }]; |
| XCTestExpectation *bufferingStateExpectation = |
| [self expectationWithDescription:@"bufferingState"]; |
| NSTimeInterval timeout = 10; |
| dispatch_time_t delay = dispatch_time(DISPATCH_TIME_NOW, timeout * NSEC_PER_SEC); |
| dispatch_after(delay, dispatch_get_main_queue(), ^{ |
| [bufferingStateExpectation fulfill]; |
| }); |
| [self waitForExpectationsWithTimeout:timeout + 1 handler:nil]; |
| } |
| |
| - (void)testVideoControls { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"TestVideoControls"]; |
| |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| NSDictionary<NSString *, id> *videoInitialization = |
| [self testPlugin:videoPlayerPlugin |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4"]; |
| XCTAssertEqualObjects(videoInitialization[@"height"], @720); |
| XCTAssertEqualObjects(videoInitialization[@"width"], @1280); |
| XCTAssertEqualWithAccuracy([videoInitialization[@"duration"] intValue], 4000, 200); |
| } |
| |
| - (void)testAudioControls { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"TestAudioControls"]; |
| |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| NSDictionary<NSString *, id> *audioInitialization = |
| [self testPlugin:videoPlayerPlugin |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/audio/rooster.mp3"]; |
| XCTAssertEqualObjects(audioInitialization[@"height"], @0); |
| XCTAssertEqualObjects(audioInitialization[@"width"], @0); |
| // Perfect precision not guaranteed. |
| XCTAssertEqualWithAccuracy([audioInitialization[@"duration"] intValue], 5400, 200); |
| } |
| |
| - (void)testHLSControls { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"TestHLSControls"]; |
| |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| NSDictionary<NSString *, id> *videoInitialization = |
| [self testPlugin:videoPlayerPlugin |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/hls/bee.m3u8"]; |
| XCTAssertEqualObjects(videoInitialization[@"height"], @720); |
| XCTAssertEqualObjects(videoInitialization[@"width"], @1280); |
| XCTAssertEqualWithAccuracy([videoInitialization[@"duration"] intValue], 4000, 200); |
| } |
| |
| #if TARGET_OS_IOS |
| - (void)testTransformFix { |
| [self validateTransformFixForOrientation:UIImageOrientationUp]; |
| [self validateTransformFixForOrientation:UIImageOrientationDown]; |
| [self validateTransformFixForOrientation:UIImageOrientationLeft]; |
| [self validateTransformFixForOrientation:UIImageOrientationRight]; |
| [self validateTransformFixForOrientation:UIImageOrientationUpMirrored]; |
| [self validateTransformFixForOrientation:UIImageOrientationDownMirrored]; |
| [self validateTransformFixForOrientation:UIImageOrientationLeftMirrored]; |
| [self validateTransformFixForOrientation:UIImageOrientationRightMirrored]; |
| } |
| #endif |
| |
| - (void)testSeekToleranceWhenNotSeekingToEnd { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"TestSeekTolerance"]; |
| |
| StubAVPlayer *stubAVPlayer = [[StubAVPlayer alloc] init]; |
| StubFVPAVFactory *stubAVFactory = [[StubFVPAVFactory alloc] initWithPlayer:stubAVPlayer |
| output:nil]; |
| FVPVideoPlayerPlugin *pluginWithMockAVPlayer = |
| [[FVPVideoPlayerPlugin alloc] initWithAVFactory:stubAVFactory |
| displayLinkFactory:nil |
| registrar:registrar]; |
| |
| FlutterError *initializationError; |
| [pluginWithMockAVPlayer initialize:&initializationError]; |
| XCTAssertNil(initializationError); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FlutterError *createError; |
| FVPTextureMessage *textureMessage = [pluginWithMockAVPlayer create:create error:&createError]; |
| NSInteger textureId = textureMessage.textureId; |
| |
| XCTestExpectation *initializedExpectation = |
| [self expectationWithDescription:@"seekTo has zero tolerance when seeking not to end"]; |
| FVPPositionMessage *message = [FVPPositionMessage makeWithTextureId:textureId position:1234]; |
| [pluginWithMockAVPlayer seekTo:message |
| completion:^(FlutterError *_Nullable error) { |
| [initializedExpectation fulfill]; |
| }]; |
| |
| [self waitForExpectationsWithTimeout:30.0 handler:nil]; |
| XCTAssertEqual([stubAVPlayer.beforeTolerance intValue], 0); |
| XCTAssertEqual([stubAVPlayer.afterTolerance intValue], 0); |
| } |
| |
| - (void)testSeekToleranceWhenSeekingToEnd { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"TestSeekToEndTolerance"]; |
| |
| StubAVPlayer *stubAVPlayer = [[StubAVPlayer alloc] init]; |
| StubFVPAVFactory *stubAVFactory = [[StubFVPAVFactory alloc] initWithPlayer:stubAVPlayer |
| output:nil]; |
| FVPVideoPlayerPlugin *pluginWithMockAVPlayer = |
| [[FVPVideoPlayerPlugin alloc] initWithAVFactory:stubAVFactory |
| displayLinkFactory:nil |
| registrar:registrar]; |
| |
| FlutterError *initializationError; |
| [pluginWithMockAVPlayer initialize:&initializationError]; |
| XCTAssertNil(initializationError); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FlutterError *createError; |
| FVPTextureMessage *textureMessage = [pluginWithMockAVPlayer create:create error:&createError]; |
| NSInteger textureId = textureMessage.textureId; |
| |
| XCTestExpectation *initializedExpectation = |
| [self expectationWithDescription:@"seekTo has non-zero tolerance when seeking to end"]; |
| // The duration of this video is "0" due to the non standard initiliatazion process. |
| FVPPositionMessage *message = [FVPPositionMessage makeWithTextureId:textureId position:0]; |
| [pluginWithMockAVPlayer seekTo:message |
| completion:^(FlutterError *_Nullable error) { |
| [initializedExpectation fulfill]; |
| }]; |
| [self waitForExpectationsWithTimeout:30.0 handler:nil]; |
| XCTAssertGreaterThan([stubAVPlayer.beforeTolerance intValue], 0); |
| XCTAssertGreaterThan([stubAVPlayer.afterTolerance intValue], 0); |
| } |
| |
| - (NSDictionary<NSString *, id> *)testPlugin:(FVPVideoPlayerPlugin *)videoPlayerPlugin |
| uri:(NSString *)uri { |
| FlutterError *error; |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| |
| FVPCreateMessage *create = [FVPCreateMessage makeWithAsset:nil |
| uri:uri |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&error]; |
| |
| NSInteger textureId = textureMessage.textureId; |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureId)]; |
| XCTAssertNotNil(player); |
| |
| XCTestExpectation *initializedExpectation = [self expectationWithDescription:@"initialized"]; |
| __block NSDictionary<NSString *, id> *initializationEvent; |
| [player onListenWithArguments:nil |
| eventSink:^(NSDictionary<NSString *, id> *event) { |
| if ([event[@"event"] isEqualToString:@"initialized"]) { |
| initializationEvent = event; |
| XCTAssertEqual(event.count, 4); |
| [initializedExpectation fulfill]; |
| } |
| }]; |
| [self waitForExpectationsWithTimeout:30.0 handler:nil]; |
| |
| // Starts paused. |
| AVPlayer *avPlayer = player.player; |
| XCTAssertEqual(avPlayer.rate, 0); |
| XCTAssertEqual(avPlayer.volume, 1); |
| XCTAssertEqual(avPlayer.timeControlStatus, AVPlayerTimeControlStatusPaused); |
| |
| // Change playback speed. |
| FVPPlaybackSpeedMessage *playback = [FVPPlaybackSpeedMessage makeWithTextureId:textureId speed:2]; |
| [videoPlayerPlugin setPlaybackSpeed:playback error:&error]; |
| XCTAssertNil(error); |
| XCTAssertEqual(avPlayer.rate, 2); |
| XCTAssertEqual(avPlayer.timeControlStatus, AVPlayerTimeControlStatusWaitingToPlayAtSpecifiedRate); |
| |
| // Volume |
| FVPVolumeMessage *volume = [FVPVolumeMessage makeWithTextureId:textureId volume:0.1]; |
| [videoPlayerPlugin setVolume:volume error:&error]; |
| XCTAssertNil(error); |
| XCTAssertEqual(avPlayer.volume, 0.1f); |
| |
| [player onCancelWithArguments:nil]; |
| |
| return initializationEvent; |
| } |
| |
| // Checks whether [AVPlayer rate] KVO observations are correctly detached. |
| // - https://github.com/flutter/flutter/issues/124937 |
| // |
| // Failing to de-register results in a crash in [AVPlayer willChangeValueForKey:]. |
| - (void)testDoesNotCrashOnRateObservationAfterDisposal { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"testDoesNotCrashOnRateObservationAfterDisposal"]; |
| |
| AVPlayer *avPlayer = nil; |
| __weak FVPVideoPlayer *weakPlayer = nil; |
| |
| // Autoreleasepool is needed to simulate conditions of FVPVideoPlayer deallocation. |
| @autoreleasepool { |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| FlutterError *error; |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&error]; |
| XCTAssertNil(error); |
| XCTAssertNotNil(textureMessage); |
| |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureMessage.textureId)]; |
| XCTAssertNotNil(player); |
| weakPlayer = player; |
| avPlayer = player.player; |
| |
| [videoPlayerPlugin dispose:textureMessage error:&error]; |
| XCTAssertNil(error); |
| } |
| |
| // [FVPVideoPlayerPlugin dispose:error:] selector is dispatching the [FVPVideoPlayer dispose] call |
| // with a 1-second delay keeping a strong reference to the player. The polling ensures the player |
| // was truly deallocated. |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Warc-repeated-use-of-weak" |
| [self expectationForPredicate:[NSPredicate predicateWithFormat:@"self != nil"] |
| evaluatedWithObject:weakPlayer |
| handler:nil]; |
| #pragma clang diagnostic pop |
| [self waitForExpectationsWithTimeout:10.0 handler:nil]; |
| |
| [avPlayer willChangeValueForKey:@"rate"]; // No assertions needed. Lack of crash is a success. |
| } |
| |
| // During the hot reload: |
| // 1. `[FVPVideoPlayer onTextureUnregistered:]` gets called. |
| // 2. `[FVPVideoPlayerPlugin initialize:]` gets called. |
| // |
| // Both of these methods dispatch [FVPVideoPlayer dispose] on the main thread |
| // leading to a possible crash when de-registering observers twice. |
| - (void)testHotReloadDoesNotCrash { |
| NSObject<FlutterPluginRegistrar> *registrar = |
| [GetPluginRegistry() registrarForPlugin:@"testHotReloadDoesNotCrash"]; |
| |
| __weak FVPVideoPlayer *weakPlayer = nil; |
| |
| // Autoreleasepool is needed to simulate conditions of FVPVideoPlayer deallocation. |
| @autoreleasepool { |
| FVPVideoPlayerPlugin *videoPlayerPlugin = |
| (FVPVideoPlayerPlugin *)[[FVPVideoPlayerPlugin alloc] initWithRegistrar:registrar]; |
| |
| FlutterError *error; |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| |
| FVPCreateMessage *create = [FVPCreateMessage |
| makeWithAsset:nil |
| uri:@"https://flutter.github.io/assets-for-api-docs/assets/videos/bee.mp4" |
| packageName:nil |
| formatHint:nil |
| httpHeaders:@{}]; |
| FVPTextureMessage *textureMessage = [videoPlayerPlugin create:create error:&error]; |
| XCTAssertNil(error); |
| XCTAssertNotNil(textureMessage); |
| |
| FVPVideoPlayer *player = videoPlayerPlugin.playersByTextureId[@(textureMessage.textureId)]; |
| XCTAssertNotNil(player); |
| weakPlayer = player; |
| |
| [player onTextureUnregistered:nil]; |
| XCTAssertNil(error); |
| |
| [videoPlayerPlugin initialize:&error]; |
| XCTAssertNil(error); |
| } |
| |
| // [FVPVideoPlayerPlugin dispose:error:] selector is dispatching the [FVPVideoPlayer dispose] call |
| // with a 1-second delay keeping a strong reference to the player. The polling ensures the player |
| // was truly deallocated. |
| #pragma clang diagnostic push |
| #pragma clang diagnostic ignored "-Warc-repeated-use-of-weak" |
| [self expectationForPredicate:[NSPredicate predicateWithFormat:@"self != nil"] |
| evaluatedWithObject:weakPlayer |
| handler:nil]; |
| #pragma clang diagnostic pop |
| [self waitForExpectationsWithTimeout:10.0 |
| handler:nil]; // No assertions needed. Lack of crash is a success. |
| } |
| |
| #if TARGET_OS_IOS |
| - (void)validateTransformFixForOrientation:(UIImageOrientation)orientation { |
| AVAssetTrack *track = [[FakeAVAssetTrack alloc] initWithOrientation:orientation]; |
| CGAffineTransform t = FVPGetStandardizedTransformForTrack(track); |
| CGSize size = track.naturalSize; |
| CGFloat expectX, expectY; |
| switch (orientation) { |
| case UIImageOrientationUp: |
| expectX = 0; |
| expectY = 0; |
| break; |
| case UIImageOrientationDown: |
| expectX = size.width; |
| expectY = size.height; |
| break; |
| case UIImageOrientationLeft: |
| expectX = 0; |
| expectY = size.width; |
| break; |
| case UIImageOrientationRight: |
| expectX = size.height; |
| expectY = 0; |
| break; |
| case UIImageOrientationUpMirrored: |
| expectX = size.width; |
| expectY = 0; |
| break; |
| case UIImageOrientationDownMirrored: |
| expectX = 0; |
| expectY = size.height; |
| break; |
| case UIImageOrientationLeftMirrored: |
| expectX = size.height; |
| expectY = size.width; |
| break; |
| case UIImageOrientationRightMirrored: |
| expectX = 0; |
| expectY = 0; |
| break; |
| } |
| XCTAssertEqual(t.tx, expectX); |
| XCTAssertEqual(t.ty, expectY); |
| } |
| #endif |
| |
| @end |