Flutter Windows Embedder
flutter_windows_engine.cc
Go to the documentation of this file.
1 // Copyright 2013 The Flutter Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4 
6 
7 #include <dwmapi.h>
8 
9 #include <filesystem>
10 #include <shared_mutex>
11 #include <sstream>
12 
13 #include "flutter/fml/logging.h"
14 #include "flutter/fml/paths.h"
15 #include "flutter/fml/platform/win/wstring_conversion.h"
16 #include "flutter/fml/synchronization/waitable_event.h"
20 #include "flutter/shell/platform/embedder/embedder_struct_macros.h"
28 #include "flutter/third_party/accessibility/ax/ax_node.h"
30 
31 // winbase.h defines GetCurrentTime as a macro.
32 #undef GetCurrentTime
33 
34 static constexpr char kAccessibilityChannelName[] = "flutter/accessibility";
35 
36 namespace flutter {
37 
38 namespace {
39 
40 // Lifted from vsync_waiter_fallback.cc
41 static std::chrono::nanoseconds SnapToNextTick(
42  std::chrono::nanoseconds value,
43  std::chrono::nanoseconds tick_phase,
44  std::chrono::nanoseconds tick_interval) {
45  std::chrono::nanoseconds offset = (tick_phase - value) % tick_interval;
46  if (offset != std::chrono::nanoseconds::zero())
47  offset = offset + tick_interval;
48  return value + offset;
49 }
50 
51 // Creates and returns a FlutterRendererConfig that renders to the view (if any)
52 // of a FlutterWindowsEngine, using OpenGL (via ANGLE).
53 // The user_data received by the render callbacks refers to the
54 // FlutterWindowsEngine.
55 FlutterRendererConfig GetOpenGLRendererConfig() {
56  FlutterRendererConfig config = {};
57  config.type = kOpenGL;
58  config.open_gl.struct_size = sizeof(config.open_gl);
59  config.open_gl.make_current = [](void* user_data) -> bool {
60  auto host = static_cast<FlutterWindowsEngine*>(user_data);
61  if (!host->egl_manager()) {
62  return false;
63  }
64  return host->egl_manager()->render_context()->MakeCurrent();
65  };
66  config.open_gl.clear_current = [](void* user_data) -> bool {
67  auto host = static_cast<FlutterWindowsEngine*>(user_data);
68  if (!host->egl_manager()) {
69  return false;
70  }
71  return host->egl_manager()->render_context()->ClearCurrent();
72  };
73  config.open_gl.present = [](void* user_data) -> bool { FML_UNREACHABLE(); };
74  config.open_gl.fbo_reset_after_present = true;
75  config.open_gl.fbo_with_frame_info_callback =
76  [](void* user_data, const FlutterFrameInfo* info) -> uint32_t {
77  FML_UNREACHABLE();
78  };
79  config.open_gl.gl_proc_resolver = [](void* user_data,
80  const char* what) -> void* {
81  return reinterpret_cast<void*>(eglGetProcAddress(what));
82  };
83  config.open_gl.make_resource_current = [](void* user_data) -> bool {
84  auto host = static_cast<FlutterWindowsEngine*>(user_data);
85  if (!host->egl_manager()) {
86  return false;
87  }
88  return host->egl_manager()->resource_context()->MakeCurrent();
89  };
90  config.open_gl.gl_external_texture_frame_callback =
91  [](void* user_data, int64_t texture_id, size_t width, size_t height,
92  FlutterOpenGLTexture* texture) -> bool {
93  auto host = static_cast<FlutterWindowsEngine*>(user_data);
94  if (!host->texture_registrar()) {
95  return false;
96  }
97  return host->texture_registrar()->PopulateTexture(texture_id, width, height,
98  texture);
99  };
100  return config;
101 }
102 
103 // Creates and returns a FlutterRendererConfig that renders to the view (if any)
104 // of a FlutterWindowsEngine, using software rasterization.
105 // The user_data received by the render callbacks refers to the
106 // FlutterWindowsEngine.
107 FlutterRendererConfig GetSoftwareRendererConfig() {
108  FlutterRendererConfig config = {};
109  config.type = kSoftware;
110  config.software.struct_size = sizeof(config.software);
111  config.software.surface_present_callback =
112  [](void* user_data, const void* allocation, size_t row_bytes,
113  size_t height) {
114  FML_UNREACHABLE();
115  return false;
116  };
117  return config;
118 }
119 
120 // Converts a FlutterPlatformMessage to an equivalent FlutterDesktopMessage.
121 static FlutterDesktopMessage ConvertToDesktopMessage(
122  const FlutterPlatformMessage& engine_message) {
124  message.struct_size = sizeof(message);
125  message.channel = engine_message.channel;
126  message.message = engine_message.message;
127  message.message_size = engine_message.message_size;
128  message.response_handle = engine_message.response_handle;
129  return message;
130 }
131 
132 // Converts a LanguageInfo struct to a FlutterLocale struct. |info| must outlive
133 // the returned value, since the returned FlutterLocale has pointers into it.
134 FlutterLocale CovertToFlutterLocale(const LanguageInfo& info) {
135  FlutterLocale locale = {};
136  locale.struct_size = sizeof(FlutterLocale);
137  locale.language_code = info.language.c_str();
138  if (!info.region.empty()) {
139  locale.country_code = info.region.c_str();
140  }
141  if (!info.script.empty()) {
142  locale.script_code = info.script.c_str();
143  }
144  return locale;
145 }
146 
147 } // namespace
148 
150  const FlutterProjectBundle& project,
151  std::shared_ptr<WindowsProcTable> windows_proc_table)
152  : project_(std::make_unique<FlutterProjectBundle>(project)),
153  windows_proc_table_(std::move(windows_proc_table)),
154  aot_data_(nullptr, nullptr),
155  lifecycle_manager_(std::make_unique<WindowsLifecycleManager>(this)) {
156  if (windows_proc_table_ == nullptr) {
157  windows_proc_table_ = std::make_shared<WindowsProcTable>();
158  }
159 
160  gl_ = egl::ProcTable::Create();
161 
162  embedder_api_.struct_size = sizeof(FlutterEngineProcTable);
163  FlutterEngineGetProcAddresses(&embedder_api_);
164 
165  task_runner_ =
166  std::make_unique<TaskRunner>(
167  embedder_api_.GetCurrentTime, [this](const auto* task) {
168  if (!engine_) {
169  FML_LOG(ERROR)
170  << "Cannot post an engine task when engine is not running.";
171  return;
172  }
173  if (embedder_api_.RunTask(engine_, task) != kSuccess) {
174  FML_LOG(ERROR) << "Failed to post an engine task.";
175  }
176  });
177 
178  // Set up the legacy structs backing the API handles.
179  messenger_ =
180  fml::RefPtr<FlutterDesktopMessenger>(new FlutterDesktopMessenger());
181  messenger_->SetEngine(this);
182  plugin_registrar_ = std::make_unique<FlutterDesktopPluginRegistrar>();
183  plugin_registrar_->engine = this;
184 
185  messenger_wrapper_ =
186  std::make_unique<BinaryMessengerImpl>(messenger_->ToRef());
187  message_dispatcher_ =
188  std::make_unique<IncomingMessageDispatcher>(messenger_->ToRef());
189 
190  texture_registrar_ =
191  std::make_unique<FlutterWindowsTextureRegistrar>(this, gl_);
192 
193  // Check for impeller support.
194  auto& switches = project_->GetSwitches();
195  enable_impeller_ = std::find(switches.begin(), switches.end(),
196  "--enable-impeller=true") != switches.end();
197 
198  egl_manager_ = egl::Manager::Create(
199  static_cast<egl::GpuPreference>(project_->gpu_preference()));
200  window_proc_delegate_manager_ = std::make_unique<WindowProcDelegateManager>();
201  window_proc_delegate_manager_->RegisterTopLevelWindowProcDelegate(
202  [](HWND hwnd, UINT msg, WPARAM wpar, LPARAM lpar, void* user_data,
203  LRESULT* result) {
204  BASE_DCHECK(user_data);
205  FlutterWindowsEngine* that =
206  static_cast<FlutterWindowsEngine*>(user_data);
207  BASE_DCHECK(that->lifecycle_manager_);
208  return that->lifecycle_manager_->WindowProc(hwnd, msg, wpar, lpar,
209  result);
210  },
211  static_cast<void*>(this));
212 
213  // Set up internal channels.
214  // TODO: Replace this with an embedder.h API. See
215  // https://github.com/flutter/flutter/issues/71099
216  internal_plugin_registrar_ =
217  std::make_unique<PluginRegistrar>(plugin_registrar_.get());
218 
219  accessibility_plugin_ = std::make_unique<AccessibilityPlugin>(this);
220  AccessibilityPlugin::SetUp(messenger_wrapper_.get(),
221  accessibility_plugin_.get());
222 
223  cursor_handler_ =
224  std::make_unique<CursorHandler>(messenger_wrapper_.get(), this);
225  platform_handler_ =
226  std::make_unique<PlatformHandler>(messenger_wrapper_.get(), this);
227  settings_plugin_ = std::make_unique<SettingsPlugin>(messenger_wrapper_.get(),
228  task_runner_.get());
229 }
230 
231 FlutterWindowsEngine::~FlutterWindowsEngine() {
232  messenger_->SetEngine(nullptr);
233  Stop();
234 }
235 
236 FlutterWindowsEngine* FlutterWindowsEngine::GetEngineForId(int64_t engine_id) {
237  return reinterpret_cast<FlutterWindowsEngine*>(engine_id);
238 }
239 
240 void FlutterWindowsEngine::SetSwitches(
241  const std::vector<std::string>& switches) {
242  project_->SetSwitches(switches);
243 }
244 
245 bool FlutterWindowsEngine::Run() {
246  return Run("");
247 }
248 
249 bool FlutterWindowsEngine::Run(std::string_view entrypoint) {
250  if (!project_->HasValidPaths()) {
251  FML_LOG(ERROR) << "Missing or unresolvable paths to assets.";
252  return false;
253  }
254  std::string assets_path_string = project_->assets_path().u8string();
255  std::string icu_path_string = project_->icu_path().u8string();
256  if (embedder_api_.RunsAOTCompiledDartCode()) {
257  aot_data_ = project_->LoadAotData(embedder_api_);
258  if (!aot_data_) {
259  FML_LOG(ERROR) << "Unable to start engine without AOT data.";
260  return false;
261  }
262  }
263 
264  // FlutterProjectArgs is expecting a full argv, so when processing it for
265  // flags the first item is treated as the executable and ignored. Add a dummy
266  // value so that all provided arguments are used.
267  std::string executable_name = GetExecutableName();
268  std::vector<const char*> argv = {executable_name.c_str()};
269  std::vector<std::string> switches = project_->GetSwitches();
270  std::transform(
271  switches.begin(), switches.end(), std::back_inserter(argv),
272  [](const std::string& arg) -> const char* { return arg.c_str(); });
273 
274  const std::vector<std::string>& entrypoint_args =
275  project_->dart_entrypoint_arguments();
276  std::vector<const char*> entrypoint_argv;
277  std::transform(
278  entrypoint_args.begin(), entrypoint_args.end(),
279  std::back_inserter(entrypoint_argv),
280  [](const std::string& arg) -> const char* { return arg.c_str(); });
281 
282  // Configure task runners.
283  FlutterTaskRunnerDescription platform_task_runner = {};
284  platform_task_runner.struct_size = sizeof(FlutterTaskRunnerDescription);
285  platform_task_runner.user_data = task_runner_.get();
286  platform_task_runner.runs_task_on_current_thread_callback =
287  [](void* user_data) -> bool {
288  return static_cast<TaskRunner*>(user_data)->RunsTasksOnCurrentThread();
289  };
290  platform_task_runner.post_task_callback = [](FlutterTask task,
291  uint64_t target_time_nanos,
292  void* user_data) -> void {
293  static_cast<TaskRunner*>(user_data)->PostFlutterTask(task,
294  target_time_nanos);
295  };
296  FlutterCustomTaskRunners custom_task_runners = {};
297  custom_task_runners.struct_size = sizeof(FlutterCustomTaskRunners);
298  custom_task_runners.platform_task_runner = &platform_task_runner;
299  custom_task_runners.thread_priority_setter =
301 
302  if (project_->ui_thread_policy() ==
304  FML_LOG(WARNING)
305  << "Running with merged platform and UI thread. Experimental.";
306  custom_task_runners.ui_task_runner = &platform_task_runner;
307  }
308 
309  FlutterProjectArgs args = {};
310  args.struct_size = sizeof(FlutterProjectArgs);
311  args.shutdown_dart_vm_when_done = true;
312  args.assets_path = assets_path_string.c_str();
313  args.icu_data_path = icu_path_string.c_str();
314  args.command_line_argc = static_cast<int>(argv.size());
315  args.command_line_argv = argv.empty() ? nullptr : argv.data();
316  args.engine_id = reinterpret_cast<int64_t>(this);
317 
318  // Fail if conflicting non-default entrypoints are specified in the method
319  // argument and the project.
320  //
321  // TODO(cbracken): https://github.com/flutter/flutter/issues/109285
322  // The entrypoint method parameter should eventually be removed from this
323  // method and only the entrypoint specified in project_ should be used.
324  if (!project_->dart_entrypoint().empty() && !entrypoint.empty() &&
325  project_->dart_entrypoint() != entrypoint) {
326  FML_LOG(ERROR) << "Conflicting entrypoints were specified in "
327  "FlutterDesktopEngineProperties.dart_entrypoint and "
328  "FlutterDesktopEngineRun(engine, entry_point). ";
329  return false;
330  }
331  if (!entrypoint.empty()) {
332  args.custom_dart_entrypoint = entrypoint.data();
333  } else if (!project_->dart_entrypoint().empty()) {
334  args.custom_dart_entrypoint = project_->dart_entrypoint().c_str();
335  }
336  args.dart_entrypoint_argc = static_cast<int>(entrypoint_argv.size());
337  args.dart_entrypoint_argv =
338  entrypoint_argv.empty() ? nullptr : entrypoint_argv.data();
339  args.platform_message_callback =
340  [](const FlutterPlatformMessage* engine_message,
341  void* user_data) -> void {
342  auto host = static_cast<FlutterWindowsEngine*>(user_data);
343  return host->HandlePlatformMessage(engine_message);
344  };
345  args.vsync_callback = [](void* user_data, intptr_t baton) -> void {
346  auto host = static_cast<FlutterWindowsEngine*>(user_data);
347  host->OnVsync(baton);
348  };
349  args.on_pre_engine_restart_callback = [](void* user_data) {
350  auto host = static_cast<FlutterWindowsEngine*>(user_data);
351  host->OnPreEngineRestart();
352  };
353  args.update_semantics_callback2 = [](const FlutterSemanticsUpdate2* update,
354  void* user_data) {
355  auto host = static_cast<FlutterWindowsEngine*>(user_data);
356 
357  // TODO(loicsharma): Remove implicit view assumption.
358  // https://github.com/flutter/flutter/issues/142845
359  auto view = host->view(kImplicitViewId);
360  if (!view) {
361  return;
362  }
363 
364  auto accessibility_bridge = view->accessibility_bridge().lock();
365  if (!accessibility_bridge) {
366  return;
367  }
368 
369  for (size_t i = 0; i < update->node_count; i++) {
370  const FlutterSemanticsNode2* node = update->nodes[i];
371  accessibility_bridge->AddFlutterSemanticsNodeUpdate(*node);
372  }
373 
374  for (size_t i = 0; i < update->custom_action_count; i++) {
375  const FlutterSemanticsCustomAction2* action = update->custom_actions[i];
376  accessibility_bridge->AddFlutterSemanticsCustomActionUpdate(*action);
377  }
378 
379  accessibility_bridge->CommitUpdates();
380  };
381  args.root_isolate_create_callback = [](void* user_data) {
382  auto host = static_cast<FlutterWindowsEngine*>(user_data);
383  if (host->root_isolate_create_callback_) {
384  host->root_isolate_create_callback_();
385  }
386  };
387  args.channel_update_callback = [](const FlutterChannelUpdate* update,
388  void* user_data) {
389  auto host = static_cast<FlutterWindowsEngine*>(user_data);
390  if (SAFE_ACCESS(update, channel, nullptr) != nullptr) {
391  std::string channel_name(update->channel);
392  host->OnChannelUpdate(std::move(channel_name),
393  SAFE_ACCESS(update, listening, false));
394  }
395  };
396  args.view_focus_change_request_callback =
397  [](const FlutterViewFocusChangeRequest* request, void* user_data) {
398  auto host = static_cast<FlutterWindowsEngine*>(user_data);
399  host->OnViewFocusChangeRequest(request);
400  };
401 
402  args.custom_task_runners = &custom_task_runners;
403 
404  if (!platform_view_plugin_) {
405  platform_view_plugin_ = std::make_unique<PlatformViewPlugin>(
406  messenger_wrapper_.get(), task_runner_.get());
407  }
408  if (egl_manager_) {
409  auto resolver = [](const char* name) -> void* {
410  return reinterpret_cast<void*>(::eglGetProcAddress(name));
411  };
412 
413  // TODO(schectman) Pass the platform view manager to the compositor
414  // constructors: https://github.com/flutter/flutter/issues/143375
415  compositor_ =
416  std::make_unique<CompositorOpenGL>(this, resolver, enable_impeller_);
417  } else {
418  compositor_ = std::make_unique<CompositorSoftware>();
419  }
420 
421  FlutterCompositor compositor = {};
422  compositor.struct_size = sizeof(FlutterCompositor);
423  compositor.user_data = this;
424  compositor.create_backing_store_callback =
425  [](const FlutterBackingStoreConfig* config,
426  FlutterBackingStore* backing_store_out, void* user_data) -> bool {
427  auto host = static_cast<FlutterWindowsEngine*>(user_data);
428 
429  return host->compositor_->CreateBackingStore(*config, backing_store_out);
430  };
431 
432  compositor.collect_backing_store_callback =
433  [](const FlutterBackingStore* backing_store, void* user_data) -> bool {
434  auto host = static_cast<FlutterWindowsEngine*>(user_data);
435 
436  return host->compositor_->CollectBackingStore(backing_store);
437  };
438 
439  compositor.present_view_callback =
440  [](const FlutterPresentViewInfo* info) -> bool {
441  auto host = static_cast<FlutterWindowsEngine*>(info->user_data);
442 
443  return host->Present(info);
444  };
445  args.compositor = &compositor;
446 
447  if (aot_data_) {
448  args.aot_data = aot_data_.get();
449  }
450 
451  // The platform thread creates OpenGL contexts. These
452  // must be released to be used by the engine's threads.
453  FML_DCHECK(!egl_manager_ || !egl_manager_->HasContextCurrent());
454 
455  FlutterRendererConfig renderer_config;
456 
457  if (enable_impeller_) {
458  // Impeller does not support a Software backend. Avoid falling back and
459  // confusing the engine on which renderer is selected.
460  if (!egl_manager_) {
461  FML_LOG(ERROR) << "Could not create surface manager. Impeller backend "
462  "does not support software rendering.";
463  return false;
464  }
465  renderer_config = GetOpenGLRendererConfig();
466  } else {
467  renderer_config =
468  egl_manager_ ? GetOpenGLRendererConfig() : GetSoftwareRendererConfig();
469  }
470 
471  auto result = embedder_api_.Run(FLUTTER_ENGINE_VERSION, &renderer_config,
472  &args, this, &engine_);
473  if (result != kSuccess || engine_ == nullptr) {
474  FML_LOG(ERROR) << "Failed to start Flutter engine: error " << result;
475  return false;
476  }
477 
478  // Configure device frame rate displayed via devtools.
479  FlutterEngineDisplay display = {};
480  display.struct_size = sizeof(FlutterEngineDisplay);
481  display.display_id = 0;
482  display.single_display = true;
483  display.refresh_rate =
484  1.0 / (static_cast<double>(FrameInterval().count()) / 1000000000.0);
485 
486  std::vector<FlutterEngineDisplay> displays = {display};
487  embedder_api_.NotifyDisplayUpdate(engine_,
488  kFlutterEngineDisplaysUpdateTypeStartup,
489  displays.data(), displays.size());
490 
491  SendSystemLocales();
492  SetLifecycleState(flutter::AppLifecycleState::kResumed);
493 
494  settings_plugin_->StartWatching();
495  settings_plugin_->SendSettings();
496 
497  InitializeKeyboard();
498 
499  return true;
500 }
501 
502 bool FlutterWindowsEngine::Stop() {
503  if (engine_) {
504  for (const auto& [callback, registrar] :
505  plugin_registrar_destruction_callbacks_) {
506  callback(registrar);
507  }
508  FlutterEngineResult result = embedder_api_.Shutdown(engine_);
509  engine_ = nullptr;
510  return (result == kSuccess);
511  }
512  return false;
513 }
514 
515 std::unique_ptr<FlutterWindowsView> FlutterWindowsEngine::CreateView(
516  std::unique_ptr<WindowBindingHandler> window) {
517  auto view_id = next_view_id_;
518  auto view = std::make_unique<FlutterWindowsView>(
519  view_id, this, std::move(window), windows_proc_table_);
520 
521  view->CreateRenderSurface();
522 
523  next_view_id_++;
524 
525  {
526  // Add the view to the embedder. This must happen before the engine
527  // is notified the view exists and starts presenting to it.
528  std::unique_lock write_lock(views_mutex_);
529  FML_DCHECK(views_.find(view_id) == views_.end());
530  views_[view_id] = view.get();
531  }
532 
533  if (!view->IsImplicitView()) {
534  FML_DCHECK(running());
535 
536  struct Captures {
537  fml::AutoResetWaitableEvent latch;
538  bool added;
539  };
540  Captures captures = {};
541 
542  FlutterWindowMetricsEvent metrics = view->CreateWindowMetricsEvent();
543 
544  FlutterAddViewInfo info = {};
545  info.struct_size = sizeof(FlutterAddViewInfo);
546  info.view_id = view_id;
547  info.view_metrics = &metrics;
548  info.user_data = &captures;
549  info.add_view_callback = [](const FlutterAddViewResult* result) {
550  Captures* captures = reinterpret_cast<Captures*>(result->user_data);
551  captures->added = result->added;
552  captures->latch.Signal();
553  };
554 
555  FlutterEngineResult result = embedder_api_.AddView(engine_, &info);
556  if (result != kSuccess) {
557  FML_LOG(ERROR)
558  << "Starting the add view operation failed. FlutterEngineAddView "
559  "returned an unexpected result: "
560  << result << ". This indicates a bug in the Windows embedder.";
561  FML_DCHECK(false);
562  return nullptr;
563  }
564 
565  // Block the platform thread until the engine has added the view.
566  // TODO(loicsharma): This blocks the platform thread eagerly and can
567  // cause unnecessary delay in input processing. Instead, this should block
568  // lazily only when the app does an operation which needs the view.
569  // https://github.com/flutter/flutter/issues/146248
570  captures.latch.Wait();
571 
572  if (!captures.added) {
573  // Adding the view failed. Update the embedder's state to match the
574  // engine's state. This is unexpected and indicates a bug in the Windows
575  // embedder.
576  FML_LOG(ERROR) << "FlutterEngineAddView failed to add view";
577  std::unique_lock write_lock(views_mutex_);
578  views_.erase(view_id);
579  return nullptr;
580  }
581  }
582 
583  return std::move(view);
584 }
585 
586 void FlutterWindowsEngine::RemoveView(FlutterViewId view_id) {
587  FML_DCHECK(running());
588 
589  // Notify the engine to stop rendering to the view if it isn't the implicit
590  // view. The engine and framework assume the implicit view always exists and
591  // can continue presenting.
592  if (view_id != kImplicitViewId) {
593  struct Captures {
594  fml::AutoResetWaitableEvent latch;
595  bool removed;
596  };
597  Captures captures = {};
598 
599  FlutterRemoveViewInfo info = {};
600  info.struct_size = sizeof(FlutterRemoveViewInfo);
601  info.view_id = view_id;
602  info.user_data = &captures;
603  info.remove_view_callback = [](const FlutterRemoveViewResult* result) {
604  // This is invoked on an engine thread. If
605  // |FlutterRemoveViewResult.removed| is `true`, the engine guarantees the
606  // view won't be presented.
607  Captures* captures = reinterpret_cast<Captures*>(result->user_data);
608  captures->removed = result->removed;
609  captures->latch.Signal();
610  };
611 
612  FlutterEngineResult result = embedder_api_.RemoveView(engine_, &info);
613  if (result != kSuccess) {
614  FML_LOG(ERROR) << "Starting the remove view operation failed. "
615  "FlutterEngineRemoveView "
616  "returned an unexpected result: "
617  << result
618  << ". This indicates a bug in the Windows embedder.";
619  FML_DCHECK(false);
620  return;
621  }
622 
623  // Block the platform thread until the engine has removed the view.
624  // TODO(loicsharma): This blocks the platform thread eagerly and can
625  // cause unnecessary delay in input processing. Instead, this should block
626  // lazily only when an operation needs the view.
627  // https://github.com/flutter/flutter/issues/146248
628  captures.latch.Wait();
629 
630  if (!captures.removed) {
631  // Removing the view failed. This is unexpected and indicates a bug in the
632  // Windows embedder.
633  FML_LOG(ERROR) << "FlutterEngineRemoveView failed to remove view";
634  return;
635  }
636  }
637 
638  {
639  // The engine no longer presents to the view. Remove the view from the
640  // embedder.
641  std::unique_lock write_lock(views_mutex_);
642 
643  FML_DCHECK(views_.find(view_id) != views_.end());
644  views_.erase(view_id);
645  }
646 }
647 
648 void FlutterWindowsEngine::OnVsync(intptr_t baton) {
649  std::chrono::nanoseconds current_time =
650  std::chrono::nanoseconds(embedder_api_.GetCurrentTime());
651  std::chrono::nanoseconds frame_interval = FrameInterval();
652  auto next = SnapToNextTick(current_time, start_time_, frame_interval);
653  embedder_api_.OnVsync(engine_, baton, next.count(),
654  (next + frame_interval).count());
655 }
656 
657 std::chrono::nanoseconds FlutterWindowsEngine::FrameInterval() {
658  if (frame_interval_override_.has_value()) {
659  return frame_interval_override_.value();
660  }
661  uint64_t interval = 16600000;
662 
663  DWM_TIMING_INFO timing_info = {};
664  timing_info.cbSize = sizeof(timing_info);
665  HRESULT result = DwmGetCompositionTimingInfo(NULL, &timing_info);
666  if (result == S_OK && timing_info.rateRefresh.uiDenominator > 0 &&
667  timing_info.rateRefresh.uiNumerator > 0) {
668  interval = static_cast<double>(timing_info.rateRefresh.uiDenominator *
669  1000000000.0) /
670  static_cast<double>(timing_info.rateRefresh.uiNumerator);
671  }
672 
673  return std::chrono::nanoseconds(interval);
674 }
675 
676 FlutterWindowsView* FlutterWindowsEngine::view(FlutterViewId view_id) const {
677  std::shared_lock read_lock(views_mutex_);
678 
679  auto iterator = views_.find(view_id);
680  if (iterator == views_.end()) {
681  return nullptr;
682  }
683 
684  return iterator->second;
685 }
686 
687 // Returns the currently configured Plugin Registrar.
688 FlutterDesktopPluginRegistrarRef FlutterWindowsEngine::GetRegistrar() {
689  return plugin_registrar_.get();
690 }
691 
692 void FlutterWindowsEngine::AddPluginRegistrarDestructionCallback(
695  plugin_registrar_destruction_callbacks_[callback] = registrar;
696 }
697 
698 void FlutterWindowsEngine::SendWindowMetricsEvent(
699  const FlutterWindowMetricsEvent& event) {
700  if (engine_) {
701  embedder_api_.SendWindowMetricsEvent(engine_, &event);
702  }
703 }
704 
705 void FlutterWindowsEngine::SendPointerEvent(const FlutterPointerEvent& event) {
706  if (engine_) {
707  embedder_api_.SendPointerEvent(engine_, &event, 1);
708  }
709 }
710 
711 void FlutterWindowsEngine::SendKeyEvent(const FlutterKeyEvent& event,
712  FlutterKeyEventCallback callback,
713  void* user_data) {
714  if (engine_) {
715  embedder_api_.SendKeyEvent(engine_, &event, callback, user_data);
716  }
717 }
718 
719 void FlutterWindowsEngine::SendViewFocusEvent(
720  const FlutterViewFocusEvent& event) {
721  if (engine_) {
722  embedder_api_.SendViewFocusEvent(engine_, &event);
723  }
724 }
725 
726 bool FlutterWindowsEngine::SendPlatformMessage(
727  const char* channel,
728  const uint8_t* message,
729  const size_t message_size,
730  const FlutterDesktopBinaryReply reply,
731  void* user_data) {
732  FlutterPlatformMessageResponseHandle* response_handle = nullptr;
733  if (reply != nullptr && user_data != nullptr) {
734  FlutterEngineResult result =
735  embedder_api_.PlatformMessageCreateResponseHandle(
736  engine_, reply, user_data, &response_handle);
737  if (result != kSuccess) {
738  FML_LOG(ERROR) << "Failed to create response handle";
739  return false;
740  }
741  }
742 
743  FlutterPlatformMessage platform_message = {
744  sizeof(FlutterPlatformMessage),
745  channel,
746  message,
747  message_size,
748  response_handle,
749  };
750 
751  FlutterEngineResult message_result =
752  embedder_api_.SendPlatformMessage(engine_, &platform_message);
753  if (response_handle != nullptr) {
754  embedder_api_.PlatformMessageReleaseResponseHandle(engine_,
755  response_handle);
756  }
757  return message_result == kSuccess;
758 }
759 
760 void FlutterWindowsEngine::SendPlatformMessageResponse(
762  const uint8_t* data,
763  size_t data_length) {
764  embedder_api_.SendPlatformMessageResponse(engine_, handle, data, data_length);
765 }
766 
767 void FlutterWindowsEngine::HandlePlatformMessage(
768  const FlutterPlatformMessage* engine_message) {
769  if (engine_message->struct_size != sizeof(FlutterPlatformMessage)) {
770  FML_LOG(ERROR) << "Invalid message size received. Expected: "
771  << sizeof(FlutterPlatformMessage) << " but received "
772  << engine_message->struct_size;
773  return;
774  }
775 
776  auto message = ConvertToDesktopMessage(*engine_message);
777 
778  message_dispatcher_->HandleMessage(message, [this] {}, [this] {});
779 }
780 
781 void FlutterWindowsEngine::ReloadSystemFonts() {
782  embedder_api_.ReloadSystemFonts(engine_);
783 }
784 
785 void FlutterWindowsEngine::ScheduleFrame() {
786  embedder_api_.ScheduleFrame(engine_);
787 }
788 
789 void FlutterWindowsEngine::SetNextFrameCallback(fml::closure callback) {
790  next_frame_callback_ = std::move(callback);
791 
792  embedder_api_.SetNextFrameCallback(
793  engine_,
794  [](void* user_data) {
795  // Embedder callback runs on raster thread. Switch back to platform
796  // thread.
797  FlutterWindowsEngine* self =
798  static_cast<FlutterWindowsEngine*>(user_data);
799 
800  self->task_runner_->PostTask(std::move(self->next_frame_callback_));
801  },
802  this);
803 }
804 
805 void FlutterWindowsEngine::SetLifecycleState(flutter::AppLifecycleState state) {
806  if (lifecycle_manager_) {
807  lifecycle_manager_->SetLifecycleState(state);
808  }
809 }
810 
811 void FlutterWindowsEngine::SendSystemLocales() {
812  std::vector<LanguageInfo> languages =
813  GetPreferredLanguageInfo(*windows_proc_table_);
814  std::vector<FlutterLocale> flutter_locales;
815  flutter_locales.reserve(languages.size());
816  for (const auto& info : languages) {
817  flutter_locales.push_back(CovertToFlutterLocale(info));
818  }
819  // Convert the locale list to the locale pointer list that must be provided.
820  std::vector<const FlutterLocale*> flutter_locale_list;
821  flutter_locale_list.reserve(flutter_locales.size());
822  std::transform(flutter_locales.begin(), flutter_locales.end(),
823  std::back_inserter(flutter_locale_list),
824  [](const auto& arg) -> const auto* { return &arg; });
825  embedder_api_.UpdateLocales(engine_, flutter_locale_list.data(),
826  flutter_locale_list.size());
827 }
828 
829 void FlutterWindowsEngine::InitializeKeyboard() {
830  auto internal_plugin_messenger = internal_plugin_registrar_->messenger();
831  KeyboardKeyEmbedderHandler::GetKeyStateHandler get_key_state = GetKeyState;
832  KeyboardKeyEmbedderHandler::MapVirtualKeyToScanCode map_vk_to_scan =
833  [](UINT virtual_key, bool extended) {
834  return MapVirtualKey(virtual_key,
835  extended ? MAPVK_VK_TO_VSC_EX : MAPVK_VK_TO_VSC);
836  };
837  keyboard_key_handler_ = std::move(CreateKeyboardKeyHandler(
838  internal_plugin_messenger, get_key_state, map_vk_to_scan));
839  text_input_plugin_ =
840  std::move(CreateTextInputPlugin(internal_plugin_messenger));
841 }
842 
843 std::unique_ptr<KeyboardHandlerBase>
844 FlutterWindowsEngine::CreateKeyboardKeyHandler(
845  BinaryMessenger* messenger,
848  auto keyboard_key_handler = std::make_unique<KeyboardKeyHandler>(messenger);
849  keyboard_key_handler->AddDelegate(
850  std::make_unique<KeyboardKeyEmbedderHandler>(
851  [this](const FlutterKeyEvent& event, FlutterKeyEventCallback callback,
852  void* user_data) {
853  return SendKeyEvent(event, callback, user_data);
854  },
855  get_key_state, map_vk_to_scan));
856  keyboard_key_handler->AddDelegate(
857  std::make_unique<KeyboardKeyChannelHandler>(messenger));
858  keyboard_key_handler->InitKeyboardChannel();
859  return keyboard_key_handler;
860 }
861 
862 std::unique_ptr<TextInputPlugin> FlutterWindowsEngine::CreateTextInputPlugin(
863  BinaryMessenger* messenger) {
864  return std::make_unique<TextInputPlugin>(messenger, this);
865 }
866 
867 bool FlutterWindowsEngine::RegisterExternalTexture(int64_t texture_id) {
868  return (embedder_api_.RegisterExternalTexture(engine_, texture_id) ==
869  kSuccess);
870 }
871 
872 bool FlutterWindowsEngine::UnregisterExternalTexture(int64_t texture_id) {
873  return (embedder_api_.UnregisterExternalTexture(engine_, texture_id) ==
874  kSuccess);
875 }
876 
877 bool FlutterWindowsEngine::MarkExternalTextureFrameAvailable(
878  int64_t texture_id) {
879  return (embedder_api_.MarkExternalTextureFrameAvailable(
880  engine_, texture_id) == kSuccess);
881 }
882 
883 bool FlutterWindowsEngine::PostRasterThreadTask(fml::closure callback) const {
884  struct Captures {
885  fml::closure callback;
886  };
887  auto captures = new Captures();
888  captures->callback = std::move(callback);
889  if (embedder_api_.PostRenderThreadTask(
890  engine_,
891  [](void* opaque) {
892  auto captures = reinterpret_cast<Captures*>(opaque);
893  captures->callback();
894  delete captures;
895  },
896  captures) == kSuccess) {
897  return true;
898  }
899  delete captures;
900  return false;
901 }
902 
903 bool FlutterWindowsEngine::DispatchSemanticsAction(
904  uint64_t target,
905  FlutterSemanticsAction action,
906  fml::MallocMapping data) {
907  return (embedder_api_.DispatchSemanticsAction(engine_, target, action,
908  data.GetMapping(),
909  data.GetSize()) == kSuccess);
910 }
911 
912 void FlutterWindowsEngine::UpdateSemanticsEnabled(bool enabled) {
913  if (engine_ && semantics_enabled_ != enabled) {
914  std::shared_lock read_lock(views_mutex_);
915 
916  semantics_enabled_ = enabled;
917  embedder_api_.UpdateSemanticsEnabled(engine_, enabled);
918  for (auto iterator = views_.begin(); iterator != views_.end(); iterator++) {
919  iterator->second->UpdateSemanticsEnabled(enabled);
920  }
921  }
922 }
923 
924 void FlutterWindowsEngine::OnPreEngineRestart() {
925  // Reset the keyboard's state on hot restart.
926  InitializeKeyboard();
927 }
928 
929 std::string FlutterWindowsEngine::GetExecutableName() const {
930  std::pair<bool, std::string> result = fml::paths::GetExecutablePath();
931  if (result.first) {
932  const std::string& executable_path = result.second;
933  size_t last_separator = executable_path.find_last_of("/\\");
934  if (last_separator == std::string::npos ||
935  last_separator == executable_path.size() - 1) {
936  return executable_path;
937  }
938  return executable_path.substr(last_separator + 1);
939  }
940  return "Flutter";
941 }
942 
943 void FlutterWindowsEngine::UpdateAccessibilityFeatures() {
944  UpdateHighContrastMode();
945 }
946 
947 void FlutterWindowsEngine::UpdateHighContrastMode() {
948  high_contrast_enabled_ = windows_proc_table_->GetHighContrastEnabled();
949 
950  SendAccessibilityFeatures();
951  settings_plugin_->UpdateHighContrastMode(high_contrast_enabled_);
952 }
953 
954 void FlutterWindowsEngine::SendAccessibilityFeatures() {
955  int flags = 0;
956 
957  if (high_contrast_enabled_) {
958  flags |=
959  FlutterAccessibilityFeature::kFlutterAccessibilityFeatureHighContrast;
960  }
961 
962  embedder_api_.UpdateAccessibilityFeatures(
963  engine_, static_cast<FlutterAccessibilityFeature>(flags));
964 }
965 
966 void FlutterWindowsEngine::RequestApplicationQuit(HWND hwnd,
967  WPARAM wparam,
968  LPARAM lparam,
969  AppExitType exit_type) {
970  platform_handler_->RequestAppExit(hwnd, wparam, lparam, exit_type, 0);
971 }
972 
973 void FlutterWindowsEngine::OnQuit(std::optional<HWND> hwnd,
974  std::optional<WPARAM> wparam,
975  std::optional<LPARAM> lparam,
976  UINT exit_code) {
977  lifecycle_manager_->Quit(hwnd, wparam, lparam, exit_code);
978 }
979 
980 void FlutterWindowsEngine::OnDwmCompositionChanged() {
981  std::shared_lock read_lock(views_mutex_);
982 
983  for (auto iterator = views_.begin(); iterator != views_.end(); iterator++) {
984  iterator->second->OnDwmCompositionChanged();
985  }
986 }
987 
988 void FlutterWindowsEngine::OnWindowStateEvent(HWND hwnd,
989  WindowStateEvent event) {
990  lifecycle_manager_->OnWindowStateEvent(hwnd, event);
991 }
992 
993 std::optional<LRESULT> FlutterWindowsEngine::ProcessExternalWindowMessage(
994  HWND hwnd,
995  UINT message,
996  WPARAM wparam,
997  LPARAM lparam) {
998  if (lifecycle_manager_) {
999  return lifecycle_manager_->ExternalWindowMessage(hwnd, message, wparam,
1000  lparam);
1001  }
1002  return std::nullopt;
1003 }
1004 
1005 void FlutterWindowsEngine::OnChannelUpdate(std::string name, bool listening) {
1006  if (name == "flutter/platform" && listening) {
1007  lifecycle_manager_->BeginProcessingExit();
1008  } else if (name == "flutter/lifecycle" && listening) {
1009  lifecycle_manager_->BeginProcessingLifecycle();
1010  }
1011 }
1012 
1013 void FlutterWindowsEngine::OnViewFocusChangeRequest(
1014  const FlutterViewFocusChangeRequest* request) {
1015  std::shared_lock read_lock(views_mutex_);
1016 
1017  auto iterator = views_.find(request->view_id);
1018  if (iterator == views_.end()) {
1019  return;
1020  }
1021 
1022  FlutterWindowsView* view = iterator->second;
1023  view->Focus();
1024 }
1025 
1026 bool FlutterWindowsEngine::Present(const FlutterPresentViewInfo* info) {
1027  // This runs on the raster thread. Lock the views map for the entirety of the
1028  // present operation to block the platform thread from destroying the
1029  // view during the present.
1030  std::shared_lock read_lock(views_mutex_);
1031 
1032  auto iterator = views_.find(info->view_id);
1033  if (iterator == views_.end()) {
1034  return false;
1035  }
1036 
1037  FlutterWindowsView* view = iterator->second;
1038 
1039  return compositor_->Present(view, info->layers, info->layers_count);
1040 }
1041 
1042 } // namespace flutter
flutter::egl::GpuPreference
GpuPreference
Definition: manager.h:32
flutter::kImplicitViewId
constexpr FlutterViewId kImplicitViewId
Definition: flutter_windows_engine.h:55
flutter::WindowStateEvent
WindowStateEvent
An event representing a change in window state that may update the.
Definition: windows_lifecycle_manager.h:24
flutter::FlutterProjectBundle
Definition: flutter_project_bundle.h:32
flutter::AppExitType
AppExitType
Definition: platform_handler.h:27
flutter::FlutterWindowsEngine::view
FlutterWindowsView * view(FlutterViewId view_id) const
Definition: flutter_windows_engine.cc:676
flutter::FlutterWindowsView
Definition: flutter_windows_view.h:34
extended
bool extended
Definition: keyboard_key_handler_unittests.cc:118
flutter::GetPreferredLanguageInfo
std::vector< LanguageInfo > GetPreferredLanguageInfo(const WindowsProcTable &windows_proc_table)
Definition: system_utils.cc:15
FlutterDesktopBinaryReply
void(* FlutterDesktopBinaryReply)(const uint8_t *data, size_t data_size, void *user_data)
Definition: flutter_messenger.h:26
flutter::FlutterWindowsEngine::OnVsync
void OnVsync(intptr_t baton)
Definition: flutter_windows_engine.cc:648
user_data
void * user_data
Definition: flutter_windows_view_unittests.cc:53
flutter::FlutterWindowsEngine
Definition: flutter_windows_engine.h:90
flutter::KeyboardKeyEmbedderHandler::GetKeyStateHandler
std::function< SHORT(int)> GetKeyStateHandler
Definition: keyboard_key_embedder_handler.h:41
flutter::FlutterWindowsView::accessibility_bridge
std::weak_ptr< AccessibilityBridgeWindows > accessibility_bridge()
Definition: flutter_windows_view.h:246
flutter::FlutterWindowsView::Focus
virtual bool Focus()
Definition: flutter_windows_view.cc:844
flutter::FlutterWindowsEngine::FlutterWindowsEngine
FlutterWindowsEngine(const FlutterProjectBundle &project, std::shared_ptr< WindowsProcTable > windows_proc_table=nullptr)
Definition: flutter_windows_engine.cc:149
FlutterDesktopMessageResponseHandle
struct _FlutterPlatformMessageResponseHandle FlutterDesktopMessageResponseHandle
Definition: flutter_messenger.h:22
flutter::egl::Manager::Create
static std::unique_ptr< Manager > Create(GpuPreference gpu_preference)
Definition: manager.cc:17
kAccessibilityChannelName
static constexpr char kAccessibilityChannelName[]
Definition: flutter_windows_engine.cc:34
system_utils.h
flutter::FlutterWindowsEngine::SetSwitches
void SetSwitches(const std::vector< std::string > &switches)
Definition: flutter_windows_engine.cc:240
flutter::FlutterWindowsEngine::OnPreEngineRestart
void OnPreEngineRestart()
Definition: flutter_windows_engine.cc:924
flutter::KeyboardKeyEmbedderHandler::MapVirtualKeyToScanCode
std::function< SHORT(UINT, bool)> MapVirtualKeyToScanCode
Definition: keyboard_key_embedder_handler.h:43
flutter::FlutterDesktopMessenger
Definition: flutter_desktop_messenger.h:23
flutter::BinaryMessenger
Definition: binary_messenger.h:28
path_utils.h
flutter::FlutterWindowsEngine::OnViewFocusChangeRequest
virtual void OnViewFocusChangeRequest(const FlutterViewFocusChangeRequest *request)
Definition: flutter_windows_engine.cc:1013
flutter_windows_view.h
standard_message_codec.h
accessibility_bridge_windows.h
flutter_project_bundle.h
compositor_opengl.h
flutter::FlutterViewId
int64_t FlutterViewId
Definition: flutter_view.h:13
binary_messenger_impl.h
flutter
Definition: accessibility_bridge_windows.cc:11
flutter::TaskRunner
Definition: task_runner.h:26
compositor_software.h
FlutterDesktopOnPluginRegistrarDestroyed
void(* FlutterDesktopOnPluginRegistrarDestroyed)(FlutterDesktopPluginRegistrarRef)
Definition: flutter_plugin_registrar.h:23
flutter::egl::ProcTable::Create
static std::shared_ptr< ProcTable > Create()
Definition: proc_table.cc:12
flutter_windows_engine.h
FlutterDesktopMessage
Definition: flutter_messenger.h:31
flutter::AppLifecycleState::kResumed
@ kResumed
RunOnPlatformThread
@ RunOnPlatformThread
Definition: flutter_windows.h:54
message
Win32Message message
Definition: keyboard_unittests.cc:137
action
int action
Definition: keyboard_key_handler_unittests.cc:116
flutter::AppLifecycleState
AppLifecycleState
Definition: app_lifecycle_state.h:32
task_runner.h
flutter::WindowsLifecycleManager
Definition: windows_lifecycle_manager.h:37
FlutterDesktopPluginRegistrar
Definition: window_state.h:23
keyboard_key_channel_handler.h
flutter::FlutterWindowsEngine::HandlePlatformMessage
void HandlePlatformMessage(const FlutterPlatformMessage *)
Definition: flutter_windows_engine.cc:767
flutter::WindowsPlatformThreadPrioritySetter
static void WindowsPlatformThreadPrioritySetter(FlutterThreadPriority priority)
Definition: flutter_windows_engine.h:60
texture_id
uint32_t texture_id
Definition: compositor_opengl.cc:20
flutter::AccessibilityPlugin::SetUp
static void SetUp(BinaryMessenger *binary_messenger, AccessibilityPlugin *plugin)
Definition: accessibility_plugin.cc:76
callback
FlutterDesktopBinaryReply callback
Definition: flutter_windows_view_unittests.cc:52