diff --git a/ci/licenses_golden/licenses_flutter b/ci/licenses_golden/licenses_flutter index 245980e77e921..520eca83268b9 100755 --- a/ci/licenses_golden/licenses_flutter +++ b/ci/licenses_golden/licenses_flutter @@ -942,6 +942,8 @@ FILE: ../../../flutter/shell/platform/darwin/ios/framework/Headers/FlutterPlugin FILE: ../../../flutter/shell/platform/darwin/ios/framework/Headers/FlutterViewController.h FILE: ../../../flutter/shell/platform/darwin/ios/framework/Info.plist FILE: ../../../flutter/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate.mm +FILE: ../../../flutter/shell/platform/darwin/ios/framework/Source/FlutterAppDelegateTest.mm +FILE: ../../../flutter/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate_Test.h FILE: ../../../flutter/shell/platform/darwin/ios/framework/Source/FlutterBinaryMessengerRelay.h FILE: ../../../flutter/shell/platform/darwin/ios/framework/Source/FlutterBinaryMessengerRelay.mm FILE: ../../../flutter/shell/platform/darwin/ios/framework/Source/FlutterBinaryMessengerRelayTest.mm diff --git a/shell/platform/darwin/ios/BUILD.gn b/shell/platform/darwin/ios/BUILD.gn index e06275c3cfa59..99acf3960c825 100644 --- a/shell/platform/darwin/ios/BUILD.gn +++ b/shell/platform/darwin/ios/BUILD.gn @@ -208,6 +208,7 @@ shared_library("ios_test_flutter") { "//build/config:symbol_visibility_hidden", ] sources = [ + "framework/Source/FlutterAppDelegateTest.mm", "framework/Source/FlutterBinaryMessengerRelayTest.mm", "framework/Source/FlutterDartProjectTest.mm", "framework/Source/FlutterEngineTest.mm", diff --git a/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate.mm b/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate.mm index d1834c3a41d4a..1e574aec6343d 100644 --- a/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate.mm +++ b/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate.mm @@ -4,15 +4,21 @@ #import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterAppDelegate.h" -#include "flutter/fml/logging.h" +#import "flutter/fml/logging.h" #import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterPluginAppLifeCycleDelegate.h" #import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterViewController.h" +#import "flutter/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate_Test.h" +#import "flutter/shell/platform/darwin/ios/framework/Source/FlutterEngine_Internal.h" #import "flutter/shell/platform/darwin/ios/framework/Source/FlutterPluginAppLifeCycleDelegate_internal.h" static NSString* kUIBackgroundMode = @"UIBackgroundModes"; static NSString* kRemoteNotificationCapabitiliy = @"remote-notification"; static NSString* kBackgroundFetchCapatibility = @"fetch"; +@interface FlutterAppDelegate () +@property(nonatomic, copy) FlutterViewController* (^rootFlutterViewControllerGetter)(void); +@end + @implementation FlutterAppDelegate { FlutterPluginAppLifeCycleDelegate* _lifeCycleDelegate; } @@ -26,6 +32,7 @@ - (instancetype)init { - (void)dealloc { [_lifeCycleDelegate release]; + [_rootFlutterViewControllerGetter release]; [super dealloc]; } @@ -41,10 +48,13 @@ - (BOOL)application:(UIApplication*)application // Returns the key window's rootViewController, if it's a FlutterViewController. // Otherwise, returns nil. -+ (FlutterViewController*)rootFlutterViewController { - UIViewController* viewController = [UIApplication sharedApplication].keyWindow.rootViewController; - if ([viewController isKindOfClass:[FlutterViewController class]]) { - return (FlutterViewController*)viewController; +- (FlutterViewController*)rootFlutterViewController { + if (_rootFlutterViewControllerGetter != nil) { + return _rootFlutterViewControllerGetter(); + } + UIViewController* rootViewController = _window.rootViewController; + if ([rootViewController isKindOfClass:[FlutterViewController class]]) { + return (FlutterViewController*)rootViewController; } return nil; } @@ -121,10 +131,54 @@ - (void)userNotificationCenter:(UNUserNotificationCenter*)center } } +static BOOL IsDeepLinkingEnabled(NSDictionary* infoDictionary) { + NSNumber* isEnabled = [infoDictionary objectForKey:@"FlutterDeepLinkingEnabled"]; + if (isEnabled) { + return [isEnabled boolValue]; + } else { + return NO; + } +} + +- (BOOL)application:(UIApplication*)application + openURL:(NSURL*)url + options:(NSDictionary*)options + infoPlistGetter:(NSDictionary* (^)())infoPlistGetter { + if ([_lifeCycleDelegate application:application openURL:url options:options]) { + return YES; + } else if (!IsDeepLinkingEnabled(infoPlistGetter())) { + return NO; + } else { + FlutterViewController* flutterViewController = [self rootFlutterViewController]; + if (flutterViewController) { + [flutterViewController.engine + waitForFirstFrame:3.0 + callback:^(BOOL didTimeout) { + if (didTimeout) { + FML_LOG(ERROR) + << "Timeout waiting for the first frame when launching an URL."; + } else { + [flutterViewController.engine.navigationChannel invokeMethod:@"pushRoute" + arguments:url.path]; + } + }]; + return YES; + } else { + FML_LOG(ERROR) << "Attempting to open an URL without a Flutter RootViewController."; + return NO; + } + } +} + - (BOOL)application:(UIApplication*)application openURL:(NSURL*)url options:(NSDictionary*)options { - return [_lifeCycleDelegate application:application openURL:url options:options]; + return [self application:application + openURL:url + options:options + infoPlistGetter:^NSDictionary*() { + return [[NSBundle mainBundle] infoDictionary]; + }]; } - (BOOL)application:(UIApplication*)application handleOpenURL:(NSURL*)url { @@ -175,27 +229,25 @@ - (BOOL)application:(UIApplication*)application #pragma mark - FlutterPluginRegistry methods. All delegating to the rootViewController - (NSObject*)registrarForPlugin:(NSString*)pluginKey { - UIViewController* rootViewController = _window.rootViewController; - if ([rootViewController isKindOfClass:[FlutterViewController class]]) { - return - [[(FlutterViewController*)rootViewController pluginRegistry] registrarForPlugin:pluginKey]; + FlutterViewController* flutterRootViewController = [self rootFlutterViewController]; + if (flutterRootViewController) { + return [[flutterRootViewController pluginRegistry] registrarForPlugin:pluginKey]; } return nil; } - (BOOL)hasPlugin:(NSString*)pluginKey { - UIViewController* rootViewController = _window.rootViewController; - if ([rootViewController isKindOfClass:[FlutterViewController class]]) { - return [[(FlutterViewController*)rootViewController pluginRegistry] hasPlugin:pluginKey]; + FlutterViewController* flutterRootViewController = [self rootFlutterViewController]; + if (flutterRootViewController) { + return [[flutterRootViewController pluginRegistry] hasPlugin:pluginKey]; } return false; } - (NSObject*)valuePublishedByPlugin:(NSString*)pluginKey { - UIViewController* rootViewController = _window.rootViewController; - if ([rootViewController isKindOfClass:[FlutterViewController class]]) { - return [[(FlutterViewController*)rootViewController pluginRegistry] - valuePublishedByPlugin:pluginKey]; + FlutterViewController* flutterRootViewController = [self rootFlutterViewController]; + if (flutterRootViewController) { + return [[flutterRootViewController pluginRegistry] valuePublishedByPlugin:pluginKey]; } return nil; } diff --git a/shell/platform/darwin/ios/framework/Source/FlutterAppDelegateTest.mm b/shell/platform/darwin/ios/framework/Source/FlutterAppDelegateTest.mm new file mode 100644 index 0000000000000..5e3fd48371912 --- /dev/null +++ b/shell/platform/darwin/ios/framework/Source/FlutterAppDelegateTest.mm @@ -0,0 +1,44 @@ +// 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 +#import + +#import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterAppDelegate.h" +#import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterEngine.h" +#import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterViewController.h" +#import "flutter/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate_Test.h" +#import "flutter/shell/platform/darwin/ios/framework/Source/FlutterEngine_Test.h" + +FLUTTER_ASSERT_ARC + +@interface FlutterAppDelegateTest : XCTestCase +@end + +@implementation FlutterAppDelegateTest + +- (void)testLaunchUrl { + FlutterAppDelegate* appDelegate = [[FlutterAppDelegate alloc] init]; + FlutterViewController* viewController = OCMClassMock([FlutterViewController class]); + FlutterEngine* engine = OCMClassMock([FlutterEngine class]); + FlutterMethodChannel* navigationChannel = OCMClassMock([FlutterMethodChannel class]); + OCMStub([engine navigationChannel]).andReturn(navigationChannel); + OCMStub([viewController engine]).andReturn(engine); + OCMStub([engine waitForFirstFrame:3.0 callback:([OCMArg invokeBlockWithArgs:@(NO), nil])]); + appDelegate.rootFlutterViewControllerGetter = ^{ + return viewController; + }; + NSURL* url = [NSURL URLWithString:@"http://example.com"]; + NSDictionary* options = @{}; + BOOL result = [appDelegate application:[UIApplication sharedApplication] + openURL:url + options:options + infoPlistGetter:^NSDictionary*() { + return @{@"FlutterDeepLinkingEnabled" : @(YES)}; + }]; + XCTAssertTrue(result); + OCMVerify([navigationChannel invokeMethod:@"pushRoute" arguments:url.path]); +} + +@end diff --git a/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate_Test.h b/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate_Test.h new file mode 100644 index 0000000000000..a4394fde9fdac --- /dev/null +++ b/shell/platform/darwin/ios/framework/Source/FlutterAppDelegate_Test.h @@ -0,0 +1,15 @@ +// 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. + +@class FlutterViewController; + +@interface FlutterAppDelegate (Test) +@property(nonatomic, copy) FlutterViewController* (^rootFlutterViewControllerGetter)(void); + +- (BOOL)application:(UIApplication*)application + openURL:(NSURL*)url + options:(NSDictionary*)options + infoPlistGetter:(NSDictionary* (^)())infoPlistGetter; + +@end diff --git a/shell/platform/darwin/ios/framework/Source/FlutterEngine.mm b/shell/platform/darwin/ios/framework/Source/FlutterEngine.mm index d19c696f07ec3..9c26b05081ccc 100644 --- a/shell/platform/darwin/ios/framework/Source/FlutterEngine.mm +++ b/shell/platform/darwin/ios/framework/Source/FlutterEngine.mm @@ -914,6 +914,19 @@ - (void)onLocaleUpdated:(NSNotification*)notification { [self.localizationChannel invokeMethod:@"setLocale" arguments:localeData]; } +- (void)waitForFirstFrame:(NSTimeInterval)timeout + callback:(void (^_Nonnull)(BOOL didTimeout))callback { + dispatch_queue_t queue = dispatch_get_global_queue(QOS_CLASS_BACKGROUND, 0); + dispatch_async(queue, ^{ + fml::TimeDelta waitTime = fml::TimeDelta::FromMilliseconds(timeout * 1000); + BOOL didTimeout = + self.shell.WaitForFirstFrame(waitTime).code() == fml::StatusCode::kDeadlineExceeded; + dispatch_async(dispatch_get_main_queue(), ^{ + callback(didTimeout); + }); + }); +} + @end @implementation FlutterEngineRegistrar { diff --git a/shell/platform/darwin/ios/framework/Source/FlutterEngineTest.mm b/shell/platform/darwin/ios/framework/Source/FlutterEngineTest.mm index e51bb06162a77..3684ea5374a60 100644 --- a/shell/platform/darwin/ios/framework/Source/FlutterEngineTest.mm +++ b/shell/platform/darwin/ios/framework/Source/FlutterEngineTest.mm @@ -120,4 +120,18 @@ - (void)testPlatformViewsControllerRenderingSoftware { XCTAssertEqual(renderingApi, flutter::IOSRenderingAPI::kSoftware); } + +- (void)testWaitForFirstFrameTimeout { + FlutterEngine* engine = [[FlutterEngine alloc] initWithName:@"foobar"]; + [engine run]; + XCTestExpectation* timeoutFirstFrame = [self expectationWithDescription:@"timeoutFirstFrame"]; + [engine waitForFirstFrame:0.1 + callback:^(BOOL didTimeout) { + if (timeoutFirstFrame) { + [timeoutFirstFrame fulfill]; + } + }]; + [self waitForExpectationsWithTimeout:1 handler:nil]; +} + @end diff --git a/shell/platform/darwin/ios/framework/Source/FlutterEngine_Internal.h b/shell/platform/darwin/ios/framework/Source/FlutterEngine_Internal.h index 739138dcf0099..85a82f3891530 100644 --- a/shell/platform/darwin/ios/framework/Source/FlutterEngine_Internal.h +++ b/shell/platform/darwin/ios/framework/Source/FlutterEngine_Internal.h @@ -48,6 +48,7 @@ - (void)notifyLowMemory; - (flutter::PlatformViewIOS*)iosPlatformView; +- (void)waitForFirstFrame:(NSTimeInterval)timeout callback:(void (^)(BOOL didTimeout))callback; @end #endif // FLUTTER_SHELL_PLATFORM_DARWIN_IOS_FRAMEWORK_SOURCE_FLUTTERENGINE_INTERNAL_H_ diff --git a/shell/platform/darwin/ios/framework/Source/FlutterEngine_Test.h b/shell/platform/darwin/ios/framework/Source/FlutterEngine_Test.h index 82516b1e77389..50db832916ea4 100644 --- a/shell/platform/darwin/ios/framework/Source/FlutterEngine_Test.h +++ b/shell/platform/darwin/ios/framework/Source/FlutterEngine_Test.h @@ -5,8 +5,11 @@ #import "flutter/shell/platform/darwin/ios/framework/Headers/FlutterEngine.h" #import "flutter/shell/platform/darwin/ios/rendering_api_selection.h" +@class FlutterBinaryMessengerRelay; + // Category to add test-only visibility. @interface FlutterEngine (Test) - (void)setBinaryMessenger:(FlutterBinaryMessengerRelay*)binaryMessenger; - (flutter::IOSRenderingAPI)platformViewsRenderingAPI; +- (void)waitForFirstFrame:(NSTimeInterval)timeout callback:(void (^)(BOOL didTimeout))callback; @end