5 #include "common/settings.h"
6 #define FML_USED_ON_EMBEDDER
12 #include "flutter/common/constants.h"
13 #include "flutter/fml/message_loop.h"
14 #include "flutter/fml/platform/darwin/platform_version.h"
15 #include "flutter/fml/trace_event.h"
16 #include "flutter/runtime/ptrace_check.h"
17 #include "flutter/shell/common/engine.h"
18 #include "flutter/shell/common/platform_view.h"
19 #include "flutter/shell/common/shell.h"
20 #include "flutter/shell/common/switches.h"
21 #include "flutter/shell/common/thread_host.h"
22 #include "flutter/shell/common/variable_refresh_rate_display.h"
42 #include "flutter/shell/profiling/sampling_profiler.h"
50 fml::Thread::SetCurrentThreadName(config);
53 switch (config.priority) {
54 case fml::Thread::ThreadPriority::kBackground: {
55 pthread_set_qos_class_self_np(QOS_CLASS_BACKGROUND, 0);
56 [[NSThread currentThread] setThreadPriority:0];
59 case fml::Thread::ThreadPriority::kNormal: {
60 pthread_set_qos_class_self_np(QOS_CLASS_DEFAULT, 0);
61 [[NSThread currentThread] setThreadPriority:0.5];
64 case fml::Thread::ThreadPriority::kRaster:
65 case fml::Thread::ThreadPriority::kDisplay: {
66 pthread_set_qos_class_self_np(QOS_CLASS_USER_INTERACTIVE, 0);
67 [[NSThread currentThread] setThreadPriority:1.0];
70 pthread_t thread = pthread_self();
71 if (!pthread_getschedparam(thread, &policy, ¶m)) {
72 param.sched_priority = 50;
73 pthread_setschedparam(thread, policy, ¶m);
80 #pragma mark - Public exported constants
85 #pragma mark - Internal constants
101 #pragma mark - Properties
104 @property(nonatomic, readonly, copy) NSString* labelPrefix;
105 @property(nonatomic, readonly, assign) BOOL allowHeadlessExecution;
106 @property(nonatomic, readonly, assign) BOOL restorationEnabled;
112 @property(nonatomic, readonly) NSMutableDictionary* pluginPublications;
113 @property(nonatomic, readonly) NSMutableDictionary<NSString*, FlutterEngineRegistrar*>* registrars;
115 @property(nonatomic, readwrite, copy) NSString*
isolateId;
116 @property(nonatomic, copy) NSString* initialRoute;
117 @property(nonatomic, strong) id<NSObject> flutterViewControllerWillDeallocObserver;
119 @property(nonatomic, assign) int64_t nextTextureId;
121 #pragma mark - Channel properties
143 #pragma mark - Embedder API properties
152 std::shared_ptr<flutter::ThreadHost> _threadHost;
153 std::unique_ptr<flutter::Shell>
_shell;
163 - (int64_t)engineIdentifier {
164 return reinterpret_cast<int64_t
>((__bridge
void*)
self);
167 - (instancetype)init {
171 - (instancetype)initWithName:(NSString*)labelPrefix {
175 - (instancetype)initWithName:(NSString*)labelPrefix project:(
FlutterDartProject*)project {
176 return [
self initWithName:labelPrefix project:project allowHeadlessExecution:YES];
179 - (instancetype)initWithName:(NSString*)labelPrefix
181 allowHeadlessExecution:(BOOL)allowHeadlessExecution {
182 return [
self initWithName:labelPrefix
184 allowHeadlessExecution:allowHeadlessExecution
185 restorationEnabled:NO];
188 - (instancetype)initWithName:(NSString*)labelPrefix
190 allowHeadlessExecution:(BOOL)allowHeadlessExecution
191 restorationEnabled:(BOOL)restorationEnabled {
193 NSAssert(
self,
@"Super init cannot be nil");
194 NSAssert(labelPrefix,
@"labelPrefix is required");
197 _allowHeadlessExecution = allowHeadlessExecution;
198 _labelPrefix = [labelPrefix copy];
201 _enableEmbedderAPI = _dartProject.
settings.enable_embedder_api;
202 if (_enableEmbedderAPI) {
203 NSLog(
@"============== iOS: enable_embedder_api is on ==============");
204 _embedderAPI.struct_size =
sizeof(FlutterEngineProcTable);
205 FlutterEngineGetProcAddresses(&_embedderAPI);
208 if (!EnableTracingIfNecessary(_dartProject.settings)) {
210 @"Cannot create a FlutterEngine instance in debug mode without Flutter tooling or "
211 @"Xcode.\n\nTo launch in debug mode in iOS 14+, run flutter run from Flutter tools, run "
212 @"from an IDE with a Flutter IDE plugin or run the iOS project from Xcode.\nAlternatively "
213 @"profile and release mode apps can be launched from the home screen.");
217 _pluginPublications = [[NSMutableDictionary alloc] init];
218 _registrars = [[NSMutableDictionary alloc] init];
219 [
self recreatePlatformViewsController];
224 NSNotificationCenter* center = [NSNotificationCenter defaultCenter];
225 [center addObserver:self
226 selector:@selector(onMemoryWarning:)
227 name:UIApplicationDidReceiveMemoryWarningNotification
230 #if APPLICATION_EXTENSION_API_ONLY
231 if (@available(iOS 13.0, *)) {
232 [
self setUpSceneLifecycleNotifications:center];
234 [
self setUpApplicationLifecycleNotifications:center];
237 [
self setUpApplicationLifecycleNotifications:center];
240 [center addObserver:self
241 selector:@selector(onLocaleUpdated:)
242 name:NSCurrentLocaleDidChangeNotification
249 NSAssert([[NSThread currentThread] isMainThread],
@"Must be called on the main thread.");
250 return (__bridge
FlutterEngine*)
reinterpret_cast<void*
>(identifier);
253 - (void)setUpSceneLifecycleNotifications:(NSNotificationCenter*)center API_AVAILABLE(ios(13.0)) {
254 [center addObserver:self
255 selector:@selector(sceneWillEnterForeground:)
256 name:UISceneWillEnterForegroundNotification
258 [center addObserver:self
259 selector:@selector(sceneDidEnterBackground:)
260 name:UISceneDidEnterBackgroundNotification
264 - (void)setUpApplicationLifecycleNotifications:(NSNotificationCenter*)center {
265 [center addObserver:self
266 selector:@selector(applicationWillEnterForeground:)
267 name:UIApplicationWillEnterForegroundNotification
269 [center addObserver:self
270 selector:@selector(applicationDidEnterBackground:)
271 name:UIApplicationDidEnterBackgroundNotification
275 - (void)recreatePlatformViewsController {
280 - (
flutter::IOSRenderingAPI)platformViewsRenderingAPI {
287 [_pluginPublications enumerateKeysAndObjectsUsingBlock:^(id key, id object, BOOL* stop) {
288 if ([object respondsToSelector:@selector(detachFromEngineForRegistrar:)]) {
289 NSObject<FlutterPluginRegistrar>* registrar = self.registrars[key];
290 [object detachFromEngineForRegistrar:registrar];
298 enumerateKeysAndObjectsUsingBlock:^(id key, FlutterEngineRegistrar* registrar, BOOL* stop) {
299 registrar.flutterEngine = nil;
305 NSNotificationCenter* center = [NSNotificationCenter defaultCenter];
306 if (_flutterViewControllerWillDeallocObserver) {
307 [center removeObserver:_flutterViewControllerWillDeallocObserver];
309 [center removeObserver:self];
317 - (void)updateViewportMetrics:(
flutter::ViewportMetrics)viewportMetrics {
318 if (!
self.platformView) {
321 self.platformView->SetViewportMetrics(flutter::kFlutterImplicitViewId, viewportMetrics);
324 - (void)dispatchPointerDataPacket:(std::unique_ptr<
flutter::PointerDataPacket>)packet {
325 if (!
self.platformView) {
328 self.platformView->DispatchPointerDataPacket(std::move(packet));
331 - (void)installFirstFrameCallback:(
void (^)(
void))block {
332 if (!
self.platformView) {
337 self.
platformView->SetNextFrameCallback([weakSelf, block] {
342 FML_DCHECK(strongSelf.platformTaskRunner);
343 FML_DCHECK(strongSelf.rasterTaskRunner);
344 FML_DCHECK(strongSelf.rasterTaskRunner->RunsTasksOnCurrentThread());
346 strongSelf.platformTaskRunner->PostTask([block]() { block(); });
350 - (void)enableSemantics:(BOOL)enabled withFlags:(int64_t)flags {
351 if (!
self.platformView) {
355 self.platformView->SetAccessibilityFeatures(flags);
358 - (void)notifyViewCreated {
359 if (!
self.platformView) {
362 self.platformView->NotifyCreated();
365 - (void)notifyViewDestroyed {
366 if (!
self.platformView) {
369 self.platformView->NotifyDestroyed();
372 - (
flutter::PlatformViewIOS*)platformView {
379 - (
fml::RefPtr<fml::TaskRunner>)platformTaskRunner {
383 return _shell->GetTaskRunners().GetPlatformTaskRunner();
386 - (
fml::RefPtr<fml::TaskRunner>)uiTaskRunner {
390 return _shell->GetTaskRunners().GetUITaskRunner();
393 - (
fml::RefPtr<fml::TaskRunner>)rasterTaskRunner {
397 return _shell->GetTaskRunners().GetRasterTaskRunner();
400 - (void)sendKeyEvent:(const FlutterKeyEvent&)event
401 callback:(FlutterKeyEventCallback)callback
402 userData:(
void*)userData API_AVAILABLE(ios(13.4)) {
403 if (@available(iOS 13.4, *)) {
407 if (!
self.platformView) {
410 const char* character =
event.character;
412 flutter::KeyData key_data;
414 key_data.timestamp = (uint64_t)event.timestamp;
415 switch (event.type) {
416 case kFlutterKeyEventTypeUp:
417 key_data.type = flutter::KeyEventType::kUp;
419 case kFlutterKeyEventTypeDown:
420 key_data.type = flutter::KeyEventType::kDown;
422 case kFlutterKeyEventTypeRepeat:
423 key_data.type = flutter::KeyEventType::kRepeat;
426 key_data.physical =
event.physical;
427 key_data.logical =
event.logical;
428 key_data.synthesized =
event.synthesized;
430 auto packet = std::make_unique<flutter::KeyDataPacket>(key_data, character);
431 NSData* message = [NSData dataWithBytes:packet->data().data() length:packet->data().size()];
433 auto response = ^(NSData* reply) {
434 if (callback ==
nullptr) {
437 BOOL handled = FALSE;
438 if (reply.length == 1 && *
reinterpret_cast<const uint8_t*
>(reply.bytes) == 1) {
441 callback(handled, userData);
444 [
self sendOnChannel:kFlutterKeyDataChannel message:message binaryReply:response];
447 - (void)ensureSemanticsEnabled {
448 if (!
self.platformView) {
451 self.platformView->SetSemanticsEnabled(
true);
455 FML_DCHECK(
self.platformView);
457 self.platformView->SetOwnerViewController(_viewController);
458 [
self maybeSetupPlatformViewChannels];
459 [
self updateDisplays];
464 self.flutterViewControllerWillDeallocObserver =
465 [[NSNotificationCenter defaultCenter] addObserverForName:FlutterViewControllerWillDealloc
466 object:viewController
467 queue:[NSOperationQueue mainQueue]
468 usingBlock:^(NSNotification* note) {
469 [weakSelf notifyViewControllerDeallocated];
472 self.flutterViewControllerWillDeallocObserver = nil;
473 [
self notifyLowMemory];
478 FML_DCHECK(
self.platformView);
479 self.platformView->attachView();
482 - (void)setFlutterViewControllerWillDeallocObserver:(
id<NSObject>)observer {
483 if (observer != _flutterViewControllerWillDeallocObserver) {
484 if (_flutterViewControllerWillDeallocObserver) {
485 [[NSNotificationCenter defaultCenter]
486 removeObserver:_flutterViewControllerWillDeallocObserver];
488 _flutterViewControllerWillDeallocObserver = observer;
492 - (void)notifyViewControllerDeallocated {
493 [
self.lifecycleChannel sendMessage:@"AppLifecycleState.detached"];
494 self.textInputPlugin.viewController = nil;
495 if (!
self.allowHeadlessExecution) {
496 [
self destroyContext];
497 }
else if (
self.platformView) {
498 self.platformView->SetOwnerViewController({});
500 [
self.textInputPlugin resetViewResponder];
501 _viewController = nil;
504 - (void)destroyContext {
505 [
self resetChannels];
506 self.isolateId = nil;
510 _platformViewsController = nil;
513 - (NSURL*)observatoryUrl {
514 return self.publisher.url;
517 - (NSURL*)vmServiceUrl {
518 return self.publisher.url;
521 - (void)resetChannels {
522 self.localizationChannel = nil;
523 self.navigationChannel = nil;
524 self.restorationChannel = nil;
525 self.platformChannel = nil;
526 self.platformViewsChannel = nil;
527 self.textInputChannel = nil;
528 self.undoManagerChannel = nil;
529 self.scribbleChannel = nil;
530 self.lifecycleChannel = nil;
531 self.systemChannel = nil;
532 self.settingsChannel = nil;
533 self.keyEventChannel = nil;
534 self.spellCheckChannel = nil;
537 - (void)startProfiler {
538 FML_DCHECK(!_threadHost->name_prefix.empty());
539 _profiler = std::make_shared<flutter::SamplingProfiler>(
540 _threadHost->name_prefix.c_str(), _threadHost->profiler_thread->GetTaskRunner(),
542 flutter::ProfilerMetricsIOS profiler_metrics;
543 return profiler_metrics.GenerateSample();
552 - (void)setUpChannels {
556 [_binaryMessenger setMessageHandlerOnChannel:@"flutter/isolate"
557 binaryMessageHandler:^(NSData* message, FlutterBinaryReply reply) {
566 binaryMessenger:self.binaryMessenger
569 self.navigationChannel =
571 binaryMessenger:self.binaryMessenger
574 if ([_initialRoute length] > 0) {
576 [
self.navigationChannel invokeMethod:@"setInitialRoute" arguments:_initialRoute];
580 self.restorationChannel =
582 binaryMessenger:self.binaryMessenger
585 self.platformChannel =
587 binaryMessenger:self.binaryMessenger
590 self.platformViewsChannel =
592 binaryMessenger:self.binaryMessenger
595 self.textInputChannel =
597 binaryMessenger:self.binaryMessenger
600 self.undoManagerChannel =
602 binaryMessenger:self.binaryMessenger
605 self.scribbleChannel =
607 binaryMessenger:self.binaryMessenger
610 self.spellCheckChannel =
612 binaryMessenger:self.binaryMessenger
615 self.lifecycleChannel =
617 binaryMessenger:self.binaryMessenger
622 binaryMessenger:self.binaryMessenger
625 self.settingsChannel =
627 binaryMessenger:self.binaryMessenger
630 self.keyEventChannel =
632 binaryMessenger:self.binaryMessenger
636 self.textInputPlugin.indirectScribbleDelegate =
self;
637 [
self.textInputPlugin setUpIndirectScribbleInteraction:self.viewController];
642 self.restorationPlugin =
644 restorationEnabled:self.restorationEnabled];
647 self.screenshotChannel =
649 binaryMessenger:self.binaryMessenger
652 [
self.screenshotChannel setMethodCallHandler:^(FlutterMethodCall* _Nonnull call,
653 FlutterResult _Nonnull result) {
655 if (!(strongSelf && strongSelf->_shell && strongSelf->_shell->IsSetup())) {
658 message:@"Requesting screenshot while engine is not running."
661 flutter::Rasterizer::Screenshot screenshot =
662 [strongSelf screenshot:flutter::Rasterizer::ScreenshotType::SurfaceData base64Encode:NO];
663 if (!screenshot.data) {
665 message:@"Unable to get screenshot."
669 NSData* data = [NSData dataWithBytes:screenshot.data->writable_data()
670 length:screenshot.data->size()];
671 NSString* format = [NSString stringWithUTF8String:screenshot.format.c_str()];
672 NSNumber* width = @(screenshot.frame_size.fWidth);
673 NSNumber* height = @(screenshot.frame_size.fHeight);
674 return result(@[ width, height, format ?: [NSNull null], data ]);
678 - (void)maybeSetupPlatformViewChannels {
679 if (
_shell &&
self.shell.IsSetup()) {
682 [
self.platformChannel setMethodCallHandler:^(FlutterMethodCall* call, FlutterResult result) {
683 [weakSelf.platformPlugin handleMethodCall:call result:result];
686 [
self.platformViewsChannel
687 setMethodCallHandler:^(FlutterMethodCall* call, FlutterResult result) {
689 [weakSelf.platformViewsController onMethodCall:call result:result];
693 [
self.textInputChannel setMethodCallHandler:^(FlutterMethodCall* call, FlutterResult result) {
694 [weakSelf.textInputPlugin handleMethodCall:call result:result];
697 [
self.undoManagerChannel setMethodCallHandler:^(FlutterMethodCall* call, FlutterResult result) {
698 [weakSelf.undoManagerPlugin handleMethodCall:call result:result];
701 [
self.spellCheckChannel setMethodCallHandler:^(FlutterMethodCall* call, FlutterResult result) {
702 [weakSelf.spellCheckPlugin handleMethodCall:call result:result];
707 - (
flutter::Rasterizer::Screenshot)screenshot:(
flutter::Rasterizer::ScreenshotType)type
708 base64Encode:(
bool)base64Encode {
709 return self.shell.Screenshot(type, base64Encode);
712 - (void)launchEngine:(NSString*)entrypoint
713 libraryURI:(NSString*)libraryOrNil
714 entrypointArgs:(NSArray<NSString*>*)entrypointArgs {
716 flutter::RunConfiguration configuration =
717 [
self.dartProject runConfigurationForEntrypoint:entrypoint
718 libraryOrNil:libraryOrNil
719 entrypointArgs:entrypointArgs];
721 configuration.SetEngineId(
self.engineIdentifier);
722 self.shell.RunEngine(std::move(configuration));
725 - (void)setUpShell:(std::unique_ptr<
flutter::Shell>)shell
726 withVMServicePublication:(BOOL)doesVMServicePublication {
727 _shell = std::move(shell);
728 [
self setUpChannels];
729 [
self onLocaleUpdated:nil];
730 [
self updateDisplays];
732 initWithEnableVMServicePublication:doesVMServicePublication];
733 [
self maybeSetupPlatformViewChannels];
734 _shell->SetGpuAvailability(_isGpuDisabled ? flutter::GpuAvailability::kUnavailable
735 : flutter::GpuAvailability::kAvailable);
738 + (BOOL)isProfilerEnabled {
739 bool profilerEnabled =
false;
740 #if (FLUTTER_RUNTIME_MODE == FLUTTER_RUNTIME_MODE_DEBUG) || \
741 (FLUTTER_RUNTIME_MODE == FLUTTER_RUNTIME_MODE_PROFILE)
742 profilerEnabled =
true;
744 return profilerEnabled;
747 + (NSString*)generateThreadLabel:(NSString*)labelPrefix {
748 static size_t s_shellCount = 0;
749 return [NSString stringWithFormat:@"%@.%zu", labelPrefix, ++s_shellCount];
752 static flutter::ThreadHost MakeThreadHost(NSString* thread_label,
753 const flutter::Settings& settings) {
756 fml::MessageLoop::EnsureInitializedForCurrentThread();
758 uint32_t threadHostType = flutter::ThreadHost::Type::kRaster | flutter::ThreadHost::Type::kIo;
759 if (!settings.merged_platform_ui_thread) {
760 threadHostType |= flutter::ThreadHost::Type::kUi;
764 threadHostType = threadHostType | flutter::ThreadHost::Type::kProfiler;
767 flutter::ThreadHost::ThreadHostConfig host_config(thread_label.UTF8String, threadHostType,
770 host_config.ui_config =
771 fml::Thread::ThreadConfig(flutter::ThreadHost::ThreadHostConfig::MakeThreadName(
772 flutter::ThreadHost::Type::kUi, thread_label.UTF8String),
773 fml::Thread::ThreadPriority::kDisplay);
774 host_config.raster_config =
775 fml::Thread::ThreadConfig(flutter::ThreadHost::ThreadHostConfig::MakeThreadName(
776 flutter::ThreadHost::Type::kRaster, thread_label.UTF8String),
777 fml::Thread::ThreadPriority::kRaster);
779 host_config.io_config =
780 fml::Thread::ThreadConfig(flutter::ThreadHost::ThreadHostConfig::MakeThreadName(
781 flutter::ThreadHost::Type::kIo, thread_label.UTF8String),
782 fml::Thread::ThreadPriority::kNormal);
784 return (flutter::ThreadHost){host_config};
787 static void SetEntryPoint(flutter::Settings* settings, NSString* entrypoint, NSString* libraryURI) {
789 FML_DCHECK(entrypoint) <<
"Must specify entrypoint if specifying library";
790 settings->advisory_script_entrypoint = entrypoint.UTF8String;
791 settings->advisory_script_uri = libraryURI.UTF8String;
792 }
else if (entrypoint) {
793 settings->advisory_script_entrypoint = entrypoint.UTF8String;
794 settings->advisory_script_uri = std::string(
"main.dart");
796 settings->advisory_script_entrypoint = std::string(
"main");
797 settings->advisory_script_uri = std::string(
"main.dart");
801 - (BOOL)createShell:(NSString*)entrypoint
802 libraryURI:(NSString*)libraryURI
803 initialRoute:(NSString*)initialRoute {
805 FML_LOG(WARNING) <<
"This FlutterEngine was already invoked.";
809 self.initialRoute = initialRoute;
811 auto settings = [
self.dartProject settings];
812 if (initialRoute != nil) {
813 self.initialRoute = initialRoute;
814 }
else if (settings.route.empty() ==
false) {
815 self.initialRoute = [NSString stringWithUTF8String:settings.route.c_str()];
818 auto platformData = [
self.dartProject defaultPlatformData];
820 SetEntryPoint(&settings, entrypoint, libraryURI);
822 NSString* threadLabel = [
FlutterEngine generateThreadLabel:self.labelPrefix];
823 _threadHost = std::make_shared<flutter::ThreadHost>();
824 *_threadHost = MakeThreadHost(threadLabel, settings);
827 flutter::Shell::CreateCallback<flutter::PlatformView> on_create_platform_view =
828 [weakSelf](flutter::Shell& shell) {
831 return std::unique_ptr<flutter::PlatformViewIOS>();
833 [strongSelf recreatePlatformViewsController];
834 strongSelf.platformViewsController.taskRunner =
835 shell.GetTaskRunners().GetPlatformTaskRunner();
836 return std::make_unique<flutter::PlatformViewIOS>(
837 shell, strongSelf->_renderingApi, strongSelf.platformViewsController,
838 shell.GetTaskRunners(), shell.GetConcurrentWorkerTaskRunner(),
839 shell.GetIsGpuDisabledSyncSwitch());
842 flutter::Shell::CreateCallback<flutter::Rasterizer> on_create_rasterizer =
843 [](flutter::Shell& shell) {
return std::make_unique<flutter::Rasterizer>(shell); };
845 fml::RefPtr<fml::TaskRunner> ui_runner;
846 if (settings.enable_impeller && settings.merged_platform_ui_thread) {
847 ui_runner = fml::MessageLoop::GetCurrent().GetTaskRunner();
849 ui_runner = _threadHost->ui_thread->GetTaskRunner();
851 flutter::TaskRunners task_runners(threadLabel.UTF8String,
852 fml::MessageLoop::GetCurrent().GetTaskRunner(),
853 _threadHost->raster_thread->GetTaskRunner(),
855 _threadHost->io_thread->GetTaskRunner()
858 #if APPLICATION_EXTENSION_API_ONLY
859 if (@available(iOS 13.0, *)) {
860 _isGpuDisabled =
self.viewController.flutterWindowSceneIfViewLoaded.activationState ==
861 UISceneActivationStateBackground;
869 [UIApplication sharedApplication].applicationState == UIApplicationStateBackground;
873 std::unique_ptr<flutter::Shell> shell = flutter::Shell::Create(
877 on_create_platform_view,
878 on_create_rasterizer,
881 if (shell ==
nullptr) {
882 FML_LOG(ERROR) <<
"Could not start a shell FlutterEngine with entrypoint: "
883 << entrypoint.UTF8String;
886 FML_LOG(INFO) <<
"Enabled VM Service Publication: " << settings.enable_vm_service_publication;
887 [
self setUpShell:std::move(shell)
888 withVMServicePublication:settings.enable_vm_service_publication];
890 [
self startProfiler];
897 - (void)updateDisplays {
902 auto vsync_waiter =
_shell->GetVsyncWaiter().lock();
903 auto vsync_waiter_ios = std::static_pointer_cast<flutter::VsyncWaiterIOS>(vsync_waiter);
904 std::vector<std::unique_ptr<flutter::Display>> displays;
905 auto screen_size = UIScreen.mainScreen.nativeBounds.size;
906 auto scale = UIScreen.mainScreen.scale;
907 displays.push_back(std::make_unique<flutter::VariableRefreshRateDisplay>(
908 0, vsync_waiter_ios, screen_size.width, screen_size.height, scale));
909 _shell->OnDisplayUpdates(std::move(displays));
918 - (BOOL)runWithEntrypoint:(NSString*)entrypoint libraryURI:(NSString*)libraryURI {
919 return [
self runWithEntrypoint:entrypoint
920 libraryURI:libraryURI
921 initialRoute:FlutterDefaultInitialRoute];
924 - (BOOL)runWithEntrypoint:(NSString*)entrypoint {
925 return [
self runWithEntrypoint:entrypoint libraryURI:nil initialRoute:FlutterDefaultInitialRoute];
928 - (BOOL)runWithEntrypoint:(NSString*)entrypoint initialRoute:(NSString*)initialRoute {
929 return [
self runWithEntrypoint:entrypoint libraryURI:nil initialRoute:initialRoute];
932 - (BOOL)runWithEntrypoint:(NSString*)entrypoint
933 libraryURI:(NSString*)libraryURI
934 initialRoute:(NSString*)initialRoute {
935 return [
self runWithEntrypoint:entrypoint
936 libraryURI:libraryURI
937 initialRoute:initialRoute
941 - (BOOL)runWithEntrypoint:(NSString*)entrypoint
942 libraryURI:(NSString*)libraryURI
943 initialRoute:(NSString*)initialRoute
944 entrypointArgs:(NSArray<NSString*>*)entrypointArgs {
945 if ([
self createShell:entrypoint libraryURI:libraryURI initialRoute:initialRoute]) {
946 [
self launchEngine:entrypoint libraryURI:libraryURI entrypointArgs:entrypointArgs];
952 - (void)notifyLowMemory {
954 _shell->NotifyLowMemoryWarning();
956 [
self.systemChannel sendMessage:@{@"type" : @"memoryPressure"}];
959 #pragma mark - Text input delegate
962 updateEditingClient:(
int)client
963 withState:(NSDictionary*)state {
964 [
self.textInputChannel invokeMethod:@"TextInputClient.updateEditingState"
965 arguments:@[ @(client), state ]];
969 updateEditingClient:(
int)client
970 withState:(NSDictionary*)state
971 withTag:(NSString*)tag {
972 [
self.textInputChannel invokeMethod:@"TextInputClient.updateEditingStateWithTag"
973 arguments:@[ @(client), @{tag : state} ]];
977 updateEditingClient:(
int)client
978 withDelta:(NSDictionary*)delta {
979 [
self.textInputChannel invokeMethod:@"TextInputClient.updateEditingStateWithDeltas"
980 arguments:@[ @(client), delta ]];
984 updateFloatingCursor:(FlutterFloatingCursorDragState)state
985 withClient:(
int)client
986 withPosition:(NSDictionary*)position {
987 NSString* stateString;
989 case FlutterFloatingCursorDragStateStart:
990 stateString =
@"FloatingCursorDragState.start";
992 case FlutterFloatingCursorDragStateUpdate:
993 stateString =
@"FloatingCursorDragState.update";
995 case FlutterFloatingCursorDragStateEnd:
996 stateString =
@"FloatingCursorDragState.end";
999 [
self.textInputChannel invokeMethod:@"TextInputClient.updateFloatingCursor"
1000 arguments:@[ @(client), stateString, position ]];
1004 performAction:(FlutterTextInputAction)action
1005 withClient:(
int)client {
1006 NSString* actionString;
1008 case FlutterTextInputActionUnspecified:
1013 actionString =
@"TextInputAction.unspecified";
1015 case FlutterTextInputActionDone:
1016 actionString =
@"TextInputAction.done";
1018 case FlutterTextInputActionGo:
1019 actionString =
@"TextInputAction.go";
1021 case FlutterTextInputActionSend:
1022 actionString =
@"TextInputAction.send";
1024 case FlutterTextInputActionSearch:
1025 actionString =
@"TextInputAction.search";
1027 case FlutterTextInputActionNext:
1028 actionString =
@"TextInputAction.next";
1030 case FlutterTextInputActionContinue:
1031 actionString =
@"TextInputAction.continueAction";
1033 case FlutterTextInputActionJoin:
1034 actionString =
@"TextInputAction.join";
1036 case FlutterTextInputActionRoute:
1037 actionString =
@"TextInputAction.route";
1039 case FlutterTextInputActionEmergencyCall:
1040 actionString =
@"TextInputAction.emergencyCall";
1042 case FlutterTextInputActionNewline:
1043 actionString =
@"TextInputAction.newline";
1046 [
self.textInputChannel invokeMethod:@"TextInputClient.performAction"
1047 arguments:@[ @(client), actionString ]];
1051 showAutocorrectionPromptRectForStart:(NSUInteger)start
1053 withClient:(
int)client {
1054 [
self.textInputChannel invokeMethod:@"TextInputClient.showAutocorrectionPromptRect"
1055 arguments:@[ @(client), @(start), @(end) ]];
1059 willDismissEditMenuWithTextInputClient:(
int)client {
1060 [
self.platformChannel invokeMethod:@"ContextMenu.onDismissSystemContextMenu"
1061 arguments:@[ @(client) ]];
1065 shareSelectedText:(NSString*)selectedText {
1066 [
self.platformPlugin showShareViewController:selectedText];
1070 searchWebWithSelectedText:(NSString*)selectedText {
1071 [
self.platformPlugin searchWeb:selectedText];
1075 lookUpSelectedText:(NSString*)selectedText {
1076 [
self.platformPlugin showLookUpViewController:selectedText];
1079 #pragma mark - FlutterViewEngineDelegate
1085 [
self.textInputChannel invokeMethod:@"TextInputClient.showToolbar" arguments:@[ @(client) ]];
1089 focusElement:(UIScribbleElementIdentifier)elementIdentifier
1090 atPoint:(CGPoint)referencePoint
1095 [
self.textInputChannel
1096 invokeMethod:@"TextInputClient.focusElement"
1097 arguments:@[ elementIdentifier, @(referencePoint.x), @(referencePoint.y) ]
1102 requestElementsInRect:(CGRect)rect
1107 [
self.textInputChannel
1108 invokeMethod:@"TextInputClient.requestElementsInRect"
1109 arguments:@[ @(rect.origin.x), @(rect.origin.y), @(rect.size.width), @(rect.size.height) ]
1117 [
self.textInputChannel invokeMethod:@"TextInputClient.scribbleInteractionBegan" arguments:nil];
1124 [
self.textInputChannel invokeMethod:@"TextInputClient.scribbleInteractionFinished" arguments:nil];
1128 insertTextPlaceholderWithSize:(CGSize)size
1129 withClient:(
int)client {
1133 [
self.textInputChannel invokeMethod:@"TextInputClient.insertTextPlaceholder"
1134 arguments:@[ @(client), @(size.width), @(size.height) ]];
1138 removeTextPlaceholder:(
int)client {
1142 [
self.textInputChannel invokeMethod:@"TextInputClient.removeTextPlaceholder"
1143 arguments:@[ @(client) ]];
1147 didResignFirstResponderWithTextInputClient:(
int)client {
1151 [
self.textInputChannel invokeMethod:@"TextInputClient.onConnectionClosed"
1152 arguments:@[ @(client) ]];
1173 dispatch_async(dispatch_get_main_queue(), ^(
void) {
1174 long platform_view_id = [
self.platformViewsController firstResponderPlatformViewId];
1175 if (platform_view_id == -1) {
1179 [
self.platformViewsChannel invokeMethod:@"viewFocused" arguments:@(platform_view_id)];
1183 #pragma mark - Undo Manager Delegate
1185 - (void)handleUndoWithDirection:(FlutterUndoRedoDirection)direction {
1186 NSString* action = (direction == FlutterUndoRedoDirectionUndo) ?
@"undo" :
@"redo";
1187 [
self.undoManagerChannel invokeMethod:@"UndoManagerClient.handleUndo" arguments:@[ action ]];
1190 - (UIView<UITextInput>*)activeTextInputView {
1191 return [[
self textInputPlugin] textInputView];
1194 - (NSUndoManager*)undoManager {
1195 return self.viewController.undoManager;
1198 #pragma mark - Screenshot Delegate
1200 - (
flutter::Rasterizer::Screenshot)takeScreenshot:(
flutter::Rasterizer::ScreenshotType)type
1201 asBase64Encoded:(BOOL)base64Encode {
1202 FML_DCHECK(
_shell) <<
"Cannot takeScreenshot without a shell";
1203 return _shell->Screenshot(type, base64Encode);
1206 - (void)flutterViewAccessibilityDidCall {
1208 [
self ensureSemanticsEnabled];
1230 #pragma mark - FlutterBinaryMessenger
1232 - (void)sendOnChannel:(NSString*)channel message:(NSData*)message {
1233 [
self sendOnChannel:channel message:message binaryReply:nil];
1236 - (void)sendOnChannel:(NSString*)channel
1237 message:(NSData*)message
1239 NSParameterAssert(channel);
1241 @"Sending a message before the FlutterEngine has been run.");
1242 fml::RefPtr<flutter::PlatformMessageResponseDarwin> response =
1243 (callback == nil) ?
nullptr
1244 : fml::MakeRefCounted<flutter::PlatformMessageResponseDarwin>(
1248 _shell->GetTaskRunners().GetPlatformTaskRunner());
1249 std::unique_ptr<flutter::PlatformMessage> platformMessage =
1250 (message == nil) ? std::make_unique<flutter::PlatformMessage>(channel.UTF8String, response)
1251 : std::make_unique<flutter::PlatformMessage>(
1254 _shell->GetPlatformView()->DispatchPlatformMessage(std::move(platformMessage));
1264 binaryMessageHandler:
1266 return [
self setMessageHandlerOnChannel:channel binaryMessageHandler:handler taskQueue:nil];
1270 setMessageHandlerOnChannel:(NSString*)channel
1273 NSParameterAssert(channel);
1275 self.platformView->GetPlatformMessageHandlerIos()->SetMessageHandler(channel.UTF8String,
1276 handler, taskQueue);
1277 return _connections->AquireConnection(channel.UTF8String);
1279 NSAssert(!handler,
@"Setting a message handler before the FlutterEngine has been run.");
1287 std::string channel =
_connections->CleanupConnection(connection);
1288 if (!channel.empty()) {
1289 self.platformView->GetPlatformMessageHandlerIos()->SetMessageHandler(channel.c_str(), nil,
1295 #pragma mark - FlutterTextureRegistry
1298 FML_DCHECK(
self.platformView);
1299 int64_t textureId =
self.nextTextureId++;
1300 self.platformView->RegisterExternalTexture(textureId, texture);
1304 - (void)unregisterTexture:(int64_t)textureId {
1305 _shell->GetPlatformView()->UnregisterTexture(textureId);
1308 - (void)textureFrameAvailable:(int64_t)textureId {
1309 _shell->GetPlatformView()->MarkTextureFrameAvailable(textureId);
1312 - (NSString*)lookupKeyForAsset:(NSString*)asset {
1316 - (NSString*)lookupKeyForAsset:(NSString*)asset fromPackage:(NSString*)package {
1320 - (id<FlutterPluginRegistry>)pluginRegistry {
1324 #pragma mark - FlutterPluginRegistry
1327 NSAssert(
self.pluginPublications[pluginKey] == nil,
@"Duplicate plugin key: %@", pluginKey);
1328 self.pluginPublications[pluginKey] = [NSNull null];
1330 flutterEngine:self];
1331 self.registrars[pluginKey] = result;
1335 - (BOOL)hasPlugin:(NSString*)pluginKey {
1336 return _pluginPublications[pluginKey] != nil;
1339 - (NSObject*)valuePublishedByPlugin:(NSString*)pluginKey {
1340 return _pluginPublications[pluginKey];
1343 #pragma mark - Notifications
1345 #if APPLICATION_EXTENSION_API_ONLY
1346 - (void)sceneWillEnterForeground:(NSNotification*)notification API_AVAILABLE(ios(13.0)) {
1347 [
self flutterWillEnterForeground:notification];
1350 - (void)sceneDidEnterBackground:(NSNotification*)notification API_AVAILABLE(ios(13.0)) {
1351 [
self flutterDidEnterBackground:notification];
1354 - (void)applicationWillEnterForeground:(NSNotification*)notification {
1355 [
self flutterWillEnterForeground:notification];
1358 - (void)applicationDidEnterBackground:(NSNotification*)notification {
1359 [
self flutterDidEnterBackground:notification];
1363 - (void)flutterWillEnterForeground:(NSNotification*)notification {
1364 [
self setIsGpuDisabled:NO];
1367 - (void)flutterDidEnterBackground:(NSNotification*)notification {
1368 [
self setIsGpuDisabled:YES];
1369 [
self notifyLowMemory];
1372 - (void)onMemoryWarning:(NSNotification*)notification {
1373 [
self notifyLowMemory];
1376 - (void)setIsGpuDisabled:(BOOL)value {
1378 _shell->SetGpuAvailability(value ? flutter::GpuAvailability::kUnavailable
1379 : flutter::GpuAvailability::kAvailable);
1381 _isGpuDisabled = value;
1384 #pragma mark - Locale updates
1386 - (void)onLocaleUpdated:(NSNotification*)notification {
1388 NSMutableArray<NSString*>* localeData = [[NSMutableArray alloc] init];
1389 NSArray<NSString*>* preferredLocales = [NSLocale preferredLanguages];
1390 for (NSString* localeID in preferredLocales) {
1391 NSLocale* locale = [[NSLocale alloc] initWithLocaleIdentifier:localeID];
1392 NSString* languageCode = [locale objectForKey:NSLocaleLanguageCode];
1393 NSString* countryCode = [locale objectForKey:NSLocaleCountryCode];
1394 NSString* scriptCode = [locale objectForKey:NSLocaleScriptCode];
1395 NSString* variantCode = [locale objectForKey:NSLocaleVariantCode];
1396 if (!languageCode) {
1399 [localeData addObject:languageCode];
1400 [localeData addObject:(countryCode ? countryCode : @"")];
1401 [localeData addObject:(scriptCode ? scriptCode : @"")];
1402 [localeData addObject:(variantCode ? variantCode : @"")];
1404 if (localeData.count == 0) {
1407 [
self.localizationChannel invokeMethod:@"setLocale" arguments:localeData];
1410 - (void)waitForFirstFrameSync:(NSTimeInterval)timeout
1411 callback:(NS_NOESCAPE
void (^_Nonnull)(BOOL didTimeout))callback {
1412 fml::TimeDelta waitTime = fml::TimeDelta::FromMilliseconds(timeout * 1000);
1413 fml::Status status =
self.shell.WaitForFirstFrame(waitTime);
1414 callback(status.code() == fml::StatusCode::kDeadlineExceeded);
1417 - (void)waitForFirstFrame:(NSTimeInterval)timeout
1418 callback:(
void (^_Nonnull)(BOOL didTimeout))callback {
1419 dispatch_queue_t queue = dispatch_get_global_queue(QOS_CLASS_BACKGROUND, 0);
1420 dispatch_group_t group = dispatch_group_create();
1423 __block BOOL didTimeout = NO;
1424 dispatch_group_async(group, queue, ^{
1430 fml::TimeDelta waitTime = fml::TimeDelta::FromMilliseconds(timeout * 1000);
1431 fml::Status status = strongSelf.
shell.WaitForFirstFrame(waitTime);
1432 didTimeout = status.code() == fml::StatusCode::kDeadlineExceeded;
1436 dispatch_group_notify(group, dispatch_get_main_queue(), ^{
1450 callback(didTimeout);
1454 - (
FlutterEngine*)spawnWithEntrypoint:( NSString*)entrypoint
1455 libraryURI:( NSString*)libraryURI
1456 initialRoute:( NSString*)initialRoute
1457 entrypointArgs:( NSArray<NSString*>*)entrypointArgs {
1458 NSAssert(
_shell,
@"Spawning from an engine without a shell (possibly not run).");
1460 project:self.dartProject
1461 allowHeadlessExecution:self.allowHeadlessExecution];
1462 flutter::RunConfiguration configuration =
1463 [
self.dartProject runConfigurationForEntrypoint:entrypoint
1464 libraryOrNil:libraryURI
1465 entrypointArgs:entrypointArgs];
1467 configuration.SetEngineId(result.engineIdentifier);
1474 std::shared_ptr<flutter::IOSContext> context = ios_platform_view->
GetIosContext();
1475 FML_DCHECK(context);
1479 flutter::Shell::CreateCallback<flutter::PlatformView> on_create_platform_view =
1480 [result, context](flutter::Shell& shell) {
1481 [result recreatePlatformViewsController];
1482 result.platformViewsController.taskRunner = shell.GetTaskRunners().GetPlatformTaskRunner();
1483 return std::make_unique<flutter::PlatformViewIOS>(
1484 shell, context, result.platformViewsController, shell.GetTaskRunners());
1487 flutter::Shell::CreateCallback<flutter::Rasterizer> on_create_rasterizer =
1488 [](flutter::Shell& shell) {
return std::make_unique<flutter::Rasterizer>(shell); };
1490 std::string cppInitialRoute;
1492 cppInitialRoute = [initialRoute UTF8String];
1495 std::unique_ptr<flutter::Shell> shell =
_shell->Spawn(
1496 std::move(configuration), cppInitialRoute, on_create_platform_view, on_create_rasterizer);
1498 result->_threadHost = _threadHost;
1500 result->_isGpuDisabled = _isGpuDisabled;
1501 [result setUpShell:std::move(shell) withVMServicePublication:NO];
1505 - (const
flutter::ThreadHost&)threadHost {
1506 return *_threadHost;
1510 return self.dartProject;
1516 NSString* _pluginKey;
1519 - (instancetype)initWithPlugin:(NSString*)pluginKey flutterEngine:(
FlutterEngine*)flutterEngine {
1520 self = [
super init];
1521 NSAssert(
self,
@"Super init cannot be nil");
1522 _pluginKey = [pluginKey copy];
1523 _flutterEngine = flutterEngine;
1528 return _flutterEngine.binaryMessenger;
1532 return _flutterEngine.textureRegistry;
1535 - (void)publish:(NSObject*)value {
1536 _flutterEngine.pluginPublications[_pluginKey] = value;
1539 - (void)addMethodCallDelegate:(NSObject<
FlutterPlugin>*)delegate
1546 - (void)addApplicationDelegate:(NSObject<
FlutterPlugin>*)delegate
1547 NS_EXTENSION_UNAVAILABLE_IOS("Disallowed in plugins used in app extensions") {
1548 id<UIApplicationDelegate> appDelegate = [[UIApplication sharedApplication] delegate];
1550 id<FlutterAppLifeCycleProvider> lifeCycleProvider =
1551 (id<FlutterAppLifeCycleProvider>)appDelegate;
1552 [lifeCycleProvider addApplicationLifeCycleDelegate:delegate];
1556 - (NSString*)lookupKeyForAsset:(NSString*)asset {
1557 return [_flutterEngine lookupKeyForAsset:asset];
1560 - (NSString*)lookupKeyForAsset:(NSString*)asset fromPackage:(NSString*)package {
1561 return [_flutterEngine lookupKeyForAsset:asset fromPackage:package];
1565 withId:(NSString*)factoryId {
1566 [
self registerViewFactory:factory
1568 gestureRecognizersBlockingPolicy:FlutterPlatformViewGestureRecognizersBlockingPolicyEager];
1572 withId:(NSString*)factoryId
1573 gestureRecognizersBlockingPolicy:
1575 [_flutterEngine.platformViewsController registerViewFactory:factory
1577 gestureRecognizersBlockingPolicy:gestureRecognizersBlockingPolicy];