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"
23 constexpr std::chrono::milliseconds kWindowResizeTimeout{100};
31 bool SurfaceWillUpdate(
size_t cur_width,
34 size_t target_height) {
37 bool non_zero_target_dims = target_height > 0 && target_width > 0;
39 (cur_height != target_height) || (cur_width != target_width);
40 return non_zero_target_dims && not_same_size;
45 void UpdateVsync(
const FlutterWindowsEngine& engine,
46 egl::WindowSurface* surface,
48 egl::Manager* egl_manager = engine.egl_manager();
53 auto update_vsync = [egl_manager, surface, needs_vsync]() {
54 if (!surface || !surface->IsValid()) {
58 if (!surface->MakeCurrent()) {
59 FML_LOG(ERROR) <<
"Unable to make the render surface current to update "
64 if (!surface->SetVSyncEnabled(needs_vsync)) {
65 FML_LOG(ERROR) <<
"Unable to update the render surface's swap interval";
68 if (!egl_manager->render_context()->ClearCurrent()) {
69 FML_LOG(ERROR) <<
"Unable to clear current surface after updating "
79 if (engine.running()) {
80 engine.PostRasterThreadTask(update_vsync);
87 void DestroyWindowSurface(
const FlutterWindowsEngine& engine,
88 std::unique_ptr<egl::WindowSurface> surface) {
92 if (engine.running()) {
93 engine.PostRasterThreadTask(fml::MakeCopyable(
94 [surface = std::move(surface)] { surface->Destroy(); }));
107 std::unique_ptr<WindowBindingHandler> window_binding,
108 std::shared_ptr<WindowsProcTable> windows_proc_table)
111 windows_proc_table_(std::move(windows_proc_table)) {
112 if (windows_proc_table_ ==
nullptr) {
113 windows_proc_table_ = std::make_shared<WindowsProcTable>();
117 binding_handler_ = std::move(window_binding);
118 binding_handler_->SetView(
this);
127 DestroyWindowSurface(*engine_, std::move(surface_));
133 std::unique_lock<std::mutex> lock(resize_mutex_);
135 if (surface_ ==
nullptr || !surface_->IsValid()) {
139 if (resize_status_ != ResizeState::kResizeStarted) {
143 if (!ResizeRenderSurface(resize_target_height_, resize_target_width_)) {
149 resize_status_ = ResizeState::kFrameGenerated;
155 std::unique_lock<std::mutex> lock(resize_mutex_);
157 if (surface_ ==
nullptr || !surface_->IsValid()) {
161 if (resize_status_ != ResizeState::kResizeStarted) {
165 if (resize_target_width_ != width || resize_target_height_ != height) {
169 if (!ResizeRenderSurface(resize_target_width_, resize_target_height_)) {
175 resize_status_ = ResizeState::kFrameGenerated;
180 binding_handler_->UpdateFlutterCursor(cursor_name);
184 binding_handler_->SetFlutterCursor(cursor);
188 if (resize_status_ == ResizeState::kDone) {
197 SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
201 if (!surface_ || !surface_->IsValid()) {
202 SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
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());
216 std::unique_lock<std::mutex> lock(resize_mutex_);
217 resize_status_ = ResizeState::kResizeStarted;
218 resize_target_width_ = width;
219 resize_target_height_ = height;
222 SendWindowMetrics(width, height, binding_handler_->GetDpiScale());
224 std::chrono::time_point<std::chrono::steady_clock> start_time =
225 std::chrono::steady_clock::now();
228 if (std::chrono::steady_clock::now() > start_time + kWindowResizeTimeout) {
231 std::unique_lock<std::mutex> lock(resize_mutex_);
232 if (resize_status_ == ResizeState::kDone) {
247 FlutterPointerDeviceKind device_kind,
249 int modifiers_state) {
251 SendPointerMove(x, y, GetOrCreatePointerState(device_kind, device_id));
257 FlutterPointerDeviceKind device_kind,
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);
270 FlutterPointerDeviceKind device_kind,
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);
282 FlutterPointerDeviceKind device_kind,
284 SendPointerLeave(x, y, GetOrCreatePointerState(device_kind, device_id));
289 SendPointerPanZoomStart(device_id, point.
x, point.
y);
297 SendPointerPanZoomUpdate(device_id, pan_x, pan_y, scale, rotation);
301 SendPointerPanZoomEnd(device_id);
319 FlutterViewFocusDirection direction) {
320 SendFocus(focus_state, direction);
337 SendComposeChange(
text, cursor_pos);
344 int scroll_offset_multiplier,
345 FlutterPointerDeviceKind device_kind,
347 SendScroll(x, y, delta_x, delta_y, scroll_offset_multiplier, device_kind,
353 SendScrollInertiaCancel(device_id, point.
x, point.
y);
361 if (!accessibility_bridge_) {
365 return accessibility_bridge_->GetChildOfAXFragmentRoot();
369 binding_handler_->OnCursorRectUpdated(rect);
373 binding_handler_->OnResetImeComposing();
377 void FlutterWindowsView::SendWindowMetrics(
size_t width,
379 double pixel_ratio)
const {
380 FlutterWindowMetricsEvent
event = {};
381 event.struct_size =
sizeof(event);
383 event.height = height;
384 event.pixel_ratio = pixel_ratio;
385 event.view_id = view_id_;
391 double pixel_ratio = binding_handler_->GetDpiScale();
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_;
413 FlutterWindowsView::PointerState* FlutterWindowsView::GetOrCreatePointerState(
414 FlutterPointerDeviceKind device_kind,
419 int32_t pointer_id = (
static_cast<int32_t
>(device_kind) << 28) | device_id;
421 auto [it, added] = pointer_states_.try_emplace(pointer_id,
nullptr);
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);
429 return it->second.get();
434 void FlutterWindowsView::SetEventPhaseFromCursorButtonState(
435 FlutterPointerEvent* event_data,
436 const PointerState* state)
const {
439 if (state->buttons == 0) {
440 event_data->phase = state->flutter_state_is_down
441 ? FlutterPointerPhase::kUp
442 : FlutterPointerPhase::kHover;
444 event_data->phase = state->flutter_state_is_down
445 ? FlutterPointerPhase::kMove
446 : FlutterPointerPhase::kDown;
450 void FlutterWindowsView::SendPointerMove(
double x,
452 PointerState* state) {
453 FlutterPointerEvent
event = {};
457 SetEventPhaseFromCursorButtonState(&event, state);
458 SendPointerEventWithData(event, state);
461 void FlutterWindowsView::SendPointerDown(
double x,
463 PointerState* state) {
464 FlutterPointerEvent
event = {};
468 SetEventPhaseFromCursorButtonState(&event, state);
469 SendPointerEventWithData(event, state);
471 state->flutter_state_is_down =
true;
474 void FlutterWindowsView::SendPointerUp(
double x,
476 PointerState* state) {
477 FlutterPointerEvent
event = {};
481 SetEventPhaseFromCursorButtonState(&event, state);
482 SendPointerEventWithData(event, state);
483 if (event.phase == FlutterPointerPhase::kUp) {
484 state->flutter_state_is_down =
false;
488 void FlutterWindowsView::SendPointerLeave(
double x,
490 PointerState* state) {
491 FlutterPointerEvent
event = {};
494 event.phase = FlutterPointerPhase::kRemove;
495 SendPointerEventWithData(event, state);
498 void FlutterWindowsView::SendPointerPanZoomStart(int32_t device_id,
502 GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
503 state->pan_zoom_start_x = x;
504 state->pan_zoom_start_y = y;
505 FlutterPointerEvent
event = {};
508 event.phase = FlutterPointerPhase::kPanZoomStart;
509 SendPointerEventWithData(event, state);
512 void FlutterWindowsView::SendPointerPanZoomUpdate(int32_t device_id,
518 GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
519 FlutterPointerEvent
event = {};
520 event.x = state->pan_zoom_start_x;
521 event.y = state->pan_zoom_start_y;
525 event.rotation = rotation;
526 event.phase = FlutterPointerPhase::kPanZoomUpdate;
527 SendPointerEventWithData(event, state);
530 void FlutterWindowsView::SendPointerPanZoomEnd(int32_t device_id) {
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);
540 void FlutterWindowsView::SendText(
const std::u16string&
text) {
544 void FlutterWindowsView::SendKey(
int key,
556 engine->text_input_plugin()->KeyboardHook(
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;
575 void FlutterWindowsView::SendComposeBegin() {
579 void FlutterWindowsView::SendComposeCommit() {
583 void FlutterWindowsView::SendComposeEnd() {
587 void FlutterWindowsView::SendComposeChange(
const std::u16string&
text,
592 void FlutterWindowsView::SendScroll(
double x,
596 int scroll_offset_multiplier,
597 FlutterPointerDeviceKind device_kind,
599 auto state = GetOrCreatePointerState(device_kind, device_id);
601 FlutterPointerEvent
event = {};
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);
611 void FlutterWindowsView::SendScrollInertiaCancel(int32_t device_id,
615 GetOrCreatePointerState(kFlutterPointerDeviceKindTrackpad, device_id);
617 FlutterPointerEvent
event = {};
621 FlutterPointerSignalKind::kFlutterPointerSignalKindScrollInertiaCancel;
622 SetEventPhaseFromCursorButtonState(&event, state);
623 SendPointerEventWithData(event, state);
626 void FlutterWindowsView::SendPointerEventWithData(
627 const FlutterPointerEvent& event_data,
628 PointerState* state) {
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;
638 SendPointerEventWithData(event, state);
643 if (state->flutter_state_is_added &&
644 event_data.phase == FlutterPointerPhase::kAdd) {
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_;
655 event.struct_size =
sizeof(event);
657 std::chrono::duration_cast<std::chrono::microseconds>(
658 std::chrono::high_resolution_clock::now().time_since_epoch())
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);
675 std::unique_lock<std::mutex> lock(resize_mutex_);
677 switch (resize_status_) {
678 case ResizeState::kResizeStarted:
689 case ResizeState::kFrameGenerated: {
691 resize_status_ = ResizeState::kDone;
699 windows_proc_table_->DwmFlush();
701 case ResizeState::kDone:
707 return binding_handler_->OnBitmapSurfaceCleared();
713 return binding_handler_->OnBitmapSurfaceUpdated(allocation, row_bytes,
726 FML_DCHECK(surface_ ==
nullptr);
733 UpdateVsync(*engine_, surface_.get(), NeedsVsync());
735 resize_target_width_ = bounds.
width;
736 resize_target_height_ = bounds.
height;
740 bool FlutterWindowsView::ResizeRenderSurface(
size_t width,
size_t height) {
741 FML_DCHECK(surface_ !=
nullptr);
744 if (width == surface_->width() && height == surface_->height()) {
748 auto const existing_vsync = surface_->vsync_enabled();
753 if (!surface_->Destroy()) {
754 FML_LOG(ERROR) <<
"View resize failed to destroy surface";
758 std::unique_ptr<egl::WindowSurface> resized_surface =
761 if (!resized_surface) {
762 FML_LOG(ERROR) <<
"View resize failed to create surface";
766 if (!resized_surface->MakeCurrent() ||
767 !resized_surface->SetVSyncEnabled(existing_vsync)) {
771 FML_LOG(ERROR) <<
"View resize failed to set vsync";
774 surface_ = std::move(resized_surface);
779 return surface_.get();
787 return binding_handler_->GetWindowHandle();
795 auto alert_delegate = binding_handler_->GetAlertDelegate();
796 if (!alert_delegate) {
799 alert_delegate->SetText(fml::WideStringToUtf16(
text));
800 ui::AXPlatformNodeWin* alert_node = binding_handler_->GetAlert();
805 ax::mojom::Event event) {
807 node->NotifyAccessibilityEvent(event);
812 return accessibility_bridge_.get();
816 return binding_handler_->GetAlert();
819 std::shared_ptr<AccessibilityBridgeWindows>
821 return std::make_shared<AccessibilityBridgeWindows>(
this);
825 if (semantics_enabled_ != enabled) {
826 semantics_enabled_ = enabled;
828 if (!semantics_enabled_ && accessibility_bridge_) {
829 accessibility_bridge_.reset();
830 }
else if (semantics_enabled_ && !accessibility_bridge_) {
837 UpdateVsync(*engine_, surface_.get(), NeedsVsync());
845 return binding_handler_->Focus();
848 bool FlutterWindowsView::NeedsVsync()
const {
852 return !windows_proc_table_->DwmIsCompositionEnabled();