Flutter Linux Embedder
fl_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 <epoxy/egl.h>
8 #include <gmodule.h>
9 
10 #include <cstring>
11 
12 #include "flutter/common/constants.h"
14 #include "flutter/shell/platform/embedder/embedder.h"
30 
31 // Unique number associated with platform tasks.
32 static constexpr size_t kPlatformTaskRunnerIdentifier = 1;
33 
34 // Use different device ID for mouse and pan/zoom events, since we can't
35 // differentiate the actual device (mouse v.s. trackpad)
36 static constexpr int32_t kMousePointerDeviceId = 0;
37 static constexpr int32_t kPointerPanZoomDeviceId = 1;
38 
39 struct _FlEngine {
40  GObject parent_instance;
41 
42  // Thread the GLib main loop is running on.
43  GThread* thread;
44 
45  // The project this engine is running.
46  FlDartProject* project;
47 
48  // Watches for monitors changes to update engine.
49  FlDisplayMonitor* display_monitor;
50 
51  // Type of rendering performed.
52  FlutterRendererType renderer_type;
53 
54  // Manages OpenGL contexts.
55  FlOpenGLManager* opengl_manager;
56 
57  // Messenger used to send and receive platform messages.
58  FlBinaryMessenger* binary_messenger;
59 
60  // Implements the flutter/settings channel.
61  FlSettingsHandler* settings_handler;
62 
63  // Implements the flutter/platform channel.
64  FlPlatformHandler* platform_handler;
65 
66  // Implements the flutter/accessibility channel.
67  FlAccessibilityHandler* accessibility_handler;
68 
69  // Process keyboard events.
70  FlKeyboardManager* keyboard_manager;
71 
72  // Implements the flutter/textinput channel.
73  FlTextInputHandler* text_input_handler;
74 
75  // Implements the flutter/keyboard channel.
76  FlKeyboardHandler* keyboard_handler;
77 
78  // Implements the flutter/mousecursor channel.
79  FlMouseCursorHandler* mouse_cursor_handler;
80 
81  // Manages textures rendered by native code.
82  FlTextureRegistrar* texture_registrar;
83 
84  // Schedules tasks to be run on the appropriate thread.
85  FlTaskRunner* task_runner;
86 
87  // Ahead of time data used to make engine run faster.
88  FlutterEngineAOTData aot_data;
89 
90  // The Flutter engine.
91  FLUTTER_API_SYMBOL(FlutterEngine) engine;
92 
93  // Function table for engine API, used to intercept engine calls for testing
94  // purposes.
95  FlutterEngineProcTable embedder_api;
96 
97  // Next ID to use for a view.
98  FlutterViewId next_view_id;
99 
100  // Objects rendering the views.
102 
103  // Mutex to protect access to renderables_by_view_id which is accessed by both
104  // engine threads and GTK.
106 
107  // Function to call when a platform message is received.
111 };
112 
113 G_DEFINE_QUARK(fl_engine_error_quark, fl_engine_error)
114 
116  FlPluginRegistryInterface* iface);
117 
119 
121 
123  FlEngine,
124  fl_engine,
125  G_TYPE_OBJECT,
126  G_IMPLEMENT_INTERFACE(fl_plugin_registry_get_type(),
128 
129 enum { PROP_0, PROP_BINARY_MESSENGER, PROP_LAST };
130 
131 // Parse a locale into its components.
132 static void parse_locale(const gchar* locale,
133  gchar** language,
134  gchar** territory,
135  gchar** codeset,
136  gchar** modifier) {
137  gchar* l = g_strdup(locale);
138 
139  // Locales are in the form "language[_territory][.codeset][@modifier]"
140  gchar* match = strrchr(l, '@');
141  if (match != nullptr) {
142  if (modifier != nullptr) {
143  *modifier = g_strdup(match + 1);
144  }
145  *match = '\0';
146  } else if (modifier != nullptr) {
147  *modifier = nullptr;
148  }
149 
150  match = strrchr(l, '.');
151  if (match != nullptr) {
152  if (codeset != nullptr) {
153  *codeset = g_strdup(match + 1);
154  }
155  *match = '\0';
156  } else if (codeset != nullptr) {
157  *codeset = nullptr;
158  }
159 
160  match = strrchr(l, '_');
161  if (match != nullptr) {
162  if (territory != nullptr) {
163  *territory = g_strdup(match + 1);
164  }
165  *match = '\0';
166  } else if (territory != nullptr) {
167  *territory = nullptr;
168  }
169 
170  if (language != nullptr) {
171  *language = l;
172  }
173 }
174 
175 /// Stores a weak reference to the renderable with the given ID.
176 static void set_renderable(FlEngine* self,
177  int64_t view_id,
178  FlRenderable* renderable) {
179  g_autoptr(GMutexLocker) locker = g_mutex_locker_new(&self->renderables_mutex);
180  GWeakRef* ref = g_new(GWeakRef, 1);
181  g_weak_ref_init(ref, G_OBJECT(renderable));
182  g_hash_table_insert(self->renderables_by_view_id, GINT_TO_POINTER(view_id),
183  ref);
184 }
185 
186 /// Returns the renderable with the given ID, or nullptr if no such view exists.
187 /// Returns a reference to the renderable.
188 static FlRenderable* get_renderable(FlEngine* self, int64_t view_id) {
189  g_autoptr(GMutexLocker) locker = g_mutex_locker_new(&self->renderables_mutex);
190  GWeakRef* ref = static_cast<GWeakRef*>(g_hash_table_lookup(
191  self->renderables_by_view_id, GINT_TO_POINTER(view_id)));
192  if (ref == nullptr) {
193  return nullptr;
194  }
195  return FL_RENDERABLE(g_weak_ref_get(ref));
196 }
197 
198 /// Remove a renderable that no longer exists.
199 static void remove_renderable(FlEngine* self, int64_t view_id) {
200  g_autoptr(GMutexLocker) locker = g_mutex_locker_new(&self->renderables_mutex);
201  g_hash_table_remove(self->renderables_by_view_id, GINT_TO_POINTER(view_id));
202 }
203 
204 static void view_added_cb(const FlutterAddViewResult* result) {
205  g_autoptr(GTask) task = G_TASK(result->user_data);
206 
207  if (result->added) {
208  g_task_return_boolean(task, TRUE);
209  } else {
210  g_task_return_new_error(task, fl_engine_error_quark(),
211  FL_ENGINE_ERROR_FAILED, "Failed to add view");
212  }
213 }
214 
215 static void view_removed_cb(const FlutterRemoveViewResult* result) {
216  g_autoptr(GTask) task = G_TASK(result->user_data);
217 
218  if (result->removed) {
219  g_task_return_boolean(task, TRUE);
220  } else {
221  g_task_return_new_error(task, fl_engine_error_quark(),
222  FL_ENGINE_ERROR_FAILED, "Failed to remove view");
223  }
224 }
225 
226 static void free_locale(FlutterLocale* locale) {
227  free(const_cast<gchar*>(locale->language_code));
228  free(const_cast<gchar*>(locale->country_code));
229  free(locale);
230 }
231 
232 // Passes locale information to the Flutter engine.
233 static void setup_locales(FlEngine* self) {
234  const gchar* const* languages = g_get_language_names();
235  g_autoptr(GPtrArray) locales_array = g_ptr_array_new_with_free_func(
236  reinterpret_cast<GDestroyNotify>(free_locale));
237  for (int i = 0; languages[i] != nullptr; i++) {
238  g_autofree gchar* locale_string = g_strstrip(g_strdup(languages[i]));
239 
240  // Ignore empty locales, caused by settings like `LANGUAGE=pt_BR:`
241  if (strcmp(locale_string, "") == 0) {
242  continue;
243  }
244 
245  g_autofree gchar* language = nullptr;
246  g_autofree gchar* territory = nullptr;
247  parse_locale(locale_string, &language, &territory, nullptr, nullptr);
248 
249  // Ignore duplicate locales, caused by settings like `LANGUAGE=C` (returns
250  // two "C") or `LANGUAGE=en:en`
251  gboolean has_locale = FALSE;
252  for (guint j = 0; !has_locale && j < locales_array->len; j++) {
253  FlutterLocale* locale =
254  reinterpret_cast<FlutterLocale*>(g_ptr_array_index(locales_array, j));
255  has_locale = g_strcmp0(locale->language_code, language) == 0 &&
256  g_strcmp0(locale->country_code, territory) == 0;
257  }
258  if (has_locale) {
259  continue;
260  }
261 
262  FlutterLocale* locale =
263  static_cast<FlutterLocale*>(g_malloc0(sizeof(FlutterLocale)));
264  g_ptr_array_add(locales_array, locale);
265  locale->struct_size = sizeof(FlutterLocale);
266  locale->language_code =
267  reinterpret_cast<const gchar*>(g_steal_pointer(&language));
268  locale->country_code =
269  reinterpret_cast<const gchar*>(g_steal_pointer(&territory));
270  locale->script_code = nullptr;
271  locale->variant_code = nullptr;
272  }
273  FlutterLocale** locales =
274  reinterpret_cast<FlutterLocale**>(locales_array->pdata);
275  FlutterEngineResult result = self->embedder_api.UpdateLocales(
276  self->engine, const_cast<const FlutterLocale**>(locales),
277  locales_array->len);
278  if (result != kSuccess) {
279  g_warning("Failed to set up Flutter locales");
280  }
281 }
282 
284  FlEngine* self,
285  const FlutterBackingStoreConfig* config,
286  FlutterBackingStore* backing_store_out) {
287  if (!fl_opengl_manager_make_current(self->opengl_manager)) {
288  return false;
289  }
290 
291  GLint sized_format = GL_RGBA8;
292  GLint general_format = GL_RGBA;
293  if (epoxy_has_gl_extension("GL_EXT_texture_format_BGRA8888")) {
294  sized_format = GL_BGRA8_EXT;
295  general_format = GL_BGRA_EXT;
296  }
297 
298  FlFramebuffer* framebuffer = fl_framebuffer_new(
299  general_format, config->size.width, config->size.height, FALSE);
300  if (!framebuffer) {
301  g_warning("Failed to create backing store");
302  return false;
303  }
304 
305  backing_store_out->type = kFlutterBackingStoreTypeOpenGL;
306  backing_store_out->open_gl.type = kFlutterOpenGLTargetTypeFramebuffer;
307  backing_store_out->open_gl.framebuffer.user_data = framebuffer;
308  backing_store_out->open_gl.framebuffer.name =
309  fl_framebuffer_get_id(framebuffer);
310  backing_store_out->open_gl.framebuffer.target = sized_format;
311  backing_store_out->open_gl.framebuffer.destruction_callback = [](void* p) {
312  // Backing store destroyed in fl_compositor_opengl_collect_backing_store(),
313  // set on FlutterCompositor.collect_backing_store_callback during engine
314  // start.
315  };
316 
317  return true;
318 }
319 
321  FlEngine* self,
322  const FlutterBackingStore* backing_store) {
323  if (!fl_opengl_manager_make_current(self->opengl_manager)) {
324  return false;
325  }
326 
327  // OpenGL context is required when destroying #FlFramebuffer.
328  g_object_unref(backing_store->open_gl.framebuffer.user_data);
329  return true;
330 }
331 
333  FlEngine* self,
334  const FlutterBackingStoreConfig* config,
335  FlutterBackingStore* backing_store_out) {
336  size_t allocation_length = config->size.width * config->size.height * 4;
337  uint8_t* allocation = static_cast<uint8_t*>(malloc(allocation_length));
338  if (allocation == nullptr) {
339  return false;
340  }
341 
342  backing_store_out->type = kFlutterBackingStoreTypeSoftware;
343  backing_store_out->software.allocation = allocation;
344  backing_store_out->software.height = config->size.height;
345  backing_store_out->software.row_bytes = config->size.width * 4;
346  backing_store_out->software.user_data = nullptr;
347  backing_store_out->software.destruction_callback = [](void* p) {
348  // Backing store destroyed in
349  // fl_compositor_software_collect_backing_store(), set on
350  // FlutterCompositor.collect_backing_store_callback during engine start.
351  };
352 
353  return true;
354 }
355 
357  FlEngine* self,
358  const FlutterBackingStore* backing_store) {
359  free(const_cast<void*>(backing_store->software.allocation));
360  return true;
361 }
362 
363 // Called when engine needs a backing store for a specific #FlutterLayer.
365  const FlutterBackingStoreConfig* config,
366  FlutterBackingStore* backing_store_out,
367  void* user_data) {
368  FlEngine* self = static_cast<FlEngine*>(user_data);
369  switch (self->renderer_type) {
370  case kOpenGL:
371  return create_opengl_backing_store(self, config, backing_store_out);
372  case kSoftware:
373  return create_software_backing_store(self, config, backing_store_out);
374  default:
375  return false;
376  }
377 }
378 
379 // Called when the backing store is to be released.
381  const FlutterBackingStore* backing_store,
382  void* user_data) {
383  FlEngine* self = static_cast<FlEngine*>(user_data);
384  switch (self->renderer_type) {
385  case kOpenGL:
386  return collect_opengl_backing_store(self, backing_store);
387  case kSoftware:
388  return collect_software_backing_store(self, backing_store);
389  default:
390  return false;
391  }
392 }
393 
394 // Called when embedder should composite contents of each layer onto the screen.
396  const FlutterPresentViewInfo* info) {
397  FlEngine* self = static_cast<FlEngine*>(info->user_data);
398 
399  g_autoptr(FlRenderable) renderable = get_renderable(self, info->view_id);
400  if (renderable == nullptr) {
401  return true;
402  }
403 
404  fl_renderable_present_layers(renderable, info->layers, info->layers_count);
405  return true;
406 }
407 
408 // Flutter engine rendering callbacks.
409 
410 static void* fl_engine_gl_proc_resolver(void* user_data, const char* name) {
411  return reinterpret_cast<void*>(eglGetProcAddress(name));
412 }
413 
415  FlEngine* self = static_cast<FlEngine*>(user_data);
416  return fl_opengl_manager_make_current(self->opengl_manager);
417 }
418 
420  FlEngine* self = static_cast<FlEngine*>(user_data);
421  return fl_opengl_manager_clear_current(self->opengl_manager);
422 }
423 
424 static uint32_t fl_engine_gl_get_fbo(void* user_data) {
425  // There is only one frame buffer object - always return that.
426  return 0;
427 }
428 
430  FlEngine* self = static_cast<FlEngine*>(user_data);
431  return fl_opengl_manager_make_resource_current(self->opengl_manager);
432 }
433 
434 // Called by the engine to retrieve an external texture.
436  void* user_data,
437  int64_t texture_id,
438  size_t width,
439  size_t height,
440  FlutterOpenGLTexture* opengl_texture) {
441  FlEngine* self = static_cast<FlEngine*>(user_data);
442  if (!self->texture_registrar) {
443  return false;
444  }
445 
446  FlTexture* texture =
447  fl_texture_registrar_lookup_texture(self->texture_registrar, texture_id);
448  if (texture == nullptr) {
449  g_warning("Unable to find texture %" G_GINT64_FORMAT, texture_id);
450  return false;
451  }
452 
453  gboolean result;
454  g_autoptr(GError) error = nullptr;
455  if (FL_IS_TEXTURE_GL(texture)) {
456  result = fl_texture_gl_populate(FL_TEXTURE_GL(texture), width, height,
457  opengl_texture, &error);
458  } else if (FL_IS_PIXEL_BUFFER_TEXTURE(texture)) {
459  result =
460  fl_pixel_buffer_texture_populate(FL_PIXEL_BUFFER_TEXTURE(texture),
461  width, height, opengl_texture, &error);
462  } else {
463  g_warning("Unsupported texture type %" G_GINT64_FORMAT, texture_id);
464  return false;
465  }
466 
467  if (!result) {
468  g_warning("%s", error->message);
469  return false;
470  }
471 
472  return true;
473 }
474 
475 // Called by the engine to determine if it is on the GTK thread.
477  FlEngine* self = static_cast<FlEngine*>(user_data);
478  return self->thread == g_thread_self();
479 }
480 
481 // Called when the engine has a task to perform in the GTK thread.
482 static void fl_engine_post_task(FlutterTask task,
483  uint64_t target_time_nanos,
484  void* user_data) {
485  FlEngine* self = static_cast<FlEngine*>(user_data);
486 
487  fl_task_runner_post_flutter_task(self->task_runner, task, target_time_nanos);
488 }
489 
490 // Called when a platform message is received from the engine.
491 static void fl_engine_platform_message_cb(const FlutterPlatformMessage* message,
492  void* user_data) {
493  FlEngine* self = FL_ENGINE(user_data);
494 
495  gboolean handled = FALSE;
496  if (self->platform_message_handler != nullptr) {
497  g_autoptr(GBytes) data =
498  g_bytes_new(message->message, message->message_size);
499  handled = self->platform_message_handler(
500  self, message->channel, data, message->response_handle,
501  self->platform_message_handler_data);
502  }
503 
504  if (!handled) {
505  fl_engine_send_platform_message_response(self, message->response_handle,
506  nullptr, nullptr);
507  }
508 }
509 
510 // Called when a semantic node update is received from the engine.
511 static void fl_engine_update_semantics_cb(const FlutterSemanticsUpdate2* update,
512  void* user_data) {
513  FlEngine* self = FL_ENGINE(user_data);
514 
515  g_signal_emit(self, fl_engine_signals[SIGNAL_UPDATE_SEMANTICS], 0, update);
516 }
517 
518 static void setup_keyboard(FlEngine* self) {
519  g_clear_object(&self->keyboard_manager);
520  self->keyboard_manager = fl_keyboard_manager_new(self);
521 
522  g_clear_object(&self->keyboard_handler);
523  self->keyboard_handler =
524  fl_keyboard_handler_new(self->binary_messenger, self->keyboard_manager);
525 
526  GtkWidget* widget =
527  self->text_input_handler != nullptr
528  ? fl_text_input_handler_get_widget(self->text_input_handler)
529  : nullptr;
530  g_clear_object(&self->text_input_handler);
531  self->text_input_handler = fl_text_input_handler_new(self->binary_messenger);
532  if (widget != nullptr) {
533  fl_text_input_handler_set_widget(self->text_input_handler, widget);
534  }
535 }
536 
537 // Called right before the engine is restarted.
538 //
539 // This method should reset states to as if the engine has just been started,
540 // which usually indicates the user has requested a hot restart (Shift-R in the
541 // Flutter CLI.)
543  FlEngine* self = FL_ENGINE(user_data);
544 
545  setup_keyboard(self);
546 
547  g_signal_emit(self, fl_engine_signals[SIGNAL_ON_PRE_ENGINE_RESTART], 0);
548 }
549 
550 // Called when a response to a sent platform message is received from the
551 // engine.
552 static void fl_engine_platform_message_response_cb(const uint8_t* data,
553  size_t data_length,
554  void* user_data) {
555  g_autoptr(GTask) task = G_TASK(user_data);
556  g_task_return_pointer(task, g_bytes_new(data, data_length),
557  reinterpret_cast<GDestroyNotify>(g_bytes_unref));
558 }
559 
560 // Implements FlPluginRegistry::get_registrar_for_plugin.
561 static FlPluginRegistrar* fl_engine_get_registrar_for_plugin(
562  FlPluginRegistry* registry,
563  const gchar* name) {
564  FlEngine* self = FL_ENGINE(registry);
565 
566  return fl_plugin_registrar_new(nullptr, self->binary_messenger,
567  self->texture_registrar);
568 }
569 
571  FlPluginRegistryInterface* iface) {
572  iface->get_registrar_for_plugin = fl_engine_get_registrar_for_plugin;
573 }
574 
575 static void fl_engine_set_property(GObject* object,
576  guint prop_id,
577  const GValue* value,
578  GParamSpec* pspec) {
579  FlEngine* self = FL_ENGINE(object);
580  switch (prop_id) {
581  case PROP_BINARY_MESSENGER:
582  g_set_object(&self->binary_messenger,
583  FL_BINARY_MESSENGER(g_value_get_object(value)));
584  break;
585  default:
586  G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
587  break;
588  }
589 }
590 
591 static void fl_engine_dispose(GObject* object) {
592  FlEngine* self = FL_ENGINE(object);
593 
594  if (self->engine != nullptr) {
595  if (self->embedder_api.Shutdown(self->engine) != kSuccess) {
596  g_warning("Failed to shutdown Flutter engine");
597  }
598  self->engine = nullptr;
599  }
600 
601  if (self->aot_data != nullptr) {
602  if (self->embedder_api.CollectAOTData(self->aot_data) != kSuccess) {
603  g_warning("Failed to send collect AOT data");
604  }
605  self->aot_data = nullptr;
606  }
607 
608  fl_binary_messenger_shutdown(self->binary_messenger);
609  fl_texture_registrar_shutdown(self->texture_registrar);
610 
611  g_clear_object(&self->project);
612  g_clear_object(&self->display_monitor);
613  g_clear_object(&self->opengl_manager);
614  g_clear_object(&self->texture_registrar);
615  g_clear_object(&self->binary_messenger);
616  g_clear_object(&self->settings_handler);
617  g_clear_object(&self->platform_handler);
618  g_clear_object(&self->accessibility_handler);
619  g_clear_object(&self->keyboard_manager);
620  g_clear_object(&self->text_input_handler);
621  g_clear_object(&self->keyboard_handler);
622  g_clear_object(&self->mouse_cursor_handler);
623  g_clear_object(&self->task_runner);
624  {
625  g_autoptr(GMutexLocker) locker =
626  g_mutex_locker_new(&self->renderables_mutex);
627  g_clear_pointer(&self->renderables_by_view_id, g_hash_table_unref);
628  }
629  g_mutex_clear(&self->renderables_mutex);
630 
631  if (self->platform_message_handler_destroy_notify) {
632  self->platform_message_handler_destroy_notify(
633  self->platform_message_handler_data);
634  }
635  self->platform_message_handler_data = nullptr;
636  self->platform_message_handler_destroy_notify = nullptr;
637 
638  G_OBJECT_CLASS(fl_engine_parent_class)->dispose(object);
639 }
640 
641 static void fl_engine_class_init(FlEngineClass* klass) {
642  G_OBJECT_CLASS(klass)->dispose = fl_engine_dispose;
643  G_OBJECT_CLASS(klass)->set_property = fl_engine_set_property;
644 
645  g_object_class_install_property(
646  G_OBJECT_CLASS(klass), PROP_BINARY_MESSENGER,
647  g_param_spec_object(
648  "binary-messenger", "messenger", "Binary messenger",
649  fl_binary_messenger_get_type(),
650  static_cast<GParamFlags>(G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY |
651  G_PARAM_STATIC_STRINGS)));
652 
654  "on-pre-engine-restart", fl_engine_get_type(), G_SIGNAL_RUN_LAST, 0,
655  nullptr, nullptr, nullptr, G_TYPE_NONE, 0);
657  "update-semantics", fl_engine_get_type(), G_SIGNAL_RUN_LAST, 0, nullptr,
658  nullptr, nullptr, G_TYPE_NONE, 1, G_TYPE_POINTER);
659 }
660 
661 static void fl_engine_init(FlEngine* self) {
662  self->thread = g_thread_self();
663 
664  self->embedder_api.struct_size = sizeof(FlutterEngineProcTable);
665  if (FlutterEngineGetProcAddresses(&self->embedder_api) != kSuccess) {
666  g_warning("Failed get get engine function pointers");
667  }
668 
669  self->opengl_manager = fl_opengl_manager_new();
670 
671  self->display_monitor =
672  fl_display_monitor_new(self, gdk_display_get_default());
673  self->task_runner = fl_task_runner_new(self);
674 
675  // Implicit view is 0, so start at 1.
676  self->next_view_id = 1;
677  g_mutex_init(&self->renderables_mutex);
678  self->renderables_by_view_id = g_hash_table_new_full(
679  g_direct_hash, g_direct_equal, nullptr, [](gpointer value) {
680  GWeakRef* ref = static_cast<GWeakRef*>(value);
681  g_weak_ref_clear(ref);
682  free(ref);
683  });
684 
685  self->texture_registrar = fl_texture_registrar_new(self);
686 }
687 
688 static FlEngine* fl_engine_new_full(FlDartProject* project,
689  FlBinaryMessenger* binary_messenger) {
690  g_return_val_if_fail(FL_IS_DART_PROJECT(project), nullptr);
691 
692  FlEngine* self = FL_ENGINE(g_object_new(fl_engine_get_type(), nullptr));
693 
694  self->project = FL_DART_PROJECT(g_object_ref(project));
695  const gchar* renderer = g_getenv("FLUTTER_LINUX_RENDERER");
696  if (g_strcmp0(renderer, "software") == 0) {
697  self->renderer_type = kSoftware;
698  g_warning(
699  "Using the software renderer. Not all features are supported. This is "
700  "not recommended.\n"
701  "\n"
702  "To switch back to the default renderer, unset the "
703  "FLUTTER_LINUX_RENDERER environment variable.");
704  } else {
705  if (renderer != nullptr && strcmp(renderer, "opengl") != 0) {
706  g_warning("Unknown renderer type '%s', defaulting to opengl", renderer);
707  }
708  self->renderer_type = kOpenGL;
709  }
710 
711  if (binary_messenger != nullptr) {
712  self->binary_messenger =
713  FL_BINARY_MESSENGER(g_object_ref(binary_messenger));
714  } else {
715  self->binary_messenger = fl_binary_messenger_new(self);
716  }
717  self->keyboard_manager = fl_keyboard_manager_new(self);
718  self->mouse_cursor_handler =
719  fl_mouse_cursor_handler_new(self->binary_messenger);
720  self->accessibility_handler = fl_accessibility_handler_new(self);
721 
722  return self;
723 }
724 
725 FlEngine* fl_engine_for_id(int64_t id) {
726  void* engine = reinterpret_cast<void*>(id);
727  g_return_val_if_fail(FL_IS_ENGINE(engine), nullptr);
728  return FL_ENGINE(engine);
729 }
730 
731 G_MODULE_EXPORT FlEngine* fl_engine_new(FlDartProject* project) {
732  return fl_engine_new_full(project, nullptr);
733 }
734 
736  FlBinaryMessenger* binary_messenger) {
737  g_autoptr(FlDartProject) project = fl_dart_project_new();
738  return fl_engine_new_full(project, binary_messenger);
739 }
740 
741 G_MODULE_EXPORT FlEngine* fl_engine_new_headless(FlDartProject* project) {
742  return fl_engine_new(project);
743 }
744 
745 FlutterRendererType fl_engine_get_renderer_type(FlEngine* self) {
746  g_return_val_if_fail(FL_IS_ENGINE(self), static_cast<FlutterRendererType>(0));
747  return self->renderer_type;
748 }
749 
750 FlOpenGLManager* fl_engine_get_opengl_manager(FlEngine* self) {
751  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
752  return self->opengl_manager;
753 }
754 
755 FlDisplayMonitor* fl_engine_get_display_monitor(FlEngine* self) {
756  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
757  return self->display_monitor;
758 }
759 
760 gboolean fl_engine_start(FlEngine* self, GError** error) {
761  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
762 
763  FlutterRendererConfig config = {};
764  config.type = self->renderer_type;
765  switch (config.type) {
766  case kSoftware:
767  config.software.struct_size = sizeof(FlutterSoftwareRendererConfig);
768  // No action required, as this is handled in
769  // compositor_present_view_callback.
770  config.software.surface_present_callback =
771  [](void* user_data, const void* allocation, size_t row_bytes,
772  size_t height) { return true; };
773  break;
774  case kOpenGL:
775  config.open_gl.struct_size = sizeof(FlutterOpenGLRendererConfig);
776  config.open_gl.gl_proc_resolver = fl_engine_gl_proc_resolver;
777  config.open_gl.make_current = fl_engine_gl_make_current;
778  config.open_gl.clear_current = fl_engine_gl_clear_current;
779  config.open_gl.fbo_callback = fl_engine_gl_get_fbo;
780  // No action required, as this is handled in
781  // compositor_present_view_callback.
782  config.open_gl.present = [](void* user_data) { return true; };
783  config.open_gl.make_resource_current = fl_engine_gl_make_resource_current;
784  config.open_gl.gl_external_texture_frame_callback =
786  break;
787  case kMetal:
788  case kVulkan:
789  default:
791  "Unsupported renderer type");
792  return FALSE;
793  }
794 
795  FlutterTaskRunnerDescription platform_task_runner = {};
796  platform_task_runner.struct_size = sizeof(FlutterTaskRunnerDescription);
797  platform_task_runner.user_data = self;
798  platform_task_runner.runs_task_on_current_thread_callback =
800  platform_task_runner.post_task_callback = fl_engine_post_task;
801  platform_task_runner.identifier = kPlatformTaskRunnerIdentifier;
802 
803  FlutterCustomTaskRunners custom_task_runners = {};
804  custom_task_runners.struct_size = sizeof(FlutterCustomTaskRunners);
805  custom_task_runners.platform_task_runner = &platform_task_runner;
806 
807  switch (fl_dart_project_get_ui_thread_policy(self->project)) {
809  break;
812  custom_task_runners.ui_task_runner = &platform_task_runner;
813  break;
814  }
815 
816  g_autoptr(GPtrArray) command_line_args =
817  g_ptr_array_new_with_free_func(g_free);
818  g_ptr_array_insert(command_line_args, 0, g_strdup("flutter"));
819  for (const auto& env_switch : flutter::GetSwitchesFromEnvironment()) {
820  g_ptr_array_add(command_line_args, g_strdup(env_switch.c_str()));
821  }
822 
823  gchar** dart_entrypoint_args =
825 
826  FlutterProjectArgs args = {};
827  args.struct_size = sizeof(FlutterProjectArgs);
828  args.assets_path = fl_dart_project_get_assets_path(self->project);
829  args.icu_data_path = fl_dart_project_get_icu_data_path(self->project);
830  args.command_line_argc = command_line_args->len;
831  args.command_line_argv =
832  reinterpret_cast<const char* const*>(command_line_args->pdata);
833  args.platform_message_callback = fl_engine_platform_message_cb;
834  args.update_semantics_callback2 = fl_engine_update_semantics_cb;
835  args.custom_task_runners = &custom_task_runners;
836  args.shutdown_dart_vm_when_done = true;
837  args.on_pre_engine_restart_callback = fl_engine_on_pre_engine_restart_cb;
838  args.dart_entrypoint_argc =
839  dart_entrypoint_args != nullptr ? g_strv_length(dart_entrypoint_args) : 0;
840  args.dart_entrypoint_argv =
841  reinterpret_cast<const char* const*>(dart_entrypoint_args);
842  args.engine_id = reinterpret_cast<int64_t>(self);
843 
844  FlutterCompositor compositor = {};
845  compositor.struct_size = sizeof(FlutterCompositor);
846  compositor.user_data = self;
847  compositor.create_backing_store_callback =
849  compositor.collect_backing_store_callback =
851  compositor.present_view_callback = compositor_present_view_callback;
852  args.compositor = &compositor;
853 
854  if (self->embedder_api.RunsAOTCompiledDartCode()) {
855  FlutterEngineAOTDataSource source = {};
856  source.type = kFlutterEngineAOTDataSourceTypeElfPath;
857  source.elf_path = fl_dart_project_get_aot_library_path(self->project);
858  if (self->embedder_api.CreateAOTData(&source, &self->aot_data) !=
859  kSuccess) {
861  "Failed to create AOT data");
862  return FALSE;
863  }
864  args.aot_data = self->aot_data;
865  }
866 
867  FlutterEngineResult result = self->embedder_api.Initialize(
868  FLUTTER_ENGINE_VERSION, &config, &args, self, &self->engine);
869  if (result != kSuccess) {
871  "Failed to initialize Flutter engine");
872  return FALSE;
873  }
874 
875  result = self->embedder_api.RunInitialized(self->engine);
876  if (result != kSuccess) {
878  "Failed to run Flutter engine");
879  return FALSE;
880  }
881 
882  setup_locales(self);
883 
884  g_autoptr(FlSettings) settings = fl_settings_new();
885  self->settings_handler = fl_settings_handler_new(self);
886  fl_settings_handler_start(self->settings_handler, settings);
887 
888  self->platform_handler = fl_platform_handler_new(self->binary_messenger);
889 
890  setup_keyboard(self);
891 
892  result = self->embedder_api.UpdateSemanticsEnabled(self->engine, TRUE);
893  if (result != kSuccess) {
894  g_warning("Failed to enable accessibility features on Flutter engine");
895  }
896 
897  fl_display_monitor_start(self->display_monitor);
898 
899  return TRUE;
900 }
901 
902 FlutterEngineProcTable* fl_engine_get_embedder_api(FlEngine* self) {
903  return &(self->embedder_api);
904 }
905 
907  const FlutterEngineDisplay* displays,
908  size_t displays_length) {
909  g_return_if_fail(FL_IS_ENGINE(self));
910 
911  FlutterEngineResult result = self->embedder_api.NotifyDisplayUpdate(
912  self->engine, kFlutterEngineDisplaysUpdateTypeStartup, displays,
913  displays_length);
914  if (result != kSuccess) {
915  g_warning("Failed to notify display update to Flutter engine: %d", result);
916  }
917 }
918 
919 void fl_engine_set_implicit_view(FlEngine* self, FlRenderable* renderable) {
920  set_renderable(self, flutter::kFlutterImplicitViewId, renderable);
921 }
922 
923 FlutterViewId fl_engine_add_view(FlEngine* self,
924  FlRenderable* renderable,
925  size_t min_width,
926  size_t min_height,
927  size_t max_width,
928  size_t max_height,
929  double pixel_ratio,
930  GCancellable* cancellable,
931  GAsyncReadyCallback callback,
932  gpointer user_data) {
933  g_return_val_if_fail(FL_IS_ENGINE(self), -1);
934 
935  g_autoptr(GTask) task = g_task_new(self, cancellable, callback, user_data);
936 
937  FlutterViewId view_id = self->next_view_id;
938  self->next_view_id++;
939 
941 
942  // We don't know which display this view will open on, so set to zero and this
943  // will be updated in a following FlutterWindowMetricsEvent
944  FlutterEngineDisplayId display_id = 0;
945 
946  FlutterWindowMetricsEvent metrics = {};
947  metrics.struct_size = sizeof(FlutterWindowMetricsEvent);
948  metrics.width = min_width;
949  metrics.height = min_height;
950  metrics.pixel_ratio = pixel_ratio;
951  metrics.display_id = display_id;
952  metrics.view_id = view_id;
953  metrics.has_constraints = true;
954  metrics.min_width_constraint = min_width;
955  metrics.min_height_constraint = min_height;
956  metrics.max_width_constraint = max_width;
957  metrics.max_height_constraint = max_height;
958  FlutterAddViewInfo info;
959  info.struct_size = sizeof(FlutterAddViewInfo);
960  info.view_id = view_id;
961  info.view_metrics = &metrics;
962  info.user_data = g_object_ref(task);
963  info.add_view_callback = view_added_cb;
964  FlutterEngineResult result = self->embedder_api.AddView(self->engine, &info);
965  if (result != kSuccess) {
966  g_task_return_new_error(task, fl_engine_error_quark(),
967  FL_ENGINE_ERROR_FAILED, "AddView returned %d",
968  result);
969  // This would have been done in the callback, but that won't occur now.
970  g_object_unref(task);
971  }
972 
973  return view_id;
974 }
975 
976 gboolean fl_engine_add_view_finish(FlEngine* self,
977  GAsyncResult* result,
978  GError** error) {
979  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
980  return g_task_propagate_boolean(G_TASK(result), error);
981 }
982 
983 FlRenderable* fl_engine_get_renderable(FlEngine* self, FlutterViewId view_id) {
984  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
985 
986  return get_renderable(self, view_id);
987 }
988 
989 void fl_engine_remove_view(FlEngine* self,
990  FlutterViewId view_id,
991  GCancellable* cancellable,
992  GAsyncReadyCallback callback,
993  gpointer user_data) {
994  g_return_if_fail(FL_IS_ENGINE(self));
995 
996  remove_renderable(self, view_id);
997 
998  g_autoptr(GTask) task = g_task_new(self, cancellable, callback, user_data);
999 
1000  FlutterRemoveViewInfo info;
1001  info.struct_size = sizeof(FlutterRemoveViewInfo);
1002  info.view_id = view_id;
1003  info.user_data = g_object_ref(task);
1004  info.remove_view_callback = view_removed_cb;
1005  FlutterEngineResult result =
1006  self->embedder_api.RemoveView(self->engine, &info);
1007  if (result != kSuccess) {
1008  g_task_return_new_error(task, fl_engine_error_quark(),
1009  FL_ENGINE_ERROR_FAILED, "RemoveView returned %d",
1010  result);
1011  // This would have been done in the callback, but that won't occur now.
1012  g_object_unref(task);
1013  }
1014 }
1015 
1016 gboolean fl_engine_remove_view_finish(FlEngine* self,
1017  GAsyncResult* result,
1018  GError** error) {
1019  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1020  return g_task_propagate_boolean(G_TASK(result), error);
1021 }
1022 
1024  FlEngine* self,
1026  gpointer user_data,
1027  GDestroyNotify destroy_notify) {
1028  g_return_if_fail(FL_IS_ENGINE(self));
1029  g_return_if_fail(handler != nullptr);
1030 
1031  if (self->platform_message_handler_destroy_notify) {
1032  self->platform_message_handler_destroy_notify(
1033  self->platform_message_handler_data);
1034  }
1035 
1036  self->platform_message_handler = handler;
1037  self->platform_message_handler_data = user_data;
1038  self->platform_message_handler_destroy_notify = destroy_notify;
1039 }
1040 
1041 // Note: This function can be called from any thread.
1043  FlEngine* self,
1044  const FlutterPlatformMessageResponseHandle* handle,
1045  GBytes* response,
1046  GError** error) {
1047  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1048  g_return_val_if_fail(handle != nullptr, FALSE);
1049 
1050  if (self->engine == nullptr) {
1052  "No engine to send response to");
1053  return FALSE;
1054  }
1055 
1056  gsize data_length = 0;
1057  const uint8_t* data = nullptr;
1058  if (response != nullptr) {
1059  data =
1060  static_cast<const uint8_t*>(g_bytes_get_data(response, &data_length));
1061  }
1062  FlutterEngineResult result = self->embedder_api.SendPlatformMessageResponse(
1063  self->engine, handle, data, data_length);
1064 
1065  if (result != kSuccess) {
1067  "Failed to send platform message response");
1068  return FALSE;
1069  }
1070 
1071  return TRUE;
1072 }
1073 
1075  const gchar* channel,
1076  GBytes* message,
1077  GCancellable* cancellable,
1078  GAsyncReadyCallback callback,
1079  gpointer user_data) {
1080  g_return_if_fail(FL_IS_ENGINE(self));
1081 
1082  GTask* task = nullptr;
1083  FlutterPlatformMessageResponseHandle* response_handle = nullptr;
1084  if (callback != nullptr) {
1085  task = g_task_new(self, cancellable, callback, user_data);
1086 
1087  if (self->engine == nullptr) {
1088  g_task_return_new_error(task, fl_engine_error_quark(),
1089  FL_ENGINE_ERROR_FAILED, "No engine to send to");
1090  return;
1091  }
1092 
1093  FlutterEngineResult result =
1094  self->embedder_api.PlatformMessageCreateResponseHandle(
1095  self->engine, fl_engine_platform_message_response_cb, task,
1096  &response_handle);
1097  if (result != kSuccess) {
1098  g_task_return_new_error(task, fl_engine_error_quark(),
1100  "Failed to create response handle");
1101  g_object_unref(task);
1102  return;
1103  }
1104  } else if (self->engine == nullptr) {
1105  return;
1106  }
1107 
1108  FlutterPlatformMessage fl_message = {};
1109  fl_message.struct_size = sizeof(fl_message);
1110  fl_message.channel = channel;
1111  fl_message.message =
1112  message != nullptr
1113  ? static_cast<const uint8_t*>(g_bytes_get_data(message, nullptr))
1114  : nullptr;
1115  fl_message.message_size = message != nullptr ? g_bytes_get_size(message) : 0;
1116  fl_message.response_handle = response_handle;
1117  FlutterEngineResult result =
1118  self->embedder_api.SendPlatformMessage(self->engine, &fl_message);
1119 
1120  if (result != kSuccess && task != nullptr) {
1121  g_task_return_new_error(task, fl_engine_error_quark(),
1123  "Failed to send platform messages");
1124  g_object_unref(task);
1125  }
1126 
1127  if (response_handle != nullptr) {
1128  if (self->embedder_api.PlatformMessageReleaseResponseHandle(
1129  self->engine, response_handle) != kSuccess) {
1130  g_warning("Failed to release response handle");
1131  }
1132  }
1133 }
1134 
1136  GAsyncResult* result,
1137  GError** error) {
1138  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1139  g_return_val_if_fail(g_task_is_valid(result, self), FALSE);
1140 
1141  return static_cast<GBytes*>(g_task_propagate_pointer(G_TASK(result), error));
1142 }
1143 
1145  FlutterEngineDisplayId display_id,
1146  FlutterViewId view_id,
1147  size_t min_width,
1148  size_t min_height,
1149  size_t max_width,
1150  size_t max_height,
1151  double pixel_ratio) {
1152  g_return_if_fail(FL_IS_ENGINE(self));
1153 
1154  if (self->engine == nullptr) {
1155  return;
1156  }
1157 
1158  FlutterWindowMetricsEvent event = {};
1159  event.struct_size = sizeof(FlutterWindowMetricsEvent);
1160  event.width = min_width;
1161  event.height = min_height;
1162  event.pixel_ratio = pixel_ratio;
1163  event.display_id = display_id;
1164  event.view_id = view_id;
1165  event.has_constraints = true;
1166  event.min_width_constraint = min_width;
1167  event.min_height_constraint = min_height;
1168  event.max_width_constraint = max_width;
1169  event.max_height_constraint = max_height;
1170  if (self->embedder_api.SendWindowMetricsEvent(self->engine, &event) !=
1171  kSuccess) {
1172  g_warning("Failed to send window metrics");
1173  }
1174 }
1175 
1177  FlutterViewId view_id,
1178  FlutterPointerPhase phase,
1179  size_t timestamp,
1180  double x,
1181  double y,
1182  FlutterPointerDeviceKind device_kind,
1183  double scroll_delta_x,
1184  double scroll_delta_y,
1185  int64_t buttons) {
1186  g_return_if_fail(FL_IS_ENGINE(self));
1187 
1188  if (self->engine == nullptr) {
1189  return;
1190  }
1191 
1192  FlutterPointerEvent fl_event = {};
1193  fl_event.struct_size = sizeof(fl_event);
1194  fl_event.phase = phase;
1195  fl_event.timestamp = timestamp;
1196  fl_event.x = x;
1197  fl_event.y = y;
1198  if (scroll_delta_x != 0 || scroll_delta_y != 0) {
1199  fl_event.signal_kind = kFlutterPointerSignalKindScroll;
1200  }
1201  fl_event.scroll_delta_x = scroll_delta_x;
1202  fl_event.scroll_delta_y = scroll_delta_y;
1203  fl_event.device_kind = device_kind;
1204  fl_event.buttons = buttons;
1205  fl_event.device = kMousePointerDeviceId;
1206  fl_event.view_id = view_id;
1207  if (self->embedder_api.SendPointerEvent(self->engine, &fl_event, 1) !=
1208  kSuccess) {
1209  g_warning("Failed to send pointer event");
1210  }
1211 }
1212 
1213 void fl_engine_send_touch_up_event(FlEngine* self,
1214  FlutterViewId view_id,
1215  size_t timestamp,
1216  double x,
1217  double y,
1218  int32_t device) {
1219  g_return_if_fail(FL_IS_ENGINE(self));
1220 
1221  if (self->engine == nullptr) {
1222  return;
1223  }
1224 
1225  FlutterPointerEvent event;
1226  event.timestamp = timestamp;
1227  event.x = x;
1228  event.y = y;
1229  event.device_kind = kFlutterPointerDeviceKindTouch;
1230  event.device = device;
1231  event.buttons = 0;
1232  event.view_id = view_id;
1233  event.phase = FlutterPointerPhase::kUp;
1234  event.struct_size = sizeof(event);
1235 
1236  if (self->embedder_api.SendPointerEvent(self->engine, &event, 1) !=
1237  kSuccess) {
1238  g_warning("Failed to send pointer event");
1239  }
1240 }
1241 
1243  FlutterViewId view_id,
1244  size_t timestamp,
1245  double x,
1246  double y,
1247  int32_t device) {
1248  g_return_if_fail(FL_IS_ENGINE(self));
1249 
1250  if (self->engine == nullptr) {
1251  return;
1252  }
1253 
1254  FlutterPointerEvent event;
1255  event.timestamp = timestamp;
1256  event.x = x;
1257  event.y = y;
1258  event.device_kind = kFlutterPointerDeviceKindTouch;
1259  event.device = device;
1260  event.buttons = FlutterPointerMouseButtons::kFlutterPointerButtonMousePrimary;
1261  event.view_id = view_id;
1262  event.phase = FlutterPointerPhase::kDown;
1263  event.struct_size = sizeof(event);
1264 
1265  if (self->embedder_api.SendPointerEvent(self->engine, &event, 1) !=
1266  kSuccess) {
1267  g_warning("Failed to send pointer event");
1268  }
1269 }
1270 
1272  FlutterViewId view_id,
1273  size_t timestamp,
1274  double x,
1275  double y,
1276  int32_t device) {
1277  g_return_if_fail(FL_IS_ENGINE(self));
1278 
1279  if (self->engine == nullptr) {
1280  return;
1281  }
1282 
1283  FlutterPointerEvent event;
1284  event.timestamp = timestamp;
1285  event.x = x;
1286  event.y = y;
1287  event.device_kind = kFlutterPointerDeviceKindTouch;
1288  event.device = device;
1289  event.buttons = FlutterPointerMouseButtons::kFlutterPointerButtonMousePrimary;
1290  event.view_id = view_id;
1291  event.phase = FlutterPointerPhase::kMove;
1292  event.struct_size = sizeof(event);
1293 
1294  if (self->embedder_api.SendPointerEvent(self->engine, &event, 1) !=
1295  kSuccess) {
1296  g_warning("Failed to send pointer event");
1297  }
1298 }
1299 
1301  FlutterViewId view_id,
1302  size_t timestamp,
1303  double x,
1304  double y,
1305  int32_t device) {
1306  g_return_if_fail(FL_IS_ENGINE(self));
1307 
1308  if (self->engine == nullptr) {
1309  return;
1310  }
1311 
1312  FlutterPointerEvent event;
1313  event.timestamp = timestamp;
1314  event.x = x;
1315  event.y = y;
1316  event.device_kind = kFlutterPointerDeviceKindTouch;
1317  event.device = device;
1318  event.buttons = 0;
1319  event.view_id = view_id;
1320  event.phase = FlutterPointerPhase::kAdd;
1321  event.struct_size = sizeof(event);
1322 
1323  if (self->embedder_api.SendPointerEvent(self->engine, &event, 1) !=
1324  kSuccess) {
1325  g_warning("Failed to send pointer event");
1326  }
1327 }
1328 
1330  FlutterViewId view_id,
1331  size_t timestamp,
1332  double x,
1333  double y,
1334  int32_t device) {
1335  g_return_if_fail(FL_IS_ENGINE(self));
1336 
1337  if (self->engine == nullptr) {
1338  return;
1339  }
1340 
1341  FlutterPointerEvent event;
1342  event.timestamp = timestamp;
1343  event.x = x;
1344  event.y = y;
1345  event.device_kind = kFlutterPointerDeviceKindTouch;
1346  event.device = device;
1347  event.buttons = 0;
1348  event.view_id = view_id;
1349  event.phase = FlutterPointerPhase::kRemove;
1350  event.struct_size = sizeof(event);
1351 
1352  if (self->embedder_api.SendPointerEvent(self->engine, &event, 1) !=
1353  kSuccess) {
1354  g_warning("Failed to send pointer event");
1355  }
1356 }
1357 
1359  FlutterViewId view_id,
1360  size_t timestamp,
1361  double x,
1362  double y,
1363  FlutterPointerPhase phase,
1364  double pan_x,
1365  double pan_y,
1366  double scale,
1367  double rotation) {
1368  g_return_if_fail(FL_IS_ENGINE(self));
1369 
1370  if (self->engine == nullptr) {
1371  return;
1372  }
1373 
1374  FlutterPointerEvent fl_event = {};
1375  fl_event.struct_size = sizeof(fl_event);
1376  fl_event.timestamp = timestamp;
1377  fl_event.x = x;
1378  fl_event.y = y;
1379  fl_event.phase = phase;
1380  fl_event.pan_x = pan_x;
1381  fl_event.pan_y = pan_y;
1382  fl_event.scale = scale;
1383  fl_event.rotation = rotation;
1384  fl_event.device = kPointerPanZoomDeviceId;
1385  fl_event.device_kind = kFlutterPointerDeviceKindTrackpad;
1386  fl_event.view_id = view_id;
1387  if (self->embedder_api.SendPointerEvent(self->engine, &fl_event, 1) !=
1388  kSuccess) {
1389  g_warning("Failed to send pointer event");
1390  }
1391 }
1392 
1393 static void send_key_event_cb(bool handled, void* user_data) {
1394  g_autoptr(GTask) task = G_TASK(user_data);
1395  gboolean* return_value = g_new0(gboolean, 1);
1396  *return_value = handled;
1397  g_task_return_pointer(task, return_value, g_free);
1398 }
1399 
1400 void fl_engine_send_key_event(FlEngine* self,
1401  const FlutterKeyEvent* event,
1402  GCancellable* cancellable,
1403  GAsyncReadyCallback callback,
1404  gpointer user_data) {
1405  g_return_if_fail(FL_IS_ENGINE(self));
1406 
1407  g_autoptr(GTask) task = g_task_new(self, cancellable, callback, user_data);
1408 
1409  if (self->engine == nullptr) {
1410  g_task_return_new_error(task, fl_engine_error_quark(),
1411  FL_ENGINE_ERROR_FAILED, "No engine");
1412  return;
1413  }
1414 
1415  if (self->embedder_api.SendKeyEvent(self->engine, event, send_key_event_cb,
1416  g_object_ref(task)) != kSuccess) {
1417  g_task_return_new_error(task, fl_engine_error_quark(),
1418  FL_ENGINE_ERROR_FAILED, "Failed to send key event");
1419  g_object_unref(task);
1420  }
1421 }
1422 
1423 gboolean fl_engine_send_key_event_finish(FlEngine* self,
1424  GAsyncResult* result,
1425  gboolean* handled,
1426  GError** error) {
1427  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1428  g_return_val_if_fail(g_task_is_valid(result, self), FALSE);
1429 
1430  g_autofree gboolean* return_value =
1431  static_cast<gboolean*>(g_task_propagate_pointer(G_TASK(result), error));
1432  if (return_value == nullptr) {
1433  return FALSE;
1434  }
1435 
1436  *handled = *return_value;
1437  return TRUE;
1438 }
1439 
1441  FlutterViewId view_id,
1442  uint64_t node_id,
1443  FlutterSemanticsAction action,
1444  GBytes* data) {
1445  g_return_if_fail(FL_IS_ENGINE(self));
1446 
1447  if (self->engine == nullptr) {
1448  return;
1449  }
1450 
1451  const uint8_t* action_data = nullptr;
1452  size_t action_data_length = 0;
1453  if (data != nullptr) {
1454  action_data = static_cast<const uint8_t*>(
1455  g_bytes_get_data(data, &action_data_length));
1456  }
1457 
1458  FlutterSendSemanticsActionInfo info;
1459  info.struct_size = sizeof(FlutterSendSemanticsActionInfo);
1460  info.view_id = view_id;
1461  info.node_id = node_id;
1462  info.action = action;
1463  info.data = action_data;
1464  info.data_length = action_data_length;
1465  if (self->embedder_api.SendSemanticsAction(self->engine, &info) != kSuccess) {
1466  g_warning("Failed to send semantics action");
1467  }
1468 }
1469 
1471  int64_t texture_id) {
1472  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1473  return self->embedder_api.MarkExternalTextureFrameAvailable(
1474  self->engine, texture_id) == kSuccess;
1475 }
1476 
1477 gboolean fl_engine_register_external_texture(FlEngine* self,
1478  int64_t texture_id) {
1479  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1480  return self->embedder_api.RegisterExternalTexture(self->engine, texture_id) ==
1481  kSuccess;
1482 }
1483 
1485  int64_t texture_id) {
1486  g_return_val_if_fail(FL_IS_ENGINE(self), FALSE);
1487  return self->embedder_api.UnregisterExternalTexture(self->engine,
1488  texture_id) == kSuccess;
1489 }
1490 
1491 G_MODULE_EXPORT FlBinaryMessenger* fl_engine_get_binary_messenger(
1492  FlEngine* self) {
1493  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
1494  return self->binary_messenger;
1495 }
1496 
1497 FlTaskRunner* fl_engine_get_task_runner(FlEngine* self) {
1498  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
1499  return self->task_runner;
1500 }
1501 
1502 void fl_engine_execute_task(FlEngine* self, FlutterTask* task) {
1503  g_return_if_fail(FL_IS_ENGINE(self));
1504  if (self->embedder_api.RunTask(self->engine, task) != kSuccess) {
1505  g_warning("Failed to run task");
1506  }
1507 }
1508 
1509 G_MODULE_EXPORT FlTextureRegistrar* fl_engine_get_texture_registrar(
1510  FlEngine* self) {
1511  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
1512  return self->texture_registrar;
1513 }
1514 
1515 void fl_engine_update_accessibility_features(FlEngine* self, int32_t flags) {
1516  g_return_if_fail(FL_IS_ENGINE(self));
1517 
1518  if (self->engine == nullptr) {
1519  return;
1520  }
1521 
1522  if (self->embedder_api.UpdateAccessibilityFeatures(
1523  self->engine, static_cast<FlutterAccessibilityFeature>(flags)) !=
1524  kSuccess) {
1525  g_warning("Failed to update accessibility features");
1526  }
1527 }
1528 
1529 void fl_engine_request_app_exit(FlEngine* self) {
1530  g_return_if_fail(FL_IS_ENGINE(self));
1531  fl_platform_handler_request_app_exit(self->platform_handler);
1532 }
1533 
1534 FlKeyboardManager* fl_engine_get_keyboard_manager(FlEngine* self) {
1535  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
1536  return self->keyboard_manager;
1537 }
1538 
1539 FlTextInputHandler* fl_engine_get_text_input_handler(FlEngine* self) {
1540  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
1541  return self->text_input_handler;
1542 }
1543 
1544 FlMouseCursorHandler* fl_engine_get_mouse_cursor_handler(FlEngine* self) {
1545  g_return_val_if_fail(FL_IS_ENGINE(self), nullptr);
1546  return self->mouse_cursor_handler;
1547 }
FlAccessibilityHandler * fl_accessibility_handler_new(FlEngine *engine)
const char * message
g_autoptr(FlEngine) engine
FlRenderable * renderable
const char FlTextDirection FlAssertiveness gpointer user_data
@ PROP_LAST
@ PROP_0
G_DEFINE_QUARK(fl_binary_messenger_codec_error_quark, fl_binary_messenger_codec_error) G_DECLARE_FINAL_TYPE(FlBinaryMessengerImpl
FlBinaryMessenger * fl_binary_messenger_new(FlEngine *engine)
void fl_binary_messenger_shutdown(FlBinaryMessenger *self)
self height
self width
return TRUE
G_MODULE_EXPORT FlDartProject * fl_dart_project_new()
G_MODULE_EXPORT const gchar * fl_dart_project_get_aot_library_path(FlDartProject *self)
G_MODULE_EXPORT gchar ** fl_dart_project_get_dart_entrypoint_arguments(FlDartProject *self)
G_MODULE_EXPORT const gchar * fl_dart_project_get_icu_data_path(FlDartProject *self)
G_MODULE_EXPORT const gchar * fl_dart_project_get_assets_path(FlDartProject *self)
G_MODULE_EXPORT FlUIThreadPolicy fl_dart_project_get_ui_thread_policy(FlDartProject *project)
@ FL_UI_THREAD_POLICY_RUN_ON_PLATFORM_THREAD
@ FL_UI_THREAD_POLICY_RUN_ON_SEPARATE_THREAD
@ FL_UI_THREAD_POLICY_DEFAULT
void fl_display_monitor_start(FlDisplayMonitor *self)
FlDisplayMonitor * fl_display_monitor_new(FlEngine *engine, GdkDisplay *display)
static bool fl_engine_gl_external_texture_frame_callback(void *user_data, int64_t texture_id, size_t width, size_t height, FlutterOpenGLTexture *opengl_texture)
Definition: fl_engine.cc:435
G_MODULE_EXPORT FlTextureRegistrar * fl_engine_get_texture_registrar(FlEngine *self)
Definition: fl_engine.cc:1509
FlMouseCursorHandler * fl_engine_get_mouse_cursor_handler(FlEngine *self)
Definition: fl_engine.cc:1544
static bool fl_engine_runs_task_on_current_thread(void *user_data)
Definition: fl_engine.cc:476
static void fl_engine_on_pre_engine_restart_cb(void *user_data)
Definition: fl_engine.cc:542
void fl_engine_send_mouse_pointer_event(FlEngine *self, FlutterViewId view_id, FlutterPointerPhase phase, size_t timestamp, double x, double y, FlutterPointerDeviceKind device_kind, double scroll_delta_x, double scroll_delta_y, int64_t buttons)
Definition: fl_engine.cc:1176
void fl_engine_set_platform_message_handler(FlEngine *self, FlEnginePlatformMessageHandler handler, gpointer user_data, GDestroyNotify destroy_notify)
Definition: fl_engine.cc:1023
static FlPluginRegistrar * fl_engine_get_registrar_for_plugin(FlPluginRegistry *registry, const gchar *name)
Definition: fl_engine.cc:561
G_DEFINE_TYPE_WITH_CODE(FlEngine, fl_engine, G_TYPE_OBJECT, G_IMPLEMENT_INTERFACE(fl_plugin_registry_get_type(), fl_engine_plugin_registry_iface_init)) enum
Definition: fl_engine.cc:122
static void view_added_cb(const FlutterAddViewResult *result)
Definition: fl_engine.cc:204
static bool compositor_create_backing_store_callback(const FlutterBackingStoreConfig *config, FlutterBackingStore *backing_store_out, void *user_data)
Definition: fl_engine.cc:364
static bool fl_engine_gl_make_current(void *user_data)
Definition: fl_engine.cc:414
gboolean fl_engine_send_key_event_finish(FlEngine *self, GAsyncResult *result, gboolean *handled, GError **error)
Definition: fl_engine.cc:1423
FlutterEngineProcTable * fl_engine_get_embedder_api(FlEngine *self)
Definition: fl_engine.cc:902
void fl_engine_dispatch_semantics_action(FlEngine *self, FlutterViewId view_id, uint64_t node_id, FlutterSemanticsAction action, GBytes *data)
Definition: fl_engine.cc:1440
static FlEngine * fl_engine_new_full(FlDartProject *project, FlBinaryMessenger *binary_messenger)
Definition: fl_engine.cc:688
static void parse_locale(const gchar *locale, gchar **language, gchar **territory, gchar **codeset, gchar **modifier)
Definition: fl_engine.cc:132
static void set_renderable(FlEngine *self, int64_t view_id, FlRenderable *renderable)
Stores a weak reference to the renderable with the given ID.
Definition: fl_engine.cc:176
void fl_engine_send_touch_up_event(FlEngine *self, FlutterViewId view_id, size_t timestamp, double x, double y, int32_t device)
Definition: fl_engine.cc:1213
void fl_engine_send_window_metrics_event(FlEngine *self, FlutterEngineDisplayId display_id, FlutterViewId view_id, size_t min_width, size_t min_height, size_t max_width, size_t max_height, double pixel_ratio)
Definition: fl_engine.cc:1144
void fl_engine_notify_display_update(FlEngine *self, const FlutterEngineDisplay *displays, size_t displays_length)
Definition: fl_engine.cc:906
static bool collect_opengl_backing_store(FlEngine *self, const FlutterBackingStore *backing_store)
Definition: fl_engine.cc:320
static bool compositor_present_view_callback(const FlutterPresentViewInfo *info)
Definition: fl_engine.cc:395
FlOpenGLManager * fl_engine_get_opengl_manager(FlEngine *self)
Definition: fl_engine.cc:750
static bool compositor_collect_backing_store_callback(const FlutterBackingStore *backing_store, void *user_data)
Definition: fl_engine.cc:380
FlutterRendererType fl_engine_get_renderer_type(FlEngine *self)
Definition: fl_engine.cc:745
static void remove_renderable(FlEngine *self, int64_t view_id)
Remove a renderable that no longer exists.
Definition: fl_engine.cc:199
static void fl_engine_dispose(GObject *object)
Definition: fl_engine.cc:591
gboolean fl_engine_mark_texture_frame_available(FlEngine *self, int64_t texture_id)
Definition: fl_engine.cc:1470
static bool create_opengl_backing_store(FlEngine *self, const FlutterBackingStoreConfig *config, FlutterBackingStore *backing_store_out)
Definition: fl_engine.cc:283
gboolean fl_engine_remove_view_finish(FlEngine *self, GAsyncResult *result, GError **error)
Definition: fl_engine.cc:1016
FlTaskRunner * fl_engine_get_task_runner(FlEngine *self)
Definition: fl_engine.cc:1497
FlKeyboardManager * fl_engine_get_keyboard_manager(FlEngine *self)
Definition: fl_engine.cc:1534
FlDisplayMonitor * fl_engine_get_display_monitor(FlEngine *self)
Definition: fl_engine.cc:755
FlRenderable * fl_engine_get_renderable(FlEngine *self, FlutterViewId view_id)
Definition: fl_engine.cc:983
static guint fl_engine_signals[LAST_SIGNAL]
Definition: fl_engine.cc:120
static void fl_engine_platform_message_response_cb(const uint8_t *data, size_t data_length, void *user_data)
Definition: fl_engine.cc:552
gboolean fl_engine_send_platform_message_response(FlEngine *self, const FlutterPlatformMessageResponseHandle *handle, GBytes *response, GError **error)
Definition: fl_engine.cc:1042
static void fl_engine_init(FlEngine *self)
Definition: fl_engine.cc:661
static void fl_engine_post_task(FlutterTask task, uint64_t target_time_nanos, void *user_data)
Definition: fl_engine.cc:482
FlTextInputHandler * fl_engine_get_text_input_handler(FlEngine *self)
Definition: fl_engine.cc:1539
void fl_engine_set_implicit_view(FlEngine *self, FlRenderable *renderable)
Definition: fl_engine.cc:919
static uint32_t fl_engine_gl_get_fbo(void *user_data)
Definition: fl_engine.cc:424
FlutterViewId fl_engine_add_view(FlEngine *self, FlRenderable *renderable, size_t min_width, size_t min_height, size_t max_width, size_t max_height, double pixel_ratio, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
Definition: fl_engine.cc:923
static void view_removed_cb(const FlutterRemoveViewResult *result)
Definition: fl_engine.cc:215
gboolean fl_engine_unregister_external_texture(FlEngine *self, int64_t texture_id)
Definition: fl_engine.cc:1484
void fl_engine_send_platform_message(FlEngine *self, const gchar *channel, GBytes *message, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
Definition: fl_engine.cc:1074
gboolean fl_engine_register_external_texture(FlEngine *self, int64_t texture_id)
Definition: fl_engine.cc:1477
FlEngine * fl_engine_for_id(int64_t id)
Definition: fl_engine.cc:725
static void * fl_engine_gl_proc_resolver(void *user_data, const char *name)
Definition: fl_engine.cc:410
void fl_engine_send_touch_move_event(FlEngine *self, FlutterViewId view_id, size_t timestamp, double x, double y, int32_t device)
Definition: fl_engine.cc:1271
static void fl_engine_set_property(GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec)
Definition: fl_engine.cc:575
void fl_engine_send_touch_add_event(FlEngine *self, FlutterViewId view_id, size_t timestamp, double x, double y, int32_t device)
Definition: fl_engine.cc:1300
@ LAST_SIGNAL
Definition: fl_engine.cc:118
@ SIGNAL_UPDATE_SEMANTICS
Definition: fl_engine.cc:118
@ SIGNAL_ON_PRE_ENGINE_RESTART
Definition: fl_engine.cc:118
void fl_engine_remove_view(FlEngine *self, FlutterViewId view_id, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
Definition: fl_engine.cc:989
G_MODULE_EXPORT FlBinaryMessenger * fl_engine_get_binary_messenger(FlEngine *self)
Definition: fl_engine.cc:1491
gboolean fl_engine_add_view_finish(FlEngine *self, GAsyncResult *result, GError **error)
Definition: fl_engine.cc:976
void fl_engine_send_key_event(FlEngine *self, const FlutterKeyEvent *event, GCancellable *cancellable, GAsyncReadyCallback callback, gpointer user_data)
Definition: fl_engine.cc:1400
static void setup_keyboard(FlEngine *self)
Definition: fl_engine.cc:518
static void free_locale(FlutterLocale *locale)
Definition: fl_engine.cc:226
void fl_engine_request_app_exit(FlEngine *self)
Definition: fl_engine.cc:1529
static void setup_locales(FlEngine *self)
Definition: fl_engine.cc:233
void fl_engine_send_pointer_pan_zoom_event(FlEngine *self, FlutterViewId view_id, size_t timestamp, double x, double y, FlutterPointerPhase phase, double pan_x, double pan_y, double scale, double rotation)
Definition: fl_engine.cc:1358
static void fl_engine_update_semantics_cb(const FlutterSemanticsUpdate2 *update, void *user_data)
Definition: fl_engine.cc:511
static bool fl_engine_gl_make_resource_current(void *user_data)
Definition: fl_engine.cc:429
G_MODULE_EXPORT FlEngine * fl_engine_new_headless(FlDartProject *project)
Definition: fl_engine.cc:741
static FlRenderable * get_renderable(FlEngine *self, int64_t view_id)
Definition: fl_engine.cc:188
void fl_engine_update_accessibility_features(FlEngine *self, int32_t flags)
Definition: fl_engine.cc:1515
static void fl_engine_platform_message_cb(const FlutterPlatformMessage *message, void *user_data)
Definition: fl_engine.cc:491
void fl_engine_send_touch_remove_event(FlEngine *self, FlutterViewId view_id, size_t timestamp, double x, double y, int32_t device)
Definition: fl_engine.cc:1329
void fl_engine_execute_task(FlEngine *self, FlutterTask *task)
Definition: fl_engine.cc:1502
void fl_engine_send_touch_down_event(FlEngine *self, FlutterViewId view_id, size_t timestamp, double x, double y, int32_t device)
Definition: fl_engine.cc:1242
gboolean fl_engine_start(FlEngine *self, GError **error)
Definition: fl_engine.cc:760
static void send_key_event_cb(bool handled, void *user_data)
Definition: fl_engine.cc:1393
static bool collect_software_backing_store(FlEngine *self, const FlutterBackingStore *backing_store)
Definition: fl_engine.cc:356
static constexpr size_t kPlatformTaskRunnerIdentifier
Definition: fl_engine.cc:32
FlEngine * fl_engine_new_with_binary_messenger(FlBinaryMessenger *binary_messenger)
Definition: fl_engine.cc:735
static constexpr int32_t kPointerPanZoomDeviceId
Definition: fl_engine.cc:37
G_MODULE_EXPORT FlEngine * fl_engine_new(FlDartProject *project)
Definition: fl_engine.cc:731
static constexpr int32_t kMousePointerDeviceId
Definition: fl_engine.cc:36
static void fl_engine_class_init(FlEngineClass *klass)
Definition: fl_engine.cc:641
static bool fl_engine_gl_clear_current(void *user_data)
Definition: fl_engine.cc:419
GBytes * fl_engine_send_platform_message_finish(FlEngine *self, GAsyncResult *result, GError **error)
Definition: fl_engine.cc:1135
static void fl_engine_plugin_registry_iface_init(FlPluginRegistryInterface *iface)
Definition: fl_engine.cc:570
static bool create_software_backing_store(FlEngine *self, const FlutterBackingStoreConfig *config, FlutterBackingStore *backing_store_out)
Definition: fl_engine.cc:332
GQuark fl_engine_error_quark(void) G_GNUC_CONST
@ FL_ENGINE_ERROR_FAILED
gboolean(* FlEnginePlatformMessageHandler)(FlEngine *engine, const gchar *channel, GBytes *message, const FlutterPlatformMessageResponseHandle *response_handle, gpointer user_data)
G_BEGIN_DECLS G_MODULE_EXPORT FlValue * args
GLuint fl_framebuffer_get_id(FlFramebuffer *self)
FlFramebuffer * fl_framebuffer_new(GLint format, size_t width, size_t height, gboolean shareable)
FlKeyboardHandler * fl_keyboard_handler_new(FlBinaryMessenger *messenger, FlKeyboardManager *keyboard_manager)
FlKeyboardManager * fl_keyboard_manager_new(FlEngine *engine)
FlMouseCursorHandler * fl_mouse_cursor_handler_new(FlBinaryMessenger *messenger)
FlOpenGLManager * fl_opengl_manager_new()
gboolean fl_opengl_manager_make_resource_current(FlOpenGLManager *self)
gboolean fl_opengl_manager_clear_current(FlOpenGLManager *self)
gboolean fl_opengl_manager_make_current(FlOpenGLManager *self)
int64_t id
gboolean fl_pixel_buffer_texture_populate(FlPixelBufferTexture *texture, uint32_t width, uint32_t height, FlutterOpenGLTexture *opengl_texture, GError **error)
const uint8_t uint32_t uint32_t GError ** error
FlPlatformHandler * fl_platform_handler_new(FlBinaryMessenger *messenger)
void fl_platform_handler_request_app_exit(FlPlatformHandler *self)
FlPluginRegistrar * fl_plugin_registrar_new(FlView *view, FlBinaryMessenger *messenger, FlTextureRegistrar *texture_registrar)
void fl_renderable_present_layers(FlRenderable *self, const FlutterLayer **layers, size_t layers_count)
FlSettings * fl_settings_new()
Definition: fl_settings.cc:55
void fl_settings_handler_start(FlSettingsHandler *self, FlSettings *settings)
FlSettingsHandler * fl_settings_handler_new(FlEngine *engine)
uint8_t value
guint prop_id
guint const GValue GParamSpec * pspec
void fl_task_runner_post_flutter_task(FlTaskRunner *self, FlutterTask task, uint64_t target_time_nanos)
FlTaskRunner * fl_task_runner_new(FlEngine *engine)
FlTextInputHandler * fl_text_input_handler_new(FlBinaryMessenger *messenger)
void fl_text_input_handler_set_widget(FlTextInputHandler *self, GtkWidget *widget)
GtkWidget * fl_text_input_handler_get_widget(FlTextInputHandler *self)
gboolean fl_texture_gl_populate(FlTextureGL *self, uint32_t width, uint32_t height, FlutterOpenGLTexture *opengl_texture, GError **error)
void fl_texture_registrar_shutdown(FlTextureRegistrar *self)
FlTextureRegistrar * fl_texture_registrar_new(FlEngine *engine)
FlTexture * fl_texture_registrar_lookup_texture(FlTextureRegistrar *self, int64_t texture_id)
G_BEGIN_DECLS FlutterViewId view_id
std::vector< std::string > GetSwitchesFromEnvironment()
FlTaskRunner * task_runner
Definition: fl_engine.cc:85
FlEnginePlatformMessageHandler platform_message_handler
Definition: fl_engine.cc:108
FlMouseCursorHandler * mouse_cursor_handler
Definition: fl_engine.cc:79
GDestroyNotify platform_message_handler_destroy_notify
Definition: fl_engine.cc:110
FlutterViewId next_view_id
Definition: fl_engine.cc:98
FlAccessibilityHandler * accessibility_handler
Definition: fl_engine.cc:67
FlSettingsHandler * settings_handler
Definition: fl_engine.cc:61
gpointer platform_message_handler_data
Definition: fl_engine.cc:109
FlOpenGLManager * opengl_manager
Definition: fl_engine.cc:55
GThread * thread
Definition: fl_engine.cc:43
FlPlatformHandler * platform_handler
Definition: fl_engine.cc:64
FLUTTER_API_SYMBOL(FlutterEngine) engine
FlutterRendererType renderer_type
Definition: fl_engine.cc:52
FlBinaryMessenger * binary_messenger
Definition: fl_engine.cc:58
FlDisplayMonitor * display_monitor
Definition: fl_engine.cc:49
GHashTable * renderables_by_view_id
Definition: fl_engine.cc:101
FlDartProject * project
Definition: fl_engine.cc:46
FlKeyboardManager * keyboard_manager
Definition: fl_engine.cc:70
GMutex renderables_mutex
Definition: fl_engine.cc:105
GObject parent_instance
Definition: fl_engine.cc:40
FlTextureRegistrar * texture_registrar
Definition: fl_engine.cc:82
FlutterEngineAOTData aot_data
Definition: fl_engine.cc:88
FlutterEngineProcTable embedder_api
Definition: fl_engine.cc:95
FlTextInputHandler * text_input_handler
Definition: fl_engine.cc:73
FlKeyboardHandler * keyboard_handler
Definition: fl_engine.cc:76
int64_t texture_id