Flutter Windows Embedder
flutter_windows_view.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 <chrono>
8 
9 #include "flutter/common/constants.h"
10 #include "flutter/fml/make_copyable.h"
11 #include "flutter/fml/platform/win/wstring_conversion.h"
12 #include "flutter/fml/synchronization/waitable_event.h"
16 #include "flutter/third_party/accessibility/ax/platform/ax_platform_node_win.h"
17 
18 namespace flutter {
19 
20 namespace {
21 // The maximum duration to block the Windows event loop while waiting
22 // for a window resize operation to complete.
23 constexpr std::chrono::milliseconds kWindowResizeTimeout{100};
24 
25 /// Returns true if the surface will be updated as part of the resize process.
26 ///
27 /// This is called on window resize to determine if the platform thread needs
28 /// to be blocked until the frame with the right size has been rendered. It
29 /// should be kept in-sync with how the engine deals with a new surface request
30 /// as seen in `CreateOrUpdateSurface` in `GPUSurfaceGL`.
31 bool SurfaceWillUpdate(size_t cur_width,
32  size_t cur_height,
33  size_t target_width,
34  size_t target_height) {
35  // TODO (https://github.com/flutter/flutter/issues/65061) : Avoid special
36  // handling for zero dimensions.
37  bool non_zero_target_dims = target_height > 0 && target_width > 0;
38  bool not_same_size =
39  (cur_height != target_height) || (cur_width != target_width);
40  return non_zero_target_dims && not_same_size;
41 }
42 
43 /// Update the surface's swap interval to block until the v-blank iff
44 /// the system compositor is disabled.
45 void UpdateVsync(const FlutterWindowsEngine& engine,
46  egl::WindowSurface* surface,
47  bool needs_vsync) {
48  egl::Manager* egl_manager = engine.egl_manager();
49  if (!egl_manager) {
50  return;
51  }
52 
53  auto update_vsync = [egl_manager, surface, needs_vsync]() {
54  if (!surface || !surface->IsValid()) {
55  return;
56  }
57 
58  if (!surface->MakeCurrent()) {
59  FML_LOG(ERROR) << "Unable to make the render surface current to update "
60  "the swap interval";
61  return;
62  }
63 
64  if (!surface->SetVSyncEnabled(needs_vsync)) {
65  FML_LOG(ERROR) << "Unable to update the render surface's swap interval";
66  }
67 
68  if (!egl_manager->render_context()->ClearCurrent()) {
69  FML_LOG(ERROR) << "Unable to clear current surface after updating "
70  "the swap interval";
71  }
72  };
73 
74  // Updating the vsync makes the EGL context and render surface current.
75  // If the engine is running, the render surface should only be made current on
76  // the raster thread. If the engine is initializing, the raster thread doesn't
77  // exist yet and the render surface can be made current on the platform
78  // thread.
79  if (engine.running()) {
80  engine.PostRasterThreadTask(update_vsync);
81  } else {
82  update_vsync();
83  }
84 }
85 
86 /// Destroys a rendering surface that backs a Flutter view.
87 void DestroyWindowSurface(const FlutterWindowsEngine& engine,
88  std::unique_ptr<egl::WindowSurface> surface) {
89  // EGL surfaces are used on the raster thread if the engine is running.
90  // There may be pending raster tasks that use this surface. Destroy the
91  // surface on the raster thread to avoid concurrent uses.
92  if (engine.running()) {
93  engine.PostRasterThreadTask(fml::MakeCopyable(
94  [surface = std::move(surface)] { surface->Destroy(); }));
95  } else {
96  // There's no raster thread if engine isn't running. The surface can be
97  // destroyed on the platform thread.
98  surface->Destroy();
99  }
100 }
101 
102 } // namespace
103 
105  FlutterViewId view_id,
106  FlutterWindowsEngine* engine,
107  std::unique_ptr<WindowBindingHandler> window_binding,
108  std::shared_ptr<WindowsProcTable> windows_proc_table)
109  : view_id_(view_id),
110  engine_(engine),
111  windows_proc_table_(std::move(windows_proc_table)) {
112  if (windows_proc_table_ == nullptr) {
113  windows_proc_table_ = std::make_shared<WindowsProcTable>();
114  }
115 
116  // Take the binding handler, and give it a pointer back to self.
117  binding_handler_ = std::move(window_binding);
118  binding_handler_->SetView(this);
119 }
120 
122  // The view owns the child window.
123  // Notify the engine the view's child window will no longer be visible.
125 
126  if (surface_) {
127  DestroyWindowSurface(*engine_, std::move(surface_));
128  }
129 }
130 
132  // Called on the raster thread.
133  std::unique_lock<std::mutex> lock(resize_mutex_);
134 
135  if (surface_ == nullptr || !surface_->IsValid()) {
136  return false;
137  }
138 
139  if (resize_status_ != ResizeState::kResizeStarted) {
140  return true;
141  }
142 
143  if (!ResizeRenderSurface(resize_target_height_, resize_target_width_)) {
144  return false;
145  }
146 
147  // Platform thread is blocked for the entire duration until the
148  // resize_status_ is set to kDone by |OnFramePresented|.
149  resize_status_ = ResizeState::kFrameGenerated;
150  return true;
151 }
152 
153 bool FlutterWindowsView::OnFrameGenerated(size_t width, size_t height) {
154  // Called on the raster thread.
155  std::unique_lock<std::mutex> lock(resize_mutex_);
156 
157  if (surface_ == nullptr || !surface_->IsValid()) {
158  return false;
159  }
160 
161  if (resize_status_ != ResizeState::kResizeStarted) {
162  return true;
163  }
164 
165  if (resize_target_width_ != width || resize_target_height_ != height) {
166  return false;
167  }
168 
169  if (!ResizeRenderSurface(resize_target_width_, resize_target_height_)) {
170  return false;
171  }
172 
173  // Platform thread is blocked for the entire duration until the
174  // resize_status_ is set to kDone by |OnFramePresented|.
175  resize_status_ = ResizeState::kFrameGenerated;
176  return true;
177 }
178 
179 void FlutterWindowsView::UpdateFlutterCursor(const std::string& cursor_name) {
180  binding_handler_->UpdateFlutterCursor(cursor_name);
181 }
182 
184  binding_handler_->SetFlutterCursor(cursor);
185 }
186 
188  if (resize_status_ == ResizeState::kDone) {
189  // Request new frame.
190  engine_->ScheduleFrame();
191  }
192 }
193 
194 // Called on the platform thread.
195 bool FlutterWindowsView::OnWindowSizeChanged(size_t width, size_t height) {
196  if (!engine_->egl_manager()) {
197  SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
198  return true;
199  }
200 
201  if (!surface_ || !surface_->IsValid()) {
202  SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
203  return true;
204  }
205 
206  // We're using OpenGL rendering. Resizing the surface must happen on the
207  // raster thread.
208  bool surface_will_update =
209  SurfaceWillUpdate(surface_->width(), surface_->height(), width, height);
210  if (!surface_will_update) {
211  SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
212  return true;
213  }
214 
215  {
216  std::unique_lock<std::mutex> lock(resize_mutex_);
217  resize_status_ = ResizeState::kResizeStarted;
218  resize_target_width_ = width;
219  resize_target_height_ = height;
220  }
221 
222  SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
223 
224  std::chrono::time_point<std::chrono::steady_clock> start_time =
225  std::chrono::steady_clock::now();
226 
227  while (true) {
228  if (std::chrono::steady_clock::now() > start_time + kWindowResizeTimeout) {
229  return false;
230  }
231  std::unique_lock<std::mutex> lock(resize_mutex_);
232  if (resize_status_ == ResizeState::kDone) {
233  break;
234  }
235  lock.unlock();
236  engine_->task_runner()->PollOnce(kWindowResizeTimeout);
237  }
238  return true;
239 }
240 
242  ForceRedraw();
243 }
244 
246  double y,
247  FlutterPointerDeviceKind device_kind,
248  int32_t device_id,
249  int modifiers_state) {
250  engine_->keyboard_key_handler()->SyncModifiersIfNeeded(modifiers_state);
251  SendPointerMove(x, y, GetOrCreatePointerState(device_kind, device_id));
252 }
253 
255  double x,
256  double y,
257  FlutterPointerDeviceKind device_kind,
258  int32_t device_id,
259  FlutterPointerMouseButtons flutter_button) {
260  if (flutter_button != 0) {
261  auto state = GetOrCreatePointerState(device_kind, device_id);
262  state->buttons |= flutter_button;
263  SendPointerDown(x, y, state);
264  }
265 }
266 
268  double x,
269  double y,
270  FlutterPointerDeviceKind device_kind,
271  int32_t device_id,
272  FlutterPointerMouseButtons flutter_button) {
273  if (flutter_button != 0) {
274  auto state = GetOrCreatePointerState(device_kind, device_id);
275  state->buttons &= ~flutter_button;
276  SendPointerUp(x, y, state);
277  }
278 }
279 
281  double y,
282  FlutterPointerDeviceKind device_kind,
283  int32_t device_id) {
284  SendPointerLeave(x, y, GetOrCreatePointerState(device_kind, device_id));
285 }
286 
288  PointerLocation point = binding_handler_->GetPrimaryPointerLocation();
289  SendPointerPanZoomStart(device_id, point.x, point.y);
290 }
291 
293  double pan_x,
294  double pan_y,
295  double scale,
296  double rotation) {
297  SendPointerPanZoomUpdate(device_id, pan_x, pan_y, scale, rotation);
298 }
299 
301  SendPointerPanZoomEnd(device_id);
302 }
303 
304 void FlutterWindowsView::OnText(const std::u16string& text) {
305  SendText(text);
306 }
307 
309  int scancode,
310  int action,
311  char32_t character,
312  bool extended,
313  bool was_down,
316 }
317 
318 void FlutterWindowsView::OnFocus(FlutterViewFocusState focus_state,
319  FlutterViewFocusDirection direction) {
320  SendFocus(focus_state, direction);
321 }
322 
324  SendComposeBegin();
325 }
326 
328  SendComposeCommit();
329 }
330 
332  SendComposeEnd();
333 }
334 
335 void FlutterWindowsView::OnComposeChange(const std::u16string& text,
336  int cursor_pos) {
337  SendComposeChange(text, cursor_pos);
338 }
339 
341  double y,
342  double delta_x,
343  double delta_y,
344  int scroll_offset_multiplier,
345  FlutterPointerDeviceKind device_kind,
346  int32_t device_id) {
347  SendScroll(x, y, delta_x, delta_y, scroll_offset_multiplier, device_kind,
348  device_id);
349 }
350 
352  PointerLocation point = binding_handler_->GetPrimaryPointerLocation();
353  SendScrollInertiaCancel(device_id, point.x, point.y);
354 }
355 
357  engine_->UpdateSemanticsEnabled(enabled);
358 }
359 
360 gfx::NativeViewAccessible FlutterWindowsView::GetNativeViewAccessible() {
361  if (!accessibility_bridge_) {
362  return nullptr;
363  }
364 
365  return accessibility_bridge_->GetChildOfAXFragmentRoot();
366 }
367 
369  binding_handler_->OnCursorRectUpdated(rect);
370 }
371 
373  binding_handler_->OnResetImeComposing();
374 }
375 
376 // Sends new size information to FlutterEngine.
377 void FlutterWindowsView::SendWindowMetrics(size_t width,
378  size_t height,
379  double pixel_ratio) const {
380  FlutterWindowMetricsEvent event = {};
381  event.struct_size = sizeof(event);
382  event.width = width;
383  event.height = height;
384  event.pixel_ratio = pixel_ratio;
385  event.view_id = view_id_;
386  engine_->SendWindowMetricsEvent(event);
387 }
388 
389 FlutterWindowMetricsEvent FlutterWindowsView::CreateWindowMetricsEvent() const {
390  PhysicalWindowBounds bounds = binding_handler_->GetPhysicalWindowBounds();
391  double pixel_ratio = binding_handler_->GetDpiScale();
392 
393  FlutterWindowMetricsEvent event = {};
394  event.struct_size = sizeof(event);
395  event.width = bounds.width;
396  event.height = bounds.height;
397  event.pixel_ratio = pixel_ratio;
398  event.view_id = view_id_;
399 
400  return event;
401 }
402 
404  // Non-implicit views' initial window metrics are sent when the view is added
405  // to the engine.
406  if (!IsImplicitView()) {
407  return;
408  }
409 
411 }
412 
413 FlutterWindowsView::PointerState* FlutterWindowsView::GetOrCreatePointerState(
414  FlutterPointerDeviceKind device_kind,
415  int32_t device_id) {
416  // Create a virtual pointer ID that is unique across all device types
417  // to prevent pointers from clashing in the engine's converter
418  // (lib/ui/window/pointer_data_packet_converter.cc)
419  int32_t pointer_id = (static_cast<int32_t>(device_kind) << 28) | device_id;
420 
421  auto [it, added] = pointer_states_.try_emplace(pointer_id, nullptr);
422  if (added) {
423  auto state = std::make_unique<PointerState>();
424  state->device_kind = device_kind;
425  state->pointer_id = pointer_id;
426  it->second = std::move(state);
427  }
428 
429  return it->second.get();
430 }
431 
432 // Set's |event_data|'s phase to either kMove or kHover depending on the current
433 // primary mouse button state.
434 void FlutterWindowsView::SetEventPhaseFromCursorButtonState(
435  FlutterPointerEvent* event_data,
436  const PointerState* state) const {
437  // For details about this logic, see FlutterPointerPhase in the embedder.h
438  // file.
439  if (state->buttons == 0) {
440  event_data->phase = state->flutter_state_is_down
441  ? FlutterPointerPhase::kUp
442  : FlutterPointerPhase::kHover;
443  } else {
444  event_data->phase = state->flutter_state_is_down
445  ? FlutterPointerPhase::kMove
446  : FlutterPointerPhase::kDown;
447  }
448 }
449 
450 void FlutterWindowsView::SendPointerMove(double x,
451  double y,
452  PointerState* state) {
453  FlutterPointerEvent event = {};
454  event.x = x;
455  event.y = y;
456 
457  SetEventPhaseFromCursorButtonState(&event, state);
458  SendPointerEventWithData(event, state);
459 }
460 
461 void FlutterWindowsView::SendPointerDown(double x,
462  double y,
463  PointerState* state) {
464  FlutterPointerEvent event = {};
465  event.x = x;
466  event.y = y;
467 
468  SetEventPhaseFromCursorButtonState(&event, state);
469  SendPointerEventWithData(event, state);
470 
471  state->flutter_state_is_down = true;
472 }
473 
474 void FlutterWindowsView::SendPointerUp(double x,
475  double y,
476  PointerState* state) {
477  FlutterPointerEvent event = {};
478  event.x = x;
479  event.y = y;
480 
481  SetEventPhaseFromCursorButtonState(&event, state);
482  SendPointerEventWithData(event, state);
483  if (event.phase == FlutterPointerPhase::kUp) {
484  state->flutter_state_is_down = false;
485  }
486 }
487 
488 void FlutterWindowsView::SendPointerLeave(double x,
489  double y,
490  PointerState* state) {
491  FlutterPointerEvent event = {};
492  event.x = x;
493  event.y = y;
494  event.phase = FlutterPointerPhase::kRemove;
495  SendPointerEventWithData(event, state);
496 }
497 
498 void FlutterWindowsView::SendPointerPanZoomStart(int32_t device_id,
499  double x,
500  double y) {
501  auto state =
502  GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
503  state->pan_zoom_start_x = x;
504  state->pan_zoom_start_y = y;
505  FlutterPointerEvent event = {};
506  event.x = x;
507  event.y = y;
508  event.phase = FlutterPointerPhase::kPanZoomStart;
509  SendPointerEventWithData(event, state);
510 }
511 
512 void FlutterWindowsView::SendPointerPanZoomUpdate(int32_t device_id,
513  double pan_x,
514  double pan_y,
515  double scale,
516  double rotation) {
517  auto state =
518  GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
519  FlutterPointerEvent event = {};
520  event.x = state->pan_zoom_start_x;
521  event.y = state->pan_zoom_start_y;
522  event.pan_x = pan_x;
523  event.pan_y = pan_y;
524  event.scale = scale;
525  event.rotation = rotation;
526  event.phase = FlutterPointerPhase::kPanZoomUpdate;
527  SendPointerEventWithData(event, state);
528 }
529 
530 void FlutterWindowsView::SendPointerPanZoomEnd(int32_t device_id) {
531  auto state =
532  GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
533  FlutterPointerEvent event = {};
534  event.x = state->pan_zoom_start_x;
535  event.y = state->pan_zoom_start_y;
536  event.phase = FlutterPointerPhase::kPanZoomEnd;
537  SendPointerEventWithData(event, state);
538 }
539 
540 void FlutterWindowsView::SendText(const std::u16string& text) {
541  engine_->text_input_plugin()->TextHook(text);
542 }
543 
544 void FlutterWindowsView::SendKey(int key,
545  int scancode,
546  int action,
547  char32_t character,
548  bool extended,
549  bool was_down,
550  KeyEventCallback callback) {
553  [engine = engine_, view_id = view_id_, key, scancode, action, character,
554  extended, was_down, callback = std::move(callback)](bool handled) {
555  if (!handled) {
556  engine->text_input_plugin()->KeyboardHook(
558  }
559  if (engine->view(view_id)) {
560  callback(handled);
561  }
562  });
563 }
564 
565 void FlutterWindowsView::SendFocus(FlutterViewFocusState focus_state,
566  FlutterViewFocusDirection direction) {
567  FlutterViewFocusEvent event = {};
568  event.struct_size = sizeof(event);
569  event.view_id = view_id_;
570  event.state = focus_state;
571  event.direction = direction;
572  engine_->SendViewFocusEvent(event);
573 }
574 
575 void FlutterWindowsView::SendComposeBegin() {
576  engine_->text_input_plugin()->ComposeBeginHook();
577 }
578 
579 void FlutterWindowsView::SendComposeCommit() {
580  engine_->text_input_plugin()->ComposeCommitHook();
581 }
582 
583 void FlutterWindowsView::SendComposeEnd() {
584  engine_->text_input_plugin()->ComposeEndHook();
585 }
586 
587 void FlutterWindowsView::SendComposeChange(const std::u16string& text,
588  int cursor_pos) {
589  engine_->text_input_plugin()->ComposeChangeHook(text, cursor_pos);
590 }
591 
592 void FlutterWindowsView::SendScroll(double x,
593  double y,
594  double delta_x,
595  double delta_y,
596  int scroll_offset_multiplier,
597  FlutterPointerDeviceKind device_kind,
598  int32_t device_id) {
599  auto state = GetOrCreatePointerState(device_kind, device_id);
600 
601  FlutterPointerEvent event = {};
602  event.x = x;
603  event.y = y;
604  event.signal_kind = FlutterPointerSignalKind::kFlutterPointerSignalKindScroll;
605  event.scroll_delta_x = delta_x * scroll_offset_multiplier;
606  event.scroll_delta_y = delta_y * scroll_offset_multiplier;
607  SetEventPhaseFromCursorButtonState(&event, state);
608  SendPointerEventWithData(event, state);
609 }
610 
611 void FlutterWindowsView::SendScrollInertiaCancel(int32_t device_id,
612  double x,
613  double y) {
614  auto state =
615  GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
616 
617  FlutterPointerEvent event = {};
618  event.x = x;
619  event.y = y;
620  event.signal_kind =
621  FlutterPointerSignalKind::kFlutterPointerSignalKindScrollInertiaCancel;
622  SetEventPhaseFromCursorButtonState(&event, state);
623  SendPointerEventWithData(event, state);
624 }
625 
626 void FlutterWindowsView::SendPointerEventWithData(
627  const FlutterPointerEvent& event_data,
628  PointerState* state) {
629  // If sending anything other than an add, and the pointer isn't already added,
630  // synthesize an add to satisfy Flutter's expectations about events.
631  if (!state->flutter_state_is_added &&
632  event_data.phase != FlutterPointerPhase::kAdd) {
633  FlutterPointerEvent event = {};
634  event.phase = FlutterPointerPhase::kAdd;
635  event.x = event_data.x;
636  event.y = event_data.y;
637  event.buttons = 0;
638  SendPointerEventWithData(event, state);
639  }
640 
641  // Don't double-add (e.g., if events are delivered out of order, so an add has
642  // already been synthesized).
643  if (state->flutter_state_is_added &&
644  event_data.phase == FlutterPointerPhase::kAdd) {
645  return;
646  }
647 
648  FlutterPointerEvent event = event_data;
649  event.device_kind = state->device_kind;
650  event.device = state->pointer_id;
651  event.buttons = state->buttons;
652  event.view_id = view_id_;
653 
654  // Set metadata that's always the same regardless of the event.
655  event.struct_size = sizeof(event);
656  event.timestamp =
657  std::chrono::duration_cast<std::chrono::microseconds>(
658  std::chrono::high_resolution_clock::now().time_since_epoch())
659  .count();
660 
661  engine_->SendPointerEvent(event);
662 
663  if (event_data.phase == FlutterPointerPhase::kAdd) {
664  state->flutter_state_is_added = true;
665  } else if (event_data.phase == FlutterPointerPhase::kRemove) {
666  auto it = pointer_states_.find(state->pointer_id);
667  if (it != pointer_states_.end()) {
668  pointer_states_.erase(it);
669  }
670  }
671 }
672 
674  // Called on the engine's raster thread.
675  std::unique_lock<std::mutex> lock(resize_mutex_);
676 
677  switch (resize_status_) {
678  case ResizeState::kResizeStarted:
679  // The caller must first call |OnFrameGenerated| or
680  // |OnEmptyFrameGenerated| before calling this method. This
681  // indicates one of the following:
682  //
683  // 1. The caller did not call these methods.
684  // 2. The caller ignored these methods' result.
685  // 3. The platform thread started a resize after the caller called these
686  // methods. We might have presented a frame of the wrong size to the
687  // view.
688  return;
689  case ResizeState::kFrameGenerated: {
690  // A frame was generated for a pending resize.
691  resize_status_ = ResizeState::kDone;
692  // Unblock the platform thread.
693  engine_->task_runner()->PostTask([this] {});
694 
695  lock.unlock();
696 
697  // Blocking the raster thread until DWM flushes alleviates glitches where
698  // previous size surface is stretched over current size view.
699  windows_proc_table_->DwmFlush();
700  }
701  case ResizeState::kDone:
702  return;
703  }
704 }
705 
707  return binding_handler_->OnBitmapSurfaceCleared();
708 }
709 
710 bool FlutterWindowsView::PresentSoftwareBitmap(const void* allocation,
711  size_t row_bytes,
712  size_t height) {
713  return binding_handler_->OnBitmapSurfaceUpdated(allocation, row_bytes,
714  height);
715 }
716 
718  return view_id_;
719 }
720 
722  return view_id_ == kImplicitViewId;
723 }
724 
726  FML_DCHECK(surface_ == nullptr);
727 
728  if (engine_->egl_manager()) {
729  PhysicalWindowBounds bounds = binding_handler_->GetPhysicalWindowBounds();
730  surface_ = engine_->egl_manager()->CreateWindowSurface(
731  GetWindowHandle(), bounds.width, bounds.height);
732 
733  UpdateVsync(*engine_, surface_.get(), NeedsVsync());
734 
735  resize_target_width_ = bounds.width;
736  resize_target_height_ = bounds.height;
737  }
738 }
739 
740 bool FlutterWindowsView::ResizeRenderSurface(size_t width, size_t height) {
741  FML_DCHECK(surface_ != nullptr);
742 
743  // No-op if the surface is already the desired size.
744  if (width == surface_->width() && height == surface_->height()) {
745  return true;
746  }
747 
748  auto const existing_vsync = surface_->vsync_enabled();
749 
750  // TODO: Destroying the surface and re-creating it is expensive.
751  // Ideally this would use ANGLE's automatic surface sizing instead.
752  // See: https://github.com/flutter/flutter/issues/79427
753  if (!surface_->Destroy()) {
754  FML_LOG(ERROR) << "View resize failed to destroy surface";
755  return false;
756  }
757 
758  std::unique_ptr<egl::WindowSurface> resized_surface =
759  engine_->egl_manager()->CreateWindowSurface(GetWindowHandle(), width,
760  height);
761  if (!resized_surface) {
762  FML_LOG(ERROR) << "View resize failed to create surface";
763  return false;
764  }
765 
766  if (!resized_surface->MakeCurrent() ||
767  !resized_surface->SetVSyncEnabled(existing_vsync)) {
768  // Surfaces block until the v-blank by default.
769  // Failing to update the vsync might result in unnecessary blocking.
770  // This regresses performance but not correctness.
771  FML_LOG(ERROR) << "View resize failed to set vsync";
772  }
773 
774  surface_ = std::move(resized_surface);
775  return true;
776 }
777 
779  return surface_.get();
780 }
781 
783  engine_->UpdateHighContrastMode();
784 }
785 
787  return binding_handler_->GetWindowHandle();
788 }
789 
791  return engine_;
792 }
793 
794 void FlutterWindowsView::AnnounceAlert(const std::wstring& text) {
795  auto alert_delegate = binding_handler_->GetAlertDelegate();
796  if (!alert_delegate) {
797  return;
798  }
799  alert_delegate->SetText(fml::WideStringToUtf16(text));
800  ui::AXPlatformNodeWin* alert_node = binding_handler_->GetAlert();
801  NotifyWinEventWrapper(alert_node, ax::mojom::Event::kAlert);
802 }
803 
804 void FlutterWindowsView::NotifyWinEventWrapper(ui::AXPlatformNodeWin* node,
805  ax::mojom::Event event) {
806  if (node) {
807  node->NotifyAccessibilityEvent(event);
808  }
809 }
810 
811 ui::AXFragmentRootDelegateWin* FlutterWindowsView::GetAxFragmentRootDelegate() {
812  return accessibility_bridge_.get();
813 }
814 
815 ui::AXPlatformNodeWin* FlutterWindowsView::AlertNode() const {
816  return binding_handler_->GetAlert();
817 }
818 
819 std::shared_ptr<AccessibilityBridgeWindows>
821  return std::make_shared<AccessibilityBridgeWindows>(this);
822 }
823 
825  if (semantics_enabled_ != enabled) {
826  semantics_enabled_ = enabled;
827 
828  if (!semantics_enabled_ && accessibility_bridge_) {
829  accessibility_bridge_.reset();
830  } else if (semantics_enabled_ && !accessibility_bridge_) {
831  accessibility_bridge_ = CreateAccessibilityBridge();
832  }
833  }
834 }
835 
837  UpdateVsync(*engine_, surface_.get(), NeedsVsync());
838 }
839 
841  engine_->OnWindowStateEvent(hwnd, event);
842 }
843 
845  return binding_handler_->Focus();
846 }
847 
848 bool FlutterWindowsView::NeedsVsync() const {
849  // If the Desktop Window Manager composition is enabled,
850  // the system itself synchronizes with vsync.
851  // See: https://learn.microsoft.com/windows/win32/dwm/composition-ovw
852  return !windows_proc_table_->DwmIsCompositionEnabled();
853 }
854 
855 } // namespace flutter
flutter::FlutterWindowsView::OnPointerMove
void OnPointerMove(double x, double y, FlutterPointerDeviceKind device_kind, int32_t device_id, int modifiers_state) override
Definition: flutter_windows_view.cc:245
flutter::TextInputPlugin::ComposeBeginHook
virtual void ComposeBeginHook()
Definition: text_input_plugin.cc:126
flutter::TextInputPlugin::ComposeChangeHook
virtual void ComposeChangeHook(const std::u16string &text, int cursor_pos)
Definition: text_input_plugin.cc:193
flutter::FlutterWindowsView::OnPointerUp
void OnPointerUp(double x, double y, FlutterPointerDeviceKind device_kind, int32_t device_id, FlutterPointerMouseButtons button) override
Definition: flutter_windows_view.cc:267
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::FlutterWindowsView::OnWindowStateEvent
void OnWindowStateEvent(HWND hwnd, WindowStateEvent event) override
Definition: flutter_windows_view.cc:840
flutter::FlutterWindowsView::CreateRenderSurface
void CreateRenderSurface()
Definition: flutter_windows_view.cc:725
flutter::FlutterWindowsView::OnUpdateSemanticsEnabled
virtual void OnUpdateSemanticsEnabled(bool enabled) override
Definition: flutter_windows_view.cc:356
flutter::WindowStateEvent::kHide
@ kHide
flutter::FlutterWindowsView::FlutterWindowsView
FlutterWindowsView(FlutterViewId view_id, FlutterWindowsEngine *engine, std::unique_ptr< WindowBindingHandler > window_binding, std::shared_ptr< WindowsProcTable > windows_proc_table=nullptr)
Definition: flutter_windows_view.cc:104
flutter::FlutterWindowsView::OnComposeCommit
void OnComposeCommit() override
Definition: flutter_windows_view.cc:327
flutter::FlutterWindowsEngine::SendPointerEvent
void SendPointerEvent(const FlutterPointerEvent &event)
Definition: flutter_windows_engine.cc:705
scancode
int scancode
Definition: keyboard_key_handler_unittests.cc:115
flutter::FlutterWindowsView::~FlutterWindowsView
virtual ~FlutterWindowsView()
Definition: flutter_windows_view.cc:121
was_down
bool was_down
Definition: keyboard_key_handler_unittests.cc:119
text_input_plugin.h
flutter::FlutterWindowsView::surface
egl::WindowSurface * surface() const
Definition: flutter_windows_view.cc:778
extended
bool extended
Definition: keyboard_key_handler_unittests.cc:118
flutter::egl::WindowSurface
Definition: window_surface.h:19
flutter::FlutterWindowsEngine::task_runner
TaskRunner * task_runner()
Definition: flutter_windows_engine.h:161
flutter::FlutterWindowsView::OnPointerDown
void OnPointerDown(double x, double y, FlutterPointerDeviceKind device_kind, int32_t device_id, FlutterPointerMouseButtons button) override
Definition: flutter_windows_view.cc:254
flutter::FlutterWindowsEngine
Definition: flutter_windows_engine.h:90
character
char32_t character
Definition: keyboard_key_handler_unittests.cc:117
flutter::FlutterWindowsView::OnComposeChange
void OnComposeChange(const std::u16string &text, int cursor_pos) override
Definition: flutter_windows_view.cc:335
flutter::TaskRunner::PostTask
void PostTask(TaskClosure task)
Definition: task_runner.cc:88
flutter::TextInputPlugin::ComposeEndHook
virtual void ComposeEndHook()
Definition: text_input_plugin.cc:176
flutter::FlutterWindowsView::OnScrollInertiaCancel
void OnScrollInertiaCancel(int32_t device_id) override
Definition: flutter_windows_view.cc:351
flutter::FlutterWindowsView::ForceRedraw
void ForceRedraw()
Definition: flutter_windows_view.cc:187
flutter::FlutterWindowsView::OnPointerPanZoomStart
virtual void OnPointerPanZoomStart(int32_t device_id) override
Definition: flutter_windows_view.cc:287
flutter::FlutterWindowsView::Focus
virtual bool Focus()
Definition: flutter_windows_view.cc:844
flutter::TextInputPlugin::ComposeCommitHook
virtual void ComposeCommitHook()
Definition: text_input_plugin.cc:141
flutter::FlutterWindowsView::IsImplicitView
bool IsImplicitView() const
Definition: flutter_windows_view.cc:721
flutter::Rect
Definition: geometry.h:56
flutter::PhysicalWindowBounds::width
size_t width
Definition: window_binding_handler.h:28
flutter::PhysicalWindowBounds
Definition: window_binding_handler.h:27
flutter::PointerLocation
Definition: window_binding_handler.h:34
flutter::FlutterWindowsView::AnnounceAlert
void AnnounceAlert(const std::wstring &text)
Definition: flutter_windows_view.cc:794
flutter::FlutterWindowsView::GetWindowHandle
virtual HWND GetWindowHandle() const
Definition: flutter_windows_view.cc:786
flutter::FlutterWindowsView::OnPointerPanZoomUpdate
virtual void OnPointerPanZoomUpdate(int32_t device_id, double pan_x, double pan_y, double scale, double rotation) override
Definition: flutter_windows_view.cc:292
flutter::FlutterWindowsView::OnWindowRepaint
void OnWindowRepaint() override
Definition: flutter_windows_view.cc:241
flutter::WindowBindingHandlerDelegate::KeyEventCallback
std::function< void(bool)> KeyEventCallback
Definition: window_binding_handler_delegate.h:20
flutter::FlutterWindowsView::OnComposeEnd
void OnComposeEnd() override
Definition: flutter_windows_view.cc:331
flutter::PointerLocation::y
size_t y
Definition: window_binding_handler.h:36
flutter_windows_view.h
text
std::u16string text
Definition: keyboard_unittests.cc:332
flutter::FlutterWindowsEngine::SendWindowMetricsEvent
void SendWindowMetricsEvent(const FlutterWindowMetricsEvent &event)
Definition: flutter_windows_engine.cc:698
flutter::TextInputPlugin::TextHook
virtual void TextHook(const std::u16string &text)
Definition: text_input_plugin.cc:68
flutter::FlutterWindowsView::view_id
FlutterViewId view_id() const
Definition: flutter_windows_view.cc:717
flutter::KeyboardHandlerBase::SyncModifiersIfNeeded
virtual void SyncModifiersIfNeeded(int modifiers_state)=0
flutter::FlutterWindowsEngine::SendViewFocusEvent
void SendViewFocusEvent(const FlutterViewFocusEvent &event)
Definition: flutter_windows_engine.cc:719
flutter::FlutterWindowsView::OnHighContrastChanged
void OnHighContrastChanged() override
Definition: flutter_windows_view.cc:782
flutter::FlutterWindowsView::OnWindowSizeChanged
bool OnWindowSizeChanged(size_t width, size_t height) override
Definition: flutter_windows_view.cc:195
flutter::FlutterViewId
int64_t FlutterViewId
Definition: flutter_view.h:13
flutter::FlutterWindowsView::OnFrameGenerated
bool OnFrameGenerated(size_t width, size_t height)
Definition: flutter_windows_view.cc:153
flutter
Definition: accessibility_bridge_windows.cc:11
flutter::FlutterWindowsView::UpdateFlutterCursor
void UpdateFlutterCursor(const std::string &cursor_name)
Definition: flutter_windows_view.cc:179
flutter::FlutterWindowsView::SetFlutterCursor
void SetFlutterCursor(HCURSOR cursor)
Definition: flutter_windows_view.cc:183
flutter::FlutterWindowsView::OnPointerLeave
void OnPointerLeave(double x, double y, FlutterPointerDeviceKind device_kind, int32_t device_id=0) override
Definition: flutter_windows_view.cc:280
flutter::FlutterWindowsView::OnText
void OnText(const std::u16string &) override
Definition: flutter_windows_view.cc:304
flutter::FlutterWindowsEngine::keyboard_key_handler
KeyboardHandlerBase * keyboard_key_handler()
Definition: flutter_windows_engine.h:191
flutter::FlutterWindowsView::PresentSoftwareBitmap
virtual bool PresentSoftwareBitmap(const void *allocation, size_t row_bytes, size_t height)
Definition: flutter_windows_view.cc:710
flutter::FlutterWindowsView::OnCursorRectUpdated
virtual void OnCursorRectUpdated(const Rect &rect)
Definition: flutter_windows_view.cc:368
flutter::FlutterWindowsView::AlertNode
ui::AXPlatformNodeWin * AlertNode() const
Definition: flutter_windows_view.cc:815
flutter::FlutterWindowsView::GetEngine
FlutterWindowsEngine * GetEngine() const
Definition: flutter_windows_view.cc:790
flutter::FlutterWindowsEngine::text_input_plugin
TextInputPlugin * text_input_plugin()
Definition: flutter_windows_engine.h:194
flutter::FlutterWindowsView::OnPointerPanZoomEnd
virtual void OnPointerPanZoomEnd(int32_t device_id) override
Definition: flutter_windows_view.cc:300
flutter::FlutterWindowsView::OnDwmCompositionChanged
void OnDwmCompositionChanged()
Definition: flutter_windows_view.cc:836
flutter::FlutterWindowsView::NotifyWinEventWrapper
virtual void NotifyWinEventWrapper(ui::AXPlatformNodeWin *node, ax::mojom::Event event)
Definition: flutter_windows_view.cc:804
flutter::FlutterWindowsEngine::ScheduleFrame
void ScheduleFrame()
Definition: flutter_windows_engine.cc:785
flutter::TaskRunner::PollOnce
void PollOnce(std::chrono::milliseconds timeout)
Definition: task_runner.cc:95
flutter::FlutterWindowsEngine::UpdateHighContrastMode
void UpdateHighContrastMode()
Definition: flutter_windows_engine.cc:947
flutter::FlutterWindowsView::OnEmptyFrameGenerated
bool OnEmptyFrameGenerated()
Definition: flutter_windows_view.cc:131
flutter::FlutterWindowsView::ClearSoftwareBitmap
virtual bool ClearSoftwareBitmap()
Definition: flutter_windows_view.cc:706
flutter::PhysicalWindowBounds::height
size_t height
Definition: window_binding_handler.h:29
flutter::FlutterWindowsView::UpdateSemanticsEnabled
virtual void UpdateSemanticsEnabled(bool enabled)
Definition: flutter_windows_view.cc:824
flutter::FlutterWindowsView::OnComposeBegin
void OnComposeBegin() override
Definition: flutter_windows_view.cc:323
flutter::FlutterWindowsView::SendInitialBounds
void SendInitialBounds()
Definition: flutter_windows_view.cc:403
flutter::FlutterWindowsView::CreateAccessibilityBridge
virtual std::shared_ptr< AccessibilityBridgeWindows > CreateAccessibilityBridge()
Definition: flutter_windows_view.cc:820
flutter::FlutterWindowsView::OnFocus
void OnFocus(FlutterViewFocusState focus_state, FlutterViewFocusDirection direction) override
Definition: flutter_windows_view.cc:318
action
int action
Definition: keyboard_key_handler_unittests.cc:116
flutter::FlutterWindowsView::OnScroll
void OnScroll(double x, double y, double delta_x, double delta_y, int scroll_offset_multiplier, FlutterPointerDeviceKind device_kind, int32_t device_id) override
Definition: flutter_windows_view.cc:340
flutter::FlutterWindowsEngine::UpdateSemanticsEnabled
void UpdateSemanticsEnabled(bool enabled)
Definition: flutter_windows_engine.cc:912
flutter::PointerLocation::x
size_t x
Definition: window_binding_handler.h:35
flutter::FlutterWindowsView::GetAxFragmentRootDelegate
virtual ui::AXFragmentRootDelegateWin * GetAxFragmentRootDelegate() override
Definition: flutter_windows_view.cc:811
flutter::FlutterWindowsEngine::egl_manager
egl::Manager * egl_manager() const
Definition: flutter_windows_engine.h:171
key
int key
Definition: keyboard_key_handler_unittests.cc:114
accessibility_bridge.h
flutter::FlutterWindowsView::CreateWindowMetricsEvent
FlutterWindowMetricsEvent CreateWindowMetricsEvent() const
Definition: flutter_windows_view.cc:389
flutter::FlutterWindowsEngine::OnWindowStateEvent
void OnWindowStateEvent(HWND hwnd, WindowStateEvent event)
Definition: flutter_windows_engine.cc:988
keyboard_key_channel_handler.h
flutter::FlutterWindowsView::OnKey
void OnKey(int key, int scancode, int action, char32_t character, bool extended, bool was_down, KeyEventCallback callback) override
Definition: flutter_windows_view.cc:308
flutter::KeyboardHandlerBase::KeyboardHook
virtual void KeyboardHook(int key, int scancode, int action, char32_t character, bool extended, bool was_down, KeyEventCallback callback)=0
flutter::FlutterWindowsView::GetNativeViewAccessible
virtual gfx::NativeViewAccessible GetNativeViewAccessible() override
Definition: flutter_windows_view.cc:360
flutter::FlutterWindowsView::OnResetImeComposing
virtual void OnResetImeComposing()
Definition: flutter_windows_view.cc:372
flutter::egl::Manager::CreateWindowSurface
virtual std::unique_ptr< WindowSurface > CreateWindowSurface(HWND hwnd, size_t width, size_t height)
Definition: manager.cc:276
flutter::FlutterWindowsView::OnFramePresented
virtual void OnFramePresented()
Definition: flutter_windows_view.cc:673
callback
FlutterDesktopBinaryReply callback
Definition: flutter_windows_view_unittests.cc:52